Annotation of embedaddon/libpdel/structs/structs_xml.c, revision 1.1.1.1

1.1       misho       1: 
                      2: /*
                      3:  * Copyright (c) 2001-2002 Packet Design, LLC.
                      4:  * All rights reserved.
                      5:  * 
                      6:  * Subject to the following obligations and disclaimer of warranty,
                      7:  * use and redistribution of this software, in source or object code
                      8:  * forms, with or without modifications are expressly permitted by
                      9:  * Packet Design; provided, however, that:
                     10:  * 
                     11:  *    (i)  Any and all reproductions of the source or object code
                     12:  *         must include the copyright notice above and the following
                     13:  *         disclaimer of warranties; and
                     14:  *    (ii) No rights are granted, in any manner or form, to use
                     15:  *         Packet Design trademarks, including the mark "PACKET DESIGN"
                     16:  *         on advertising, endorsements, or otherwise except as such
                     17:  *         appears in the above copyright notice or in the software.
                     18:  * 
                     19:  * THIS SOFTWARE IS BEING PROVIDED BY PACKET DESIGN "AS IS", AND
                     20:  * TO THE MAXIMUM EXTENT PERMITTED BY LAW, PACKET DESIGN MAKES NO
                     21:  * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING
                     22:  * THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED
                     23:  * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE,
                     24:  * OR NON-INFRINGEMENT.  PACKET DESIGN DOES NOT WARRANT, GUARANTEE,
                     25:  * OR MAKE ANY REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS
                     26:  * OF THE USE OF THIS SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY,
                     27:  * RELIABILITY OR OTHERWISE.  IN NO EVENT SHALL PACKET DESIGN BE
                     28:  * LIABLE FOR ANY DAMAGES RESULTING FROM OR ARISING OUT OF ANY USE
                     29:  * OF THIS SOFTWARE, INCLUDING WITHOUT LIMITATION, ANY DIRECT,
                     30:  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, PUNITIVE, OR CONSEQUENTIAL
                     31:  * DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, LOSS OF
                     32:  * USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY THEORY OF
                     33:  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
                     34:  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
                     35:  * THE USE OF THIS SOFTWARE, EVEN IF PACKET DESIGN IS ADVISED OF
                     36:  * THE POSSIBILITY OF SUCH DAMAGE.
                     37:  *
                     38:  * Author: Archie Cobbs <archie@freebsd.org>
                     39:  */
                     40: 
                     41: #include <sys/types.h>
                     42: #include <net/ethernet.h>
                     43: #include <netinet/in.h>
                     44: 
                     45: #include <stdlib.h>
                     46: #include <stdio.h>
                     47: #include <stdarg.h>
                     48: #include <assert.h>
                     49: #include <ctype.h>
                     50: #include <syslog.h>
                     51: #include <string.h>
                     52: #include <errno.h>
                     53: #include <pthread.h>
                     54: 
                     55: #include <expat.h>
                     56: 
                     57: #include "structs/structs.h"
                     58: #include "structs/xml.h"
                     59: #include "structs/type/array.h"
                     60: #include "structs/type/struct.h"
                     61: #include "structs/type/union.h"
                     62: #include "util/typed_mem.h"
                     63: #include "sys/alog.h"
                     64: 
                     65: /* Standard XML header */
                     66: #define XML_HEADER             "<?xml version=\"1.0\" standalone=\"yes\"?>\n"
                     67: 
                     68: /* Max parse depth */
                     69: #define MAX_XML_STACK          128
                     70: 
                     71: #define INPUT_MEM_TYPE         "structs_xml_input"
                     72: #define INFO_MEM_TYPE          "structs_xml_input.info"
                     73: #define CHARDATA_MEM_TYPE      "structs_xml_input.chardata"
                     74: #define OUTPUT_MEM_TYPE                "structs_xml_output"
                     75: 
                     76: /* Parse private context */
                     77: struct xmlinput_stackframe {
                     78:        const struct structs_type       *type;          /* type we're parsing */
                     79:        void                            *data;          /* data pointer */
                     80:        char                            *s;             /* character data */
                     81:        u_int                           s_len;          /* strlen(s) */
                     82:        u_int                           index;          /* fixed array index */
                     83:        u_char                          combined;       /* was a combined tag */
                     84: };
                     85: 
                     86: struct xml_input_info {
                     87:        XML_Parser                      p;
                     88:        int                             error;
                     89:        int                             depth;
                     90:        int                             flags;
                     91:        u_int                           skip;
                     92:        const char                      *elem_tag;
                     93:        char                            **attrp;
                     94:        const char                      *attr_mtype;
                     95:        int                             attr_len;
                     96:        struct xmlinput_stackframe      stack[MAX_XML_STACK];
                     97:        structs_xmllog_t                *logger;
                     98: };
                     99: 
                    100: /*
                    101:  * Internal functions
                    102:  */
                    103: static void    *structs_xml_malloc(size_t size);
                    104: static void    *structs_xml_realloc(void *ptr, size_t size);
                    105: static void    structs_xml_free(void *ptr);
                    106: 
                    107: static void    structs_xml_output_cleanup(void *arg);
                    108: static void    structs_xml_output_prefix(FILE *fp, int depth);
                    109: static void    structs_xml_encode(FILE *fp, const char *s);
                    110: 
                    111: static structs_xmllog_t        structs_xml_null_logger;
                    112: static structs_xmllog_t        structs_xml_stderr_logger;
                    113: static structs_xmllog_t        structs_xml_alog_logger;
                    114: 
                    115: /*
                    116:  * Internal variables
                    117:  */
                    118: static const XML_Memory_Handling_Suite memsuite = {
                    119:        structs_xml_malloc,
                    120:        structs_xml_realloc,
                    121:        structs_xml_free
                    122: };
                    123: 
                    124: static const   char separator_string[] = { STRUCTS_SEPARATOR, '\0' };
                    125: static const   char double_separator_string[] = {
                    126:        STRUCTS_SEPARATOR, STRUCTS_SEPARATOR, '\0'
                    127: };
                    128: 
                    129: /*********************************************************************
                    130:                        XML INPUT ROUTINES
                    131: *********************************************************************/
                    132: 
                    133: /*
                    134:  * Internal functions
                    135:  */
                    136: static void    structs_xml_input_cleanup(void *arg);
                    137: static void    structs_xml_input_start(void *userData,
                    138:                        const XML_Char *name, const XML_Char **atts);
                    139: static void    structs_xml_input_end(void *userData, const XML_Char *name);
                    140: static void    structs_xml_input_chardata(void *userData,
                    141:                        const XML_Char *s, int len);
                    142: static void    structs_xml_input_nest(struct xml_input_info *info,
                    143:                        const char *name, const struct structs_type **typep,
                    144:                        void **datap);
                    145: static void    structs_xml_input_prep(struct xml_input_info *info,
                    146:                        const struct structs_type *type, void *data,
                    147:                        int combined);
                    148: static void    structs_xml_unnest(struct xml_input_info *info,
                    149:                        const XML_Char *name);
                    150: static void    structs_xml_pop(struct xml_input_info *info);
                    151: 
                    152: /* Context for one XML parse run */
                    153: struct structs_xml_input_ctx {
                    154:        struct xml_input_info           *info;
                    155:        XML_Parser                      p;
                    156:        int                             data_init;
                    157:        int                             rtn;
                    158:        const struct structs_type       *type;
                    159:        void                            *data;
                    160:        char                            **attrp;
                    161:        const char                      *attr_mtype;
                    162: };
                    163: 
                    164: /*
                    165:  * Input a type from XML.
                    166:  *
                    167:  * Note: it is safe for the calling thread to be canceled.
                    168:  */
                    169: int
                    170: structs_xml_input(const struct structs_type *type,
                    171:        const char *elem_tag, char **attrp, const char *attr_mtype,
                    172:        FILE *fp, void *data, int flags, structs_xmllog_t *logger)
                    173: {
                    174:        struct structs_xml_input_ctx *ctx;
                    175:        int esave;
                    176:        int r;
                    177: 
                    178:        /* Special cases for logger */
                    179:        if (logger == STRUCTS_LOGGER_NONE)
                    180:                logger = structs_xml_null_logger;
                    181:        else if (logger == STRUCTS_LOGGER_STDERR)
                    182:                logger = structs_xml_stderr_logger;
                    183:        else if (logger == STRUCTS_LOGGER_ALOG)
                    184:                logger = structs_xml_alog_logger;
                    185: 
                    186:        /* Create context */
                    187:        if ((ctx = MALLOC(INPUT_MEM_TYPE, sizeof(*ctx))) == NULL)
                    188:                return (-1);
                    189:        memset(ctx, 0, sizeof(*ctx));
                    190:        ctx->attrp = attrp;
                    191:        ctx->attr_mtype = attr_mtype;
                    192:        ctx->type = type;
                    193:        ctx->data = data;
                    194:        ctx->rtn = -1;
                    195:        pthread_cleanup_push(structs_xml_input_cleanup, ctx);
                    196: 
                    197:        /* Initialize attributes */
                    198:        if (ctx->attrp != NULL
                    199:            && (*ctx->attrp = STRDUP(ctx->attr_mtype, "")) == NULL)
                    200:                goto done;
                    201: 
                    202:        /* Initialize data object if desired */
                    203:        if (ctx->type != NULL && (flags & STRUCTS_XML_UNINIT) != 0) {
                    204:                if ((*type->init)(ctx->type, ctx->data) == -1) {
                    205:                        esave = errno;
                    206:                        (*logger)(LOG_ERR, "error initializing data: %s",
                    207:                            strerror(errno));
                    208:                        errno = esave;
                    209:                        goto done;
                    210:                }
                    211:                ctx->data_init = 1;
                    212:        }
                    213: 
                    214:        /* Allocate info structure */
                    215:        if ((ctx->info = MALLOC(INFO_MEM_TYPE, sizeof(*ctx->info))) == NULL) {
                    216:                esave = errno;
                    217:                (*logger)(LOG_ERR, "%s: %s", "malloc", strerror(errno));
                    218:                errno = esave;
                    219:                goto done;
                    220:        }
                    221:        memset(ctx->info, 0, sizeof(*ctx->info));
                    222:        ctx->info->logger = logger;
                    223:        ctx->info->attrp = attrp;
                    224:        ctx->info->attr_len = 0;
                    225:        ctx->info->attr_mtype = attr_mtype;
                    226:        ctx->info->elem_tag = elem_tag;
                    227:        ctx->info->stack[0].type = type;
                    228:        ctx->info->stack[0].data = data;
                    229:        ctx->info->flags = flags;
                    230: 
                    231:        /* Create a new parser */
                    232:        if ((ctx->p = XML_ParserCreate_MM(NULL, &memsuite, NULL)) == NULL) {
                    233:                esave = errno;
                    234:                (*logger)(LOG_ERR,
                    235:                    "error creating XML parser: %s", strerror(errno));
                    236:                errno = esave;
                    237:                goto done;
                    238:        }
                    239:        ctx->info->p = ctx->p;
                    240:        XML_SetUserData(ctx->p, ctx->info);
                    241:        XML_SetElementHandler(ctx->p,
                    242:            structs_xml_input_start, structs_xml_input_end);
                    243:        XML_SetCharacterDataHandler(ctx->p, structs_xml_input_chardata);
                    244: 
                    245:        /* Parse it */
                    246:        while (1) {
                    247:                const int bufsize = 1024;
                    248:                size_t len;
                    249:                void *buf;
                    250: 
                    251:                /* Get buffer */
                    252:                if ((buf = XML_GetBuffer(ctx->p, bufsize)) == NULL) {
                    253:                        esave = errno;
                    254:                        (*logger)(LOG_ERR,
                    255:                            "error from XML_GetBuffer: %s", strerror(errno));
                    256:                        errno = esave;
                    257:                        break;
                    258:                }
                    259: 
                    260:                /* Read more bytes. Note: we could get canceled here. */
                    261:                len = fread(buf, 1, bufsize, fp);
                    262: 
                    263:                /* Check for error */
                    264:                if (ferror(fp)) {
                    265:                        esave = errno;
                    266:                        (*logger)(LOG_ERR, "read error: %s", strerror(errno));
                    267:                        errno = esave;
                    268:                        break;
                    269:                }
                    270: 
                    271:                /* Process them */
                    272:                if (len > 0 && !XML_ParseBuffer(ctx->p, len, feof(fp))) {
                    273:                        (*logger)(LOG_ERR, "line %d:%d: %s",
                    274:                            XML_GetCurrentLineNumber(ctx->p),
                    275:                            XML_GetCurrentColumnNumber(ctx->p),
                    276:                            XML_ErrorString(XML_GetErrorCode(ctx->p)));
                    277:                        errno = EINVAL;
                    278:                        break;
                    279:                }
                    280: 
                    281:                /* Done? */
                    282:                if (feof(fp)) {
                    283:                        if (ctx->info->error != 0)
                    284:                                errno = ctx->info->error;
                    285:                        else
                    286:                                ctx->rtn = 0;
                    287:                        break;
                    288:                }
                    289:        }
                    290: 
                    291: done:
                    292:        /* Clean up and exit */
                    293:        r = ctx->rtn;
                    294:        pthread_cleanup_pop(1);
                    295:        return (r);
                    296: }
                    297: 
                    298: /*
                    299:  * Cleanup for structs_xml_input()
                    300:  */
                    301: static void
                    302: structs_xml_input_cleanup(void *arg)
                    303: {
                    304:        struct structs_xml_input_ctx *const ctx = arg;
                    305:        const int esave = errno;
                    306: 
                    307:        /* Free private parse info */
                    308:        if (ctx->info != NULL) {
                    309:                while (ctx->info->depth >= 0)
                    310:                        structs_xml_pop(ctx->info);
                    311:                FREE(INFO_MEM_TYPE, ctx->info);
                    312:        }
                    313: 
                    314:        /* Free parser */
                    315:        if (ctx->p != NULL)
                    316:                XML_ParserFree(ctx->p);
                    317: 
                    318:        /* If error, free returned attributes and initialized data */
                    319:        if (ctx->rtn != 0) {
                    320:                if (ctx->attrp != NULL && *ctx->attrp != NULL) {
                    321:                        FREE(ctx->attr_mtype, *ctx->attrp);
                    322:                        *ctx->attrp = NULL;
                    323:                }
                    324:                if (ctx->data_init)
                    325:                        (*ctx->type->uninit)(ctx->type, ctx->data);
                    326:        }
                    327: 
                    328:        /* Free context */
                    329:        FREE(INPUT_MEM_TYPE, ctx);
                    330:        errno = esave;
                    331: }
                    332: 
                    333: /*
                    334:  * Start tag handler
                    335:  */
                    336: static void
                    337: structs_xml_input_start(void *userData,
                    338:        const XML_Char *name, const XML_Char **attrs)
                    339: {
                    340:        struct xml_input_info *const info = userData;
                    341:        struct xmlinput_stackframe *const frame = &info->stack[info->depth];
                    342:        const struct structs_type *type = frame->type;
                    343:        void *data = frame->data;
                    344:        char *namebuf;
                    345:        char *ctx;
                    346:        int first;
                    347:        char *s;
                    348:        int sev;
                    349: 
                    350:        /* Skip if any errors */
                    351:        if (info->error)
                    352:                return;
                    353:        if (info->skip) {
                    354:                info->skip++;
                    355:                return;
                    356:        }
                    357: 
                    358:        /* Handle the top level tag specially */
                    359:        if (info->depth == 0) {
                    360:                int i;
                    361: 
                    362:                /* The top level tag must match what we expect */
                    363:                if (strcmp(name, info->elem_tag) != 0) {
                    364:                        (*info->logger)(LOG_ERR,
                    365:                            "line %d:%d: expecting element \"%s\" here",
                    366:                            XML_GetCurrentLineNumber(info->p),
                    367:                            XML_GetCurrentColumnNumber(info->p),
                    368:                            info->elem_tag);
                    369:                        info->error = EINVAL;
                    370:                        return;
                    371:                }
                    372: 
                    373:                /* Extract attributes */
                    374:                for (i = 0; info->attrp != NULL && attrs[i] != NULL; i += 2) {
                    375:                        const char *const name = attrs[i];
                    376:                        const char *const value = attrs[i + 1];
                    377:                        void *mem;
                    378: 
                    379:                        if ((mem = REALLOC(info->attr_mtype,
                    380:                            *info->attrp, info->attr_len + strlen(name) + 1
                    381:                              + strlen(value) + 1 + 1)) == NULL) {
                    382:                                info->error = errno;
                    383:                                (*info->logger)(LOG_ERR, "line %d:%d: %s: %s",
                    384:                                    XML_GetCurrentLineNumber(info->p),
                    385:                                    XML_GetCurrentColumnNumber(info->p),
                    386:                                    "malloc", strerror(errno));
                    387:                                return;
                    388:                        }
                    389:                        *info->attrp = mem;
                    390:                        strcpy(*info->attrp + info->attr_len, name);
                    391:                        strcpy(*info->attrp + info->attr_len
                    392:                            + strlen(name) + 1, value);
                    393:                        (*info->attrp)[info->attr_len + strlen(name) + 1
                    394:                            + strlen(value) + 1] = '\0';
                    395:                        info->attr_len += strlen(name) + 1 + strlen(value) + 1;
                    396:                }
                    397: 
                    398:                /* Are we only scanning? */
                    399:                if ((info->flags & STRUCTS_XML_SCAN) != 0) {
                    400:                        info->skip++;
                    401:                        return;
                    402:                }
                    403: 
                    404:                /* Prep the top level data structure */
                    405:                structs_xml_input_prep(info, type, data, 0);
                    406:                return;
                    407:        }
                    408: 
                    409:        /* We don't allow attributes with non top level elements */
                    410:        if (attrs[0] != NULL) {
                    411:                sev = (info->flags & STRUCTS_XML_LOOSE) == 0 ?
                    412:                    LOG_ERR : LOG_WARNING;
                    413:                (*info->logger)(sev, "line %d:%d: element \"%s\""
                    414:                    " contains attributes (not allowed)",
                    415:                    XML_GetCurrentLineNumber(info->p),
                    416:                    XML_GetCurrentColumnNumber(info->p), name);
                    417:                if (sev == LOG_ERR) {
                    418:                        info->error = EINVAL;
                    419:                        return;
                    420:                }
                    421:        }
                    422: 
                    423:        /*
                    424:         * Check the case of a structure or union field name
                    425:         * containing the separator character. Such fields take
                    426:         * precedence over combined XML tags.
                    427:         */
                    428:        if (strchr(name, STRUCTS_SEPARATOR) != NULL) {
                    429:                switch (type->tclass) {
                    430:                case STRUCTS_TYPE_STRUCTURE:
                    431:                    {
                    432:                        const struct structs_field *field;
                    433: 
                    434:                        for (field = frame->type->args[0].v;
                    435:                            field->name != NULL; field++) {
                    436:                                if (strcmp(field->name, name) == 0)
                    437:                                        goto not_combined;
                    438:                        }
                    439:                    }
                    440:                case STRUCTS_TYPE_UNION:
                    441:                    {
                    442:                        const struct structs_ufield *field;
                    443: 
                    444:                        for (field = frame->type->args[0].v;
                    445:                            field->name != NULL; field++) {
                    446:                                if (strcmp(field->name, name) == 0)
                    447:                                        goto not_combined;
                    448:                        }
                    449:                    }
                    450:                default:
                    451:                        break;
                    452:                }
                    453:        }
                    454: 
                    455:        /* Check whether we need to consider this as a combined tag */
                    456:        if ((info->flags & STRUCTS_XML_COMB_TAGS) == 0
                    457:            || strchr(name, STRUCTS_SEPARATOR) == NULL)
                    458:                goto not_combined;
                    459: 
                    460:        /* Check that the combined XML tag is well-formed */
                    461:        if (name[0] == STRUCTS_SEPARATOR
                    462:            || (name[0] != '\0' && name[strlen(name) - 1] == STRUCTS_SEPARATOR)
                    463:            || strstr(name, double_separator_string) != NULL) {
                    464:                (*info->logger)(LOG_ERR, "line %d:%d: invalid combined"
                    465:                    " element tag \"%s\"", XML_GetCurrentLineNumber(info->p),
                    466:                    XML_GetCurrentColumnNumber(info->p), name);
                    467:                info->error = EINVAL;
                    468:                return;
                    469:        }
                    470: 
                    471:        /* Copy XML tag so we can parse it */
                    472:        if ((namebuf = STRDUP(TYPED_MEM_TEMP, name)) == NULL) {
                    473:                info->error = errno;
                    474:                return;
                    475:        }
                    476: 
                    477:        /* Parse combined XML tag into individual tag names */
                    478:        for (first = 1, s = strtok_r(namebuf, separator_string, &ctx);
                    479:            s != NULL; first = 0, s = strtok_r(NULL, separator_string, &ctx)) {
                    480:                struct xmlinput_stackframe *const frame
                    481:                    = &info->stack[info->depth];
                    482: 
                    483:                type = frame->type;
                    484:                data = frame->data;
                    485:                structs_xml_input_nest(info, s, &type, &data);
                    486:                if (info->error != 0) {
                    487:                        FREE(TYPED_MEM_TEMP, namebuf);
                    488:                        return;
                    489:                }
                    490:                structs_xml_input_prep(info, type, data, !first);
                    491:                if (info->error != 0) {
                    492:                        FREE(TYPED_MEM_TEMP, namebuf);
                    493:                        return;
                    494:                }
                    495:        }
                    496:        FREE(TYPED_MEM_TEMP, namebuf);
                    497:        return;
                    498: 
                    499: not_combined:
                    500:        /* Handle a non-combined tag */
                    501:        structs_xml_input_nest(info, name, &type, &data);
                    502:        if (info->error != 0)
                    503:                return;
                    504:        structs_xml_input_prep(info, type, data, 0);
                    505:        if (info->error != 0)
                    506:                return;
                    507: }
                    508: 
                    509: /*
                    510:  * Nest one level deeper into the data structure we're inputting
                    511:  * using the sub-field "name".
                    512:  */
                    513: static void
                    514: structs_xml_input_nest(struct xml_input_info *info, const char *name,
                    515:        const struct structs_type **typep, void **datap)
                    516: {
                    517:        struct xmlinput_stackframe *const frame = &info->stack[info->depth];
                    518:        const struct structs_type *type;
                    519:        void *data;
                    520:        int sev;
                    521: 
                    522:        /* Check type type */
                    523:        switch (frame->type->tclass) {
                    524:        case STRUCTS_TYPE_STRUCTURE:
                    525:        case STRUCTS_TYPE_UNION:
                    526:            {
                    527:                /* Find field; for unions, adjust the field type if necessary */
                    528:                type = frame->type;
                    529:                data = frame->data;
                    530:                if ((type = structs_find(type, name, &data, 1)) == NULL) {
                    531:                        if (errno == ENOENT) {
                    532:                                sev = (info->flags & STRUCTS_XML_LOOSE) == 0 ?
                    533:                                    LOG_ERR : LOG_WARNING;
                    534:                                (*info->logger)(sev,
                    535:                                    "line %d:%d: element \"%s\" is not"
                    536:                                    " expected here",
                    537:                                    XML_GetCurrentLineNumber(info->p),
                    538:                                    XML_GetCurrentColumnNumber(info->p), name);
                    539:                                if (sev == LOG_ERR) {
                    540:                                        info->error = EINVAL;
                    541:                                        return;
                    542:                                }
                    543:                                info->skip++;
                    544:                                return;
                    545:                        }
                    546:                        (*info->logger)(LOG_ERR, "line %d:%d: error"
                    547:                            " initializing union field \"%s\": %s",
                    548:                            XML_GetCurrentLineNumber(info->p),
                    549:                            XML_GetCurrentColumnNumber(info->p),
                    550:                            name, strerror(errno));
                    551:                        info->error = errno;
                    552:                        return;
                    553:                }
                    554:                break;
                    555:            }
                    556: 
                    557:        case STRUCTS_TYPE_ARRAY:
                    558:            {
                    559:                const struct structs_type *const etype = frame->type->args[0].v;
                    560:                const char *mtype = frame->type->args[1].s;
                    561:                const char *elem_name = frame->type->args[2].s;
                    562:                struct structs_array *const ary = frame->data;
                    563:                void *mem;
                    564: 
                    565:                /* Check tag name */
                    566:                if (strcmp(name, elem_name) != 0) {
                    567:                        (*info->logger)(LOG_ERR, "line %d:%d: expected element"
                    568:                            " \"%s\" instead of \"%s\"",
                    569:                            XML_GetCurrentLineNumber(info->p),
                    570:                            XML_GetCurrentColumnNumber(info->p),
                    571:                            elem_name, name);
                    572:                        info->error = EINVAL;
                    573:                        return;
                    574:                }
                    575: 
                    576:                /* Expand the array by one */
                    577:                if ((mem = REALLOC(mtype,
                    578:                    ary->elems, (ary->length + 1) * etype->size)) == NULL) {
                    579:                        info->error = errno;
                    580:                        (*info->logger)(LOG_ERR, "line %d:%d: %s: %s",
                    581:                            XML_GetCurrentLineNumber(info->p),
                    582:                            XML_GetCurrentColumnNumber(info->p),
                    583:                            "realloc", strerror(errno));
                    584:                        return;
                    585:                }
                    586:                ary->elems = mem;
                    587: 
                    588:                /* Initialize the new element */
                    589:                memset((char *)ary->elems + (ary->length * etype->size),
                    590:                    0, etype->size);
                    591:                if ((*etype->init)(etype,
                    592:                    (char *)ary->elems + (ary->length * etype->size)) == -1) {
                    593:                        info->error = errno;
                    594:                        (*info->logger)(LOG_ERR, "line %d:%d: %s: %s",
                    595:                            XML_GetCurrentLineNumber(info->p),
                    596:                            XML_GetCurrentColumnNumber(info->p),
                    597:                            "error initializing new array element",
                    598:                            strerror(errno));
                    599:                        return;
                    600:                }
                    601: 
                    602:                /* Parse the element next */
                    603:                type = etype;
                    604:                data = (char *)ary->elems + (ary->length * etype->size);
                    605:                ary->length++;
                    606:                break;
                    607:            }
                    608: 
                    609:        case STRUCTS_TYPE_FIXEDARRAY:
                    610:            {
                    611:                const struct structs_type *const etype = frame->type->args[0].v;
                    612:                const char *elem_name = frame->type->args[1].s;
                    613:                const u_int length = frame->type->args[2].i;
                    614: 
                    615:                /* Check tag name */
                    616:                if (strcmp(name, elem_name) != 0) {
                    617:                        (*info->logger)(LOG_ERR, "line %d:%d: expected element"
                    618:                            " \"%s\" instead of \"%s\"",
                    619:                            XML_GetCurrentLineNumber(info->p),
                    620:                            XML_GetCurrentColumnNumber(info->p),
                    621:                            elem_name, name);
                    622:                        info->error = EINVAL;
                    623:                        return;
                    624:                }
                    625: 
                    626:                /* Check index vs. array length */
                    627:                if (frame->index >= length) {
                    628:                        sev = (info->flags & STRUCTS_XML_LOOSE) == 0 ?
                    629:                            LOG_ERR : LOG_WARNING;
                    630:                        (*info->logger)(sev, "line %d:%d: too many"
                    631:                            " elements in fixed array (length %u)",
                    632:                            XML_GetCurrentLineNumber(info->p),
                    633:                            XML_GetCurrentColumnNumber(info->p), length);
                    634:                        if (sev == LOG_ERR) {
                    635:                                info->error = EINVAL;
                    636:                                return;
                    637:                        }
                    638:                        info->skip++;
                    639:                        return;
                    640:                }
                    641: 
                    642:                /* Parse the element next */
                    643:                type = etype;
                    644:                data = (char *)frame->data + (frame->index * etype->size);
                    645:                frame->index++;
                    646:                break;
                    647:            }
                    648: 
                    649:        case STRUCTS_TYPE_PRIMITIVE:
                    650:                sev = (info->flags & STRUCTS_XML_LOOSE) == 0 ?
                    651:                    LOG_ERR : LOG_WARNING;
                    652:                (*info->logger)(sev,
                    653:                    "line %d:%d: element \"%s\" is not expected here",
                    654:                    XML_GetCurrentLineNumber(info->p),
                    655:                    XML_GetCurrentColumnNumber(info->p), name);
                    656:                if (sev == LOG_ERR) {
                    657:                        info->error = EINVAL;
                    658:                        return;
                    659:                }
                    660:                info->skip++;
                    661:                return;
                    662: 
                    663:        default:
                    664:                assert(0);
                    665:                return;
                    666:        }
                    667: 
                    668:        /* Done */
                    669:        *typep = type;
                    670:        *datap = data;
                    671: }
                    672: 
                    673: /*
                    674:  * Prepare the next level of nesting for parsing.
                    675:  */
                    676: static void
                    677: structs_xml_input_prep(struct xml_input_info *info,
                    678:        const struct structs_type *type, void *data, int combined)
                    679: {
                    680:        /* Dereference through pointer(s) */
                    681:        while (type->tclass == STRUCTS_TYPE_POINTER) {
                    682:                type = type->args[0].v;
                    683:                data = *((void **)data);
                    684:        }
                    685: 
                    686:        /* If next item is an array, re-initialize it */
                    687:        switch (type->tclass) {
                    688:        case STRUCTS_TYPE_ARRAY:
                    689:                (*type->uninit)(type, data);
                    690:                memset(data, 0, type->size);
                    691:                break;
                    692:        case STRUCTS_TYPE_FIXEDARRAY:
                    693:            {
                    694:                void *mem;
                    695: 
                    696:                /* Get temporary region for newly initialized array */
                    697:                if ((mem = MALLOC(TYPED_MEM_TEMP, type->size)) == NULL) {
                    698:                        info->error = errno;
                    699:                        (*info->logger)(LOG_ERR, "line %d:%d: %s: %s",
                    700:                            XML_GetCurrentLineNumber(info->p),
                    701:                            XML_GetCurrentColumnNumber(info->p),
                    702:                            "error initializing new array", strerror(errno));
                    703:                        return;
                    704:                }
                    705: 
                    706:                /* Initialize new array */
                    707:                if ((*type->init)(type, mem) == -1) {
                    708:                        info->error = errno;
                    709:                        (*info->logger)(LOG_ERR, "line %d:%d: %s: %s",
                    710:                            XML_GetCurrentLineNumber(info->p),
                    711:                            XML_GetCurrentColumnNumber(info->p),
                    712:                            "error initializing new array", strerror(errno));
                    713:                        FREE(TYPED_MEM_TEMP, mem);
                    714:                        return;
                    715:                }
                    716: 
                    717:                /* Replace existing array with fresh one */
                    718:                (*type->uninit)(type, data);
                    719:                memcpy(data, mem, type->size);
                    720:                FREE(TYPED_MEM_TEMP, mem);
                    721: 
                    722:                /* Remember that we're on the first element */
                    723:                info->stack[info->depth + 1].index = 0;
                    724:                break;
                    725:            }
                    726:        default:
                    727:                break;
                    728:        }
                    729: 
                    730:        /* Check stack overflow */
                    731:        if (info->depth == MAX_XML_STACK - 1) {
                    732:                (*info->logger)(LOG_ERR,
                    733:                    "line %d:%d: maximum parse stack depth (%d) exceeded",
                    734:                    XML_GetCurrentLineNumber(info->p),
                    735:                    XML_GetCurrentColumnNumber(info->p), MAX_XML_STACK);
                    736:                info->error = EMLINK;
                    737:                return;
                    738:        }
                    739: 
                    740:        /* Continue in a new stack frame */
                    741:        info->depth++;
                    742:        info->stack[info->depth].type = type;
                    743:        info->stack[info->depth].data = data;
                    744:        info->stack[info->depth].combined = combined;
                    745: }
                    746: 
                    747: /*
                    748:  * Character data handler
                    749:  */
                    750: static void
                    751: structs_xml_input_chardata(void *userData, const XML_Char *s, int len)
                    752: {
                    753:        struct xml_input_info *const info = userData;
                    754:        struct xmlinput_stackframe *const frame = &info->stack[info->depth];
                    755:        void *mem;
                    756: 
                    757:        /* Skip if any errors */
                    758:        if (info->error || info->skip)
                    759:                return;
                    760: 
                    761:        /* Expand buffer and append character data */
                    762:        if ((mem = REALLOC(CHARDATA_MEM_TYPE,
                    763:            frame->s, frame->s_len + len + 1)) == NULL) {
                    764:                info->error = errno;
                    765:                (*info->logger)(LOG_ERR, "%s: %s",
                    766:                    "realloc", strerror(errno));
                    767:                return;
                    768:        }
                    769:        frame->s = mem;
                    770:        memcpy(frame->s + frame->s_len, (char *)s, len);
                    771:        frame->s[frame->s_len + len] = '\0';
                    772:        frame->s_len += len;
                    773: }
                    774: 
                    775: /*
                    776:  * End tag handler
                    777:  */
                    778: static void
                    779: structs_xml_input_end(void *userData, const XML_Char *name)
                    780: {
                    781:        struct xml_input_info *const info = userData;
                    782:        struct xmlinput_stackframe *frame;
                    783:        int was_combined;
                    784: 
                    785:        /* Un-nest once for each structs tag */
                    786:        do {
                    787:                frame = &info->stack[info->depth];
                    788:                was_combined = frame->combined;
                    789:                structs_xml_unnest(info, name);
                    790:        } while (was_combined);
                    791: }
                    792: 
                    793: /*
                    794:  * Unnest one level
                    795:  */
                    796: static void
                    797: structs_xml_unnest(struct xml_input_info *info, const XML_Char *name)
                    798: {
                    799:        struct xmlinput_stackframe *const frame = &info->stack[info->depth];
                    800:        const struct structs_type *type;
                    801:        const char *s;
                    802:        char ebuf[64];
                    803:        void *data;
                    804: 
                    805:        /* Skip if any errors */
                    806:        if (info->error)
                    807:                return;
                    808:        if (info->skip) {
                    809:                info->skip--;
                    810:                return;
                    811:        }
                    812: 
                    813:        /* Get current type and data */
                    814:        data = frame->data;
                    815:        type = frame->type;
                    816: 
                    817:        /*
                    818:         * Convert from ASCII if possible, otherwise check only whitespace.
                    819:         * For unions, we allow the field name tag to be omitted if you
                    820:         * want to use the default field, which must have primitive type.
                    821:         */
                    822:        switch (type->tclass) {
                    823:        case STRUCTS_TYPE_UNION:
                    824:            {
                    825:                const struct structs_ufield *const field = type->args[0].v;
                    826: 
                    827:                /* Check to see if there's any non-whitespace text */
                    828:                if (frame->s == NULL)
                    829:                        goto done;
                    830:                for (s = frame->s; *s != '\0' && isspace(*s); s++);
                    831:                if (*s == '\0')
                    832:                        goto done;
                    833: 
                    834:                /* Default field must have primitive type */
                    835:                if (field->type->tclass != STRUCTS_TYPE_PRIMITIVE)
                    836:                        break;
                    837: 
                    838:                /* Switch the union to the default field */
                    839:                if (structs_union_set(type, NULL, data, field->name) == -1) {
                    840:                        info->error = errno;
                    841:                        (*info->logger)(LOG_ERR,
                    842:                            "%s: %s", "structs_union_set", strerror(errno));
                    843:                        return;
                    844:                }
                    845: 
                    846:                /* Point at the field instead of the union */
                    847:                type = field->type;
                    848:                data = ((const struct structs_union *)data)->un;
                    849: 
                    850:                /* FALLTHROUGH */
                    851:            }
                    852:        case STRUCTS_TYPE_PRIMITIVE:
                    853:                if (structs_set_string(type, NULL,
                    854:                    frame->s, data, ebuf, sizeof(ebuf)) == -1) {
                    855:                        info->error = errno;
                    856:                        (*info->logger)(LOG_ERR,
                    857:                            "line %d:%d: error in \"%s\" element data"
                    858:                            " \"%s\": %s",
                    859:                            XML_GetCurrentLineNumber(info->p),
                    860:                            XML_GetCurrentColumnNumber(info->p),
                    861:                            name, frame->s == NULL ? "" : frame->s, ebuf);
                    862:                        return;
                    863:                }
                    864:                goto done;
                    865:        default:
                    866:                break;
                    867:        }
                    868: 
                    869:        /* There shouldn't be any non-whitespace text here */
                    870:        if (frame->s != NULL) {
                    871:                for (s = frame->s; *s != '\0' && isspace(*s); s++);
                    872:                if (*s != '\0') {
                    873:                        (*info->logger)(LOG_ERR, "line %d:%d:"
                    874:                            " extra garbage within \"%s\" element",
                    875:                            XML_GetCurrentLineNumber(info->p),
                    876:                            XML_GetCurrentColumnNumber(info->p),
                    877:                            name);
                    878:                        info->error = EINVAL;
                    879:                        return;
                    880:                }
                    881:        }
                    882: 
                    883: done:
                    884:        /* Pop stack frame */
                    885:        structs_xml_pop(info);
                    886: }
                    887: 
                    888: /*
                    889:  * Pop the XML parse stack
                    890:  */
                    891: static void
                    892: structs_xml_pop(struct xml_input_info *info)
                    893: {
                    894:        struct xmlinput_stackframe *const frame = &info->stack[info->depth];
                    895: 
                    896:        assert(info->depth >= 0);
                    897:        if (frame->s != NULL)
                    898:                FREE(CHARDATA_MEM_TYPE, frame->s);
                    899:        memset(frame, 0, sizeof(*frame));
                    900:        info->depth--;
                    901: }
                    902: 
                    903: /*********************************************************************
                    904:                        XML OUTPUT ROUTINES
                    905: *********************************************************************/
                    906: 
                    907: #define STRUCTS_XML_SHOWONE    0x0100  /* show elem for next level only */
                    908: #define STRUCTS_XML_SHOWALL    0x0200  /* show elem and all sub elems */
                    909: 
                    910: /*
                    911:  * Internal functions
                    912:  */
                    913: static int     structs_xml_output_sub(const struct structs_type *type,
                    914:                        const void *data, const char *tag, const char *attrs,
                    915:                        FILE *fp, const char **elems, const char *posn,
                    916:                        int flags, int depth);
                    917: static void    structs_xml_output_openelem(FILE *fp,
                    918:                        int depth, const char *tag, const char *attrs);
                    919: 
                    920: /*
                    921:  * Output a structure in XML
                    922:  *
                    923:  * Note: it is safe for the calling thread to be canceled.
                    924:  */
                    925: int
                    926: structs_xml_output(const struct structs_type *type, const char *elem_tag,
                    927:        const char *attrs, const void *data, FILE *fp, const char **elems,
                    928:        int flags)
                    929: {
                    930:        static const char *all[] = { "", NULL };
                    931: 
                    932:        /* Output standard XML header */
                    933:        fputs(XML_HEADER, fp);
                    934: 
                    935:        /* NULL elems list means "show everything" */
                    936:        if (elems == NULL)
                    937:                elems = all;
                    938: 
                    939:        /* Output structure, and always show the opening and closing tags */
                    940:        return (structs_xml_output_sub(type, data, elem_tag, attrs, fp,
                    941:            elems, "", flags | STRUCTS_XML_SHOWONE, 0));
                    942: }
                    943: 
                    944: /*
                    945:  * Output a sub-structure in XML
                    946:  */
                    947: static int
                    948: structs_xml_output_sub(const struct structs_type *type, const void *data,
                    949:        const char *tag, const char *attrs, FILE *fp, const char **elems,
                    950:        const char *posn, int flags, int depth)
                    951: {
                    952:        int r = 0;
                    953:        int i;
                    954: 
                    955:        /* Dereference through pointer(s) */
                    956:        while (type->tclass == STRUCTS_TYPE_POINTER) {
                    957:                type = type->args[0].v;
                    958:                data = *((void **)data);
                    959:        }
                    960: 
                    961:        /* Determine whether to show this element */
                    962:        if ((flags & STRUCTS_XML_SHOWALL) == 0) {
                    963:                const size_t plen = strlen(posn);
                    964: 
                    965:                for (i = 0; elems[i] != NULL; i++) {
                    966:                        if (strncmp(elems[i], posn, plen) == 0
                    967:                            && (elems[i][plen] == '\0'
                    968:                              || elems[i][plen] == STRUCTS_SEPARATOR)) {
                    969:                                if (elems[i][plen] == '\0')
                    970:                                        flags |= STRUCTS_XML_SHOWALL;
                    971:                                break;
                    972:                        }
                    973:                }
                    974:                if (elems[i] == NULL && depth > 0
                    975:                    && (flags & STRUCTS_XML_SHOWONE) == 0)
                    976:                        return (0);             /* not matched, skip element */
                    977:        }
                    978: 
                    979:        /* If doing abbreviated version, compare with default value */
                    980:        if (depth > 0
                    981:            && (flags & (STRUCTS_XML_FULL|STRUCTS_XML_SHOWONE)) == 0) {
                    982:                void *init_value;
                    983:                int equal;
                    984: 
                    985:                if ((init_value = MALLOC(TYPED_MEM_TEMP, type->size)) == NULL)
                    986:                        return (-1);
                    987:                if ((*type->init)(type, init_value) == -1) {
                    988:                        FREE(TYPED_MEM_TEMP, init_value);
                    989:                        return (-1);
                    990:                }
                    991:                equal = (*type->equal)(type, data, init_value);
                    992:                (*type->uninit)(type, init_value);
                    993:                FREE(TYPED_MEM_TEMP, init_value);
                    994:                if (equal)
                    995:                        return (0);
                    996:        }
                    997: 
                    998:        /* The STRUCTS_XML_SHOWONE flag only applies to the next level down */
                    999:        flags &= ~STRUCTS_XML_SHOWONE;
                   1000: 
                   1001:        /* Output element */
                   1002:        switch (type->tclass) {
                   1003:        case STRUCTS_TYPE_UNION:
                   1004:            {
                   1005:                const struct structs_union *const un = data;
                   1006:                const struct structs_ufield *const fields = type->args[0].v;
                   1007:                const struct structs_ufield *field;
                   1008:                char *sposn;
                   1009: 
                   1010:                /* Find field */
                   1011:                for (field = fields; field->name != NULL
                   1012:                    && strcmp(un->field_name, field->name) != 0; field++);
                   1013:                if (field->name == NULL)
                   1014:                        assert(0);
                   1015: 
                   1016:                /* Generate new position tag */
                   1017:                ASPRINTF(OUTPUT_MEM_TYPE, &sposn, "%s%s%s", posn,
                   1018:                    *posn != '\0' ? separator_string : "", field->name);
                   1019:                if (sposn == NULL)
                   1020:                        return (-1);
                   1021:                pthread_cleanup_push(structs_xml_output_cleanup, sposn);
                   1022: 
                   1023:                /* Opening tag */
                   1024:                structs_xml_output_openelem(fp, depth, tag, attrs);
                   1025:                fprintf(fp, "\n");
                   1026: 
                   1027:                /*
                   1028:                 * If the union field is not the default choice for this union,
                   1029:                 * then it must always be shown so the recipient knows that.
                   1030:                 */
                   1031:                if (strcmp(un->field_name, fields[0].name) != 0)
                   1032:                        flags |= STRUCTS_XML_SHOWONE;
                   1033: 
                   1034:                /* Output chosen union field */
                   1035:                r = structs_xml_output_sub(field->type, un->un, field->name,
                   1036:                    NULL, fp, elems, sposn, flags, depth + 1);
                   1037: 
                   1038:                /* Free position tag */
                   1039:                pthread_cleanup_pop(1);
                   1040: 
                   1041:                /* Bail out if there was an error */
                   1042:                if (r == -1)
                   1043:                        break;
                   1044: 
                   1045:                /* Closing tag */
                   1046:                structs_xml_output_prefix(fp, depth);
                   1047:                fprintf(fp, "</%s>\n", tag);
                   1048:                break;
                   1049:            }
                   1050: 
                   1051:        case STRUCTS_TYPE_STRUCTURE:
                   1052:            {
                   1053:                const struct structs_field *field;
                   1054: 
                   1055:                /* Opening tag */
                   1056:                structs_xml_output_openelem(fp, depth, tag, attrs);
                   1057:                fprintf(fp, "\n");
                   1058: 
                   1059:                /* Do each structure field */
                   1060:                for (field = type->args[0].v; field->name != NULL; field++) {
                   1061:                        char *sposn;
                   1062: 
                   1063:                        /* Generate new position tag */
                   1064:                        ASPRINTF(OUTPUT_MEM_TYPE, &sposn, "%s%s%s", posn,
                   1065:                            *posn != '\0' ? separator_string : "", field->name);
                   1066:                        if (sposn == NULL)
                   1067:                                return (-1);
                   1068:                        pthread_cleanup_push(structs_xml_output_cleanup, sposn);
                   1069: 
                   1070:                        /* Do structure field */
                   1071:                        r = structs_xml_output_sub(field->type,
                   1072:                            (char *)data + field->offset, field->name, NULL,
                   1073:                            fp, elems, sposn, flags, depth + 1);
                   1074: 
                   1075:                        /* Free position tag */
                   1076:                        pthread_cleanup_pop(1);
                   1077: 
                   1078:                        /* Bail out if there was an error */
                   1079:                        if (r == -1)
                   1080:                                break;
                   1081:                }
                   1082: 
                   1083:                /* Closing tag */
                   1084:                structs_xml_output_prefix(fp, depth);
                   1085:                fprintf(fp, "</%s>\n", tag);
                   1086:                break;
                   1087:            }
                   1088: 
                   1089:        case STRUCTS_TYPE_ARRAY:
                   1090:            {
                   1091:                const struct structs_type *const etype = type->args[0].v;
                   1092:                const char *elem_name = type->args[2].s;
                   1093:                const struct structs_array *const ary = data;
                   1094:                int i;
                   1095: 
                   1096:                /* Opening tag */
                   1097:                structs_xml_output_openelem(fp, depth, tag, attrs);
                   1098:                fprintf(fp, "\n");
                   1099: 
                   1100:                /* All array elements must be shown to keep proper ordering */
                   1101:                flags |= STRUCTS_XML_SHOWONE;
                   1102: 
                   1103:                /* Do elements in order */
                   1104:                for (i = 0; i < ary->length; i++) {
                   1105:                        char *sposn;
                   1106: 
                   1107:                        /* Generate new position tag */
                   1108:                        ASPRINTF(OUTPUT_MEM_TYPE, &sposn, "%s%s%u",
                   1109:                            posn, *posn != '\0' ? separator_string : "", i);
                   1110:                        if (sposn == NULL)
                   1111:                                return (-1);
                   1112:                        pthread_cleanup_push(structs_xml_output_cleanup, sposn);
                   1113: 
                   1114:                        /* Output array element */
                   1115:                        r = structs_xml_output_sub(etype, (char *)ary->elems
                   1116:                            + (i * etype->size), elem_name, NULL, fp, elems,
                   1117:                            sposn, flags, depth + 1);
                   1118: 
                   1119:                        /* Free position tag */
                   1120:                        pthread_cleanup_pop(1);
                   1121: 
                   1122:                        /* Bail out if there was an error */
                   1123:                        if (r == -1)
                   1124:                                break;
                   1125:                }
                   1126: 
                   1127:                /* Closing tag */
                   1128:                structs_xml_output_prefix(fp, depth);
                   1129:                fprintf(fp, "</%s>\n", tag);
                   1130:                break;
                   1131:            }
                   1132: 
                   1133:        case STRUCTS_TYPE_FIXEDARRAY:
                   1134:            {
                   1135:                const struct structs_type *const etype = type->args[0].v;
                   1136:                const char *elem_name = type->args[1].s;
                   1137:                const u_int length = type->args[2].i;
                   1138:                u_int i;
                   1139: 
                   1140:                /* Opening tag */
                   1141:                structs_xml_output_openelem(fp, depth, tag, attrs);
                   1142:                fprintf(fp, "\n");
                   1143: 
                   1144:                /* All array elements must be shown to keep proper ordering */
                   1145:                flags |= STRUCTS_XML_SHOWONE;
                   1146: 
                   1147:                /* Do elements in order */
                   1148:                for (i = 0; i < length; i++) {
                   1149:                        char *sposn;
                   1150: 
                   1151:                        /* Generate new position tag */
                   1152:                        ASPRINTF(OUTPUT_MEM_TYPE, &sposn, "%s%s%u",
                   1153:                            posn, *posn != '\0' ? separator_string : "", i);
                   1154:                        if (sposn == NULL)
                   1155:                                return (-1);
                   1156:                        pthread_cleanup_push(structs_xml_output_cleanup, sposn);
                   1157: 
                   1158:                        /* Output array element */
                   1159:                        r = structs_xml_output_sub(etype, (char *)data
                   1160:                            + (i * etype->size), elem_name, NULL, fp, elems,
                   1161:                            sposn, flags, depth + 1);
                   1162: 
                   1163:                        /* Free position tag */
                   1164:                        pthread_cleanup_pop(1);
                   1165: 
                   1166:                        /* Bail out if there was an error */
                   1167:                        if (r == -1)
                   1168:                                break;
                   1169:                }
                   1170: 
                   1171:                /* Closing tag */
                   1172:                structs_xml_output_prefix(fp, depth);
                   1173:                fprintf(fp, "</%s>\n", tag);
                   1174:                break;
                   1175:            }
                   1176: 
                   1177:        case STRUCTS_TYPE_PRIMITIVE:
                   1178:            {
                   1179:                char *ascii;
                   1180: 
                   1181:                /* Get ascii string */
                   1182:                if ((ascii = (*type->ascify)(type,
                   1183:                    OUTPUT_MEM_TYPE, data)) == NULL)
                   1184:                        return (-1);
                   1185: 
                   1186:                /* Push cleanup hook to handle cancellation */
                   1187:                pthread_cleanup_push(structs_xml_output_cleanup, ascii);
                   1188: 
                   1189:                /* Output element */
                   1190:                structs_xml_output_openelem(fp, depth, tag, attrs);
                   1191:                structs_xml_encode(fp, ascii);
                   1192:                fprintf(fp, "</%s>\n", tag);
                   1193: 
                   1194:                /* Free ascii string */
                   1195:                pthread_cleanup_pop(1);
                   1196:                break;
                   1197:            }
                   1198: 
                   1199:        default:
                   1200:                assert(0);
                   1201:        }
                   1202:        return (r);
                   1203: }
                   1204: 
                   1205: /*
                   1206:  * Cleanup for structs_xml_output_sub()
                   1207:  */
                   1208: static void
                   1209: structs_xml_output_cleanup(void *arg)
                   1210: {
                   1211:        FREE(OUTPUT_MEM_TYPE, arg);
                   1212: }
                   1213: 
                   1214: /*
                   1215:  * Output opening element tag with optional attributes
                   1216:  */
                   1217: static void
                   1218: structs_xml_output_openelem(FILE *fp,
                   1219:        int depth, const char *tag, const char *attrs)
                   1220: {
                   1221:        structs_xml_output_prefix(fp, depth);
                   1222:        fprintf(fp, "<%s", tag);
                   1223:        if (attrs != NULL) {
                   1224:                while (*attrs != '\0') {
                   1225:                        fprintf(fp, " ");
                   1226:                        structs_xml_encode(fp, attrs);
                   1227:                        attrs += strlen(attrs) + 1;
                   1228:                        fprintf(fp, "=\"");
                   1229:                        structs_xml_encode(fp, attrs);
                   1230:                        attrs += strlen(attrs) + 1;
                   1231:                        fprintf(fp, "\"");
                   1232:                }
                   1233:        }
                   1234:        fprintf(fp, ">");
                   1235: }
                   1236: 
                   1237: /*********************************************************************
                   1238:                        UTILITY STUFF
                   1239: *********************************************************************/
                   1240: 
                   1241: /*
                   1242:  * Output whitespace prefix
                   1243:  */
                   1244: static void
                   1245: structs_xml_output_prefix(FILE *fp, int depth)
                   1246: {
                   1247:        while (depth >= 2) {
                   1248:                putc('\t', fp);
                   1249:                depth -= 2;
                   1250:        }
                   1251:        if (depth > 0) {
                   1252:                fputs("    ", fp);
                   1253:        }
                   1254: }
                   1255: 
                   1256: /*
                   1257:  * Output text XML-encoded
                   1258:  */
                   1259: static void
                   1260: structs_xml_encode(FILE *fp, const char *s)
                   1261: {
                   1262:        for ( ; *s != '\0'; s++) {
                   1263:                switch (*s) {
                   1264:                case '<':
                   1265:                        fprintf(fp, "&lt;");
                   1266:                        break;
                   1267:                case '>':
                   1268:                        fprintf(fp, "&gt;");
                   1269:                        break;
                   1270:                case '"':
                   1271:                        fprintf(fp, "&quot;");
                   1272:                        break;
                   1273:                case '&':
                   1274:                        fprintf(fp, "&amp;");
                   1275:                        break;
                   1276:                        break;
                   1277:                default:
                   1278:                        if (!isprint(*s)) {
                   1279:                                fprintf(fp, "&#%d;", (u_char)*s);
                   1280:                                break;
                   1281:                        }
                   1282:                        /* fall through */
                   1283:                case '\n':
                   1284:                case '\t':
                   1285:                        putc(*s, fp);
                   1286:                        break;
                   1287:                }
                   1288:        }
                   1289: }
                   1290: 
                   1291: /*********************************************************************
                   1292:                        BUILT-IN LOGGERS
                   1293: *********************************************************************/
                   1294: 
                   1295: static void
                   1296: structs_xml_null_logger(int sev, const char *fmt, ...)
                   1297: {
                   1298: }
                   1299: 
                   1300: static void
                   1301: structs_xml_stderr_logger(int sev, const char *fmt, ...)
                   1302: {
                   1303:        static const char *const sevs[] = {
                   1304:                "emerg", "alert", "crit", "err",
                   1305:                "warning", "notice", "info", "debug"
                   1306:        };
                   1307:        static const int num_sevs = sizeof(sevs) / sizeof(*sevs);
                   1308:        va_list args;
                   1309: 
                   1310:        va_start(args, fmt);
                   1311:        if (sev < 0)
                   1312:                sev = 0;
                   1313:        if (sev >= num_sevs)
                   1314:                sev = num_sevs - 1;
                   1315:        fprintf(stderr, "%s: ", sevs[sev]);
                   1316:        vfprintf(stderr, fmt, args);
                   1317:        fprintf(stderr, "\n");
                   1318:        va_end(args);
                   1319: }
                   1320: 
                   1321: static void
                   1322: structs_xml_alog_logger(int sev, const char *fmt, ...)
                   1323: {
                   1324:        va_list args;
                   1325: 
                   1326:        va_start(args, fmt);
                   1327:        valog(sev, fmt, args);
                   1328:        va_end(args);
                   1329: }
                   1330: 
                   1331: /*********************************************************************
                   1332:                        MEMORY WRAPPERS
                   1333: *********************************************************************/
                   1334: 
                   1335: #define EXPAT_MEM_TYPE                 "structs_xml_input.expat"
                   1336: 
                   1337: static void *
                   1338: structs_xml_malloc(size_t size)
                   1339: {
                   1340:        return (MALLOC(EXPAT_MEM_TYPE, size));
                   1341: }
                   1342: 
                   1343: static void *
                   1344: structs_xml_realloc(void *ptr, size_t size)
                   1345: {
                   1346:        return (REALLOC(EXPAT_MEM_TYPE, ptr, size));
                   1347: }
                   1348: 
                   1349: static void
                   1350: structs_xml_free(void *ptr)
                   1351: {
                   1352:        FREE(EXPAT_MEM_TYPE, ptr);
                   1353: }
                   1354: 

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>