00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039 #ifdef _WIN32
00040 #include "winconfig.h"
00041 #else
00042 #include "config.h"
00043 #endif
00044 #include "mxml.h"
00045
00046
00047
00048
00049
00050
00051 static mxml_node_t *mxml_new(mxml_node_t *parent, mxml_type_t type);
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065 void
00066 mxmlAdd(mxml_node_t *parent,
00067 int where,
00068 mxml_node_t *child,
00069 mxml_node_t *node)
00070 {
00071 #ifdef DEBUG
00072 fprintf(stderr, "mxmlAdd(parent=%p, where=%d, child=%p, node=%p)\n", parent,
00073 where, child, node);
00074 #endif
00075
00076
00077
00078
00079
00080 if (!parent || !node)
00081 return;
00082
00083 #if DEBUG > 1
00084 fprintf(stderr, " BEFORE: node->parent=%p\n", node->parent);
00085 if (parent)
00086 {
00087 fprintf(stderr, " BEFORE: parent->child=%p\n", parent->child);
00088 fprintf(stderr, " BEFORE: parent->last_child=%p\n", parent->last_child);
00089 fprintf(stderr, " BEFORE: parent->prev=%p\n", parent->prev);
00090 fprintf(stderr, " BEFORE: parent->next=%p\n", parent->next);
00091 }
00092 #endif
00093
00094
00095
00096
00097
00098 if (node->parent)
00099 mxmlRemove(node);
00100
00101
00102
00103
00104
00105 node->parent = parent;
00106
00107 switch (where)
00108 {
00109 case MXML_ADD_BEFORE :
00110 if (!child || child == parent->child || child->parent != parent)
00111 {
00112
00113
00114
00115
00116 node->next = parent->child;
00117
00118 if (parent->child)
00119 parent->child->prev = node;
00120 else
00121 parent->last_child = node;
00122
00123 parent->child = node;
00124 }
00125 else
00126 {
00127
00128
00129
00130
00131 node->next = child;
00132 node->prev = child->prev;
00133
00134 if (child->prev)
00135 child->prev->next = node;
00136 else
00137 parent->child = node;
00138
00139 child->prev = node;
00140 }
00141 break;
00142
00143 case MXML_ADD_AFTER :
00144 if (!child || child == parent->last_child || child->parent != parent)
00145 {
00146
00147
00148
00149
00150 node->parent = parent;
00151 node->prev = parent->last_child;
00152
00153 if (parent->last_child)
00154 parent->last_child->next = node;
00155 else
00156 parent->child = node;
00157
00158 parent->last_child = node;
00159 }
00160 else
00161 {
00162
00163
00164
00165
00166 node->prev = child;
00167 node->next = child->next;
00168
00169 if (child->next)
00170 child->next->prev = node;
00171 else
00172 parent->last_child = node;
00173
00174 child->next = node;
00175 }
00176 break;
00177 }
00178
00179 #if DEBUG > 1
00180 fprintf(stderr, " AFTER: node->parent=%p\n", node->parent);
00181 if (parent)
00182 {
00183 fprintf(stderr, " AFTER: parent->child=%p\n", parent->child);
00184 fprintf(stderr, " AFTER: parent->last_child=%p\n", parent->last_child);
00185 fprintf(stderr, " AFTER: parent->prev=%p\n", parent->prev);
00186 fprintf(stderr, " AFTER: parent->next=%p\n", parent->next);
00187 }
00188 #endif
00189 }
00190
00191
00192
00193
00194
00195
00196
00197
00198
00199 void
00200 mxmlDelete(mxml_node_t *node)
00201 {
00202 int i;
00203
00204
00205 #ifdef DEBUG
00206 fprintf(stderr, "mxmlDelete(node=%p)\n", node);
00207 #endif
00208
00209
00210
00211
00212
00213 if (!node)
00214 return;
00215
00216
00217
00218
00219
00220 mxmlRemove(node);
00221
00222
00223
00224
00225
00226 while (node->child)
00227 mxmlDelete(node->child);
00228
00229
00230
00231
00232
00233 switch (node->type)
00234 {
00235 case MXML_ELEMENT :
00236 if (node->value.element.name)
00237 free(node->value.element.name);
00238
00239 if (node->value.element.num_attrs)
00240 {
00241 for (i = 0; i < node->value.element.num_attrs; i ++)
00242 {
00243 if (node->value.element.attrs[i].name)
00244 free(node->value.element.attrs[i].name);
00245 if (node->value.element.attrs[i].value)
00246 free(node->value.element.attrs[i].value);
00247 }
00248
00249 free(node->value.element.attrs);
00250 }
00251 break;
00252 case MXML_INTEGER :
00253
00254 break;
00255 case MXML_OPAQUE :
00256 if (node->value.opaque)
00257 free(node->value.opaque);
00258 break;
00259 case MXML_REAL :
00260
00261 break;
00262 case MXML_TEXT :
00263 if (node->value.text.string)
00264 free(node->value.text.string);
00265 break;
00266 case MXML_CUSTOM :
00267 if (node->value.custom.data &&
00268 node->value.custom.destroy)
00269 (*(node->value.custom.destroy))(node->value.custom.data);
00270 break;
00271 }
00272
00273
00274
00275
00276
00277 free(node);
00278 }
00279
00280
00281
00282
00283
00284
00285
00286
00287
00288
00289
00290 mxml_node_t *
00291 mxmlNewCustom(mxml_node_t *parent,
00292 void *data,
00293 void (*destroy)(void *))
00294
00295 {
00296 mxml_node_t *node;
00297
00298
00299 #ifdef DEBUG
00300 fprintf(stderr, "mxmlNewCustom(parent=%p, data=%p, destroy=%p)\n", parent,
00301 data, destroy);
00302 #endif
00303
00304
00305
00306
00307
00308 if ((node = mxml_new(parent, MXML_CUSTOM)) != NULL)
00309 {
00310 node->value.custom.data = data;
00311 node->value.custom.destroy = destroy;
00312 }
00313
00314 return (node);
00315 }
00316
00317
00318
00319
00320
00321
00322
00323
00324
00325
00326 mxml_node_t *
00327 mxmlNewElement(mxml_node_t *parent,
00328 const char *name)
00329 {
00330 mxml_node_t *node;
00331
00332
00333 #ifdef DEBUG
00334 fprintf(stderr, "mxmlNewElement(parent=%p, name=\"%s\")\n", parent,
00335 name ? name : "(null)");
00336 #endif
00337
00338
00339
00340
00341
00342 if (!name)
00343 return (NULL);
00344
00345
00346
00347
00348
00349 if ((node = mxml_new(parent, MXML_ELEMENT)) != NULL)
00350 node->value.element.name = strdup(name);
00351
00352 return (node);
00353 }
00354
00355
00356
00357
00358
00359
00360
00361
00362
00363
00364 mxml_node_t *
00365 mxmlNewInteger(mxml_node_t *parent,
00366 int integer)
00367 {
00368 mxml_node_t *node;
00369
00370
00371 #ifdef DEBUG
00372 fprintf(stderr, "mxmlNewInteger(parent=%p, integer=%d)\n", parent, integer);
00373 #endif
00374
00375
00376
00377
00378
00379 if ((node = mxml_new(parent, MXML_INTEGER)) != NULL)
00380 node->value.integer = integer;
00381
00382 return (node);
00383 }
00384
00385
00386
00387
00388
00389
00390
00391
00392
00393
00394
00395 mxml_node_t *
00396 mxmlNewOpaque(mxml_node_t *parent,
00397 const char *opaque)
00398 {
00399 mxml_node_t *node;
00400
00401
00402 #ifdef DEBUG
00403 fprintf(stderr, "mxmlNewOpaque(parent=%p, opaque=\"%s\")\n", parent,
00404 opaque ? opaque : "(null)");
00405 #endif
00406
00407
00408
00409
00410
00411 if (!opaque)
00412 return (NULL);
00413
00414
00415
00416
00417
00418 if ((node = mxml_new(parent, MXML_OPAQUE)) != NULL)
00419 node->value.opaque = strdup(opaque);
00420
00421 return (node);
00422 }
00423
00424
00425
00426
00427
00428
00429
00430
00431
00432
00433 mxml_node_t *
00434 mxmlNewReal(mxml_node_t *parent,
00435 double real)
00436 {
00437 mxml_node_t *node;
00438
00439
00440 #ifdef DEBUG
00441 fprintf(stderr, "mxmlNewReal(parent=%p, real=%g)\n", parent, real);
00442 #endif
00443
00444
00445
00446
00447
00448 if ((node = mxml_new(parent, MXML_REAL)) != NULL)
00449 node->value.real = real;
00450
00451 return (node);
00452 }
00453
00454
00455
00456
00457
00458
00459
00460
00461
00462
00463
00464
00465 mxml_node_t *
00466 mxmlNewText(mxml_node_t *parent,
00467 int whitespace,
00468 const char *string)
00469 {
00470 mxml_node_t *node;
00471
00472
00473 #ifdef DEBUG
00474 fprintf(stderr, "mxmlNewText(parent=%p, whitespace=%d, string=\"%s\")\n",
00475 parent, whitespace, string ? string : "(null)");
00476 #endif
00477
00478
00479
00480
00481
00482 if (!string)
00483 return (NULL);
00484
00485
00486
00487
00488
00489 if ((node = mxml_new(parent, MXML_TEXT)) != NULL)
00490 {
00491 node->value.text.whitespace = whitespace;
00492 node->value.text.string = strdup(string);
00493 }
00494
00495 return (node);
00496 }
00497
00498
00499
00500
00501
00502
00503
00504
00505
00506
00507
00508
00509 mxml_node_t *
00510 mxmlNewTextf(mxml_node_t *parent,
00511 int whitespace,
00512 const char *format,
00513 ...)
00514 {
00515 mxml_node_t *node;
00516 va_list ap;
00517
00518
00519 #ifdef DEBUG
00520 fprintf(stderr, "mxmlNewTextf(parent=%p, whitespace=%d, format=\"%s\", ...)\n",
00521 parent, whitespace, format ? format : "(null)");
00522 #endif
00523
00524
00525
00526
00527
00528 if (!format)
00529 return (NULL);
00530
00531
00532
00533
00534
00535 if ((node = mxml_new(parent, MXML_TEXT)) != NULL)
00536 {
00537 va_start(ap, format);
00538
00539 node->value.text.whitespace = whitespace;
00540 node->value.text.string = mxml_strdupf(format, ap);
00541
00542 va_end(ap);
00543 }
00544
00545 return (node);
00546 }
00547
00548
00549
00550
00551
00552
00553
00554
00555
00556 void
00557 mxmlRemove(mxml_node_t *node)
00558 {
00559 #ifdef DEBUG
00560 fprintf(stderr, "mxmlRemove(node=%p)\n", node);
00561 #endif
00562
00563
00564
00565
00566
00567 if (!node || !node->parent)
00568 return;
00569
00570
00571
00572
00573
00574 #if DEBUG > 1
00575 fprintf(stderr, " BEFORE: node->parent=%p\n", node->parent);
00576 if (node->parent)
00577 {
00578 fprintf(stderr, " BEFORE: node->parent->child=%p\n", node->parent->child);
00579 fprintf(stderr, " BEFORE: node->parent->last_child=%p\n", node->parent->last_child);
00580 }
00581 fprintf(stderr, " BEFORE: node->child=%p\n", node->child);
00582 fprintf(stderr, " BEFORE: node->last_child=%p\n", node->last_child);
00583 fprintf(stderr, " BEFORE: node->prev=%p\n", node->prev);
00584 fprintf(stderr, " BEFORE: node->next=%p\n", node->next);
00585 #endif
00586
00587 if (node->prev)
00588 node->prev->next = node->next;
00589 else
00590 node->parent->child = node->next;
00591
00592 if (node->next)
00593 node->next->prev = node->prev;
00594 else
00595 node->parent->last_child = node->prev;
00596
00597 node->parent = NULL;
00598 node->prev = NULL;
00599 node->next = NULL;
00600
00601 #if DEBUG > 1
00602 fprintf(stderr, " AFTER: node->parent=%p\n", node->parent);
00603 if (node->parent)
00604 {
00605 fprintf(stderr, " AFTER: node->parent->child=%p\n", node->parent->child);
00606 fprintf(stderr, " AFTER: node->parent->last_child=%p\n", node->parent->last_child);
00607 }
00608 fprintf(stderr, " AFTER: node->child=%p\n", node->child);
00609 fprintf(stderr, " AFTER: node->last_child=%p\n", node->last_child);
00610 fprintf(stderr, " AFTER: node->prev=%p\n", node->prev);
00611 fprintf(stderr, " AFTER: node->next=%p\n", node->next);
00612 #endif
00613 }
00614
00615
00616
00617
00618
00619
00620 static mxml_node_t *
00621 mxml_new(mxml_node_t *parent,
00622 mxml_type_t type)
00623 {
00624 mxml_node_t *node;
00625
00626
00627 #if DEBUG > 1
00628 fprintf(stderr, "mxml_new(parent=%p, type=%d)\n", parent, type);
00629 #endif
00630
00631
00632
00633
00634
00635 if ((node = (mxml_node_t*)calloc(1, sizeof(mxml_node_t))) == NULL)
00636 {
00637 #if DEBUG > 1
00638 fputs(" returning NULL\n", stderr);
00639 #endif
00640
00641 return (NULL);
00642 }
00643
00644 #if DEBUG > 1
00645 fprintf(stderr, " returning %p\n", node);
00646 #endif
00647
00648
00649
00650
00651
00652 node->type = type;
00653
00654
00655
00656
00657
00658 if (parent)
00659 mxmlAdd(parent, MXML_ADD_AFTER, MXML_ADD_TO_PARENT, node);
00660
00661
00662
00663
00664
00665 return (node);
00666 }
00667
00668
00669
00670
00671