50#include "magick/studio.h"
51#include "magick/blob.h"
52#include "magick/blob-private.h"
53#include "magick/exception.h"
54#include "magick/exception-private.h"
55#include "magick/image-private.h"
56#include "magick/log.h"
57#include "magick/memory_.h"
58#include "magick/semaphore.h"
59#include "magick/string_.h"
60#include "magick/string-private.h"
61#include "magick/token-private.h"
62#include "magick/utility.h"
63#include "magick/utility-private.h"
64#include "magick/xml-tree.h"
65#include "magick/xml-tree-private.h"
70#define NumberPredefinedEntities 10
71#define XMLWhitespace "\t\r\n "
118 ***processing_instructions,
136 *sentinel[] = { (
char *) NULL };
166MagickExport XMLTreeInfo *AddChildToXMLTree(XMLTreeInfo *xml_info,
167 const char *tag,
const size_t offset)
172 if (xml_info == (XMLTreeInfo *) NULL)
173 return((XMLTreeInfo *) NULL);
174 child=(XMLTreeInfo *) AcquireMagickMemory(
sizeof(*child));
175 if (child == (XMLTreeInfo *) NULL)
176 return((XMLTreeInfo *) NULL);
177 (void) memset(child,0,
sizeof(*child));
178 child->tag=ConstantString(tag);
179 child->attributes=sentinel;
180 child->content=ConstantString(
"");
181 child->debug=IsEventLogging();
182 child->signature=MagickCoreSignature;
183 return(InsertTagIntoXMLTree(xml_info,child,offset));
214MagickExport XMLTreeInfo *AddPathToXMLTree(XMLTreeInfo *xml_info,
215 const char *path,
const size_t offset)
219 subnode[MaxTextExtent],
233 assert(xml_info != (XMLTreeInfo *) NULL);
234 assert((xml_info->signature == MagickCoreSignature) ||
235 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
236 if (IsEventLogging() != MagickFalse)
237 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
239 components=GetPathComponents(path,&number_components);
240 if (components == (
char **) NULL)
241 return((XMLTreeInfo *) NULL);
242 for (i=0; i < (ssize_t) number_components; i++)
244 GetPathComponent(components[i],SubimagePath,subnode);
245 GetPathComponent(components[i],CanonicalPath,tag);
246 child=GetXMLTreeChild(node,tag);
247 if (child == (XMLTreeInfo *) NULL)
248 child=AddChildToXMLTree(node,tag,offset);
250 if (node == (XMLTreeInfo *) NULL)
252 for (j=(ssize_t) StringToLong(subnode)-1; j > 0; j--)
254 node=GetXMLTreeOrdered(node);
255 if (node == (XMLTreeInfo *) NULL)
258 if (node == (XMLTreeInfo *) NULL)
260 components[i]=DestroyString(components[i]);
262 for ( ; i < (ssize_t) number_components; i++)
263 components[i]=DestroyString(components[i]);
264 components=(
char **) RelinquishMagickMemory(components);
296MagickExport
char *CanonicalXMLContent(
const char *content,
297 const MagickBooleanType pedantic)
316 utf8=ConvertLatin1ToUTF8((
const unsigned char *) content);
317 if (utf8 == (
unsigned char *) NULL)
318 return((
char *) NULL);
319 for (p=utf8; *p !=
'\0'; p++)
320 if ((*p < 0x20) && (*p != 0x09) && (*p != 0x0a) && (*p != 0x0d))
327 base64=Base64Encode(utf8,strlen((
char *) utf8),&length);
328 utf8=(
unsigned char *) RelinquishMagickMemory(utf8);
329 if (base64 == (
char *) NULL)
330 return((
char *) NULL);
331 canonical_content=AcquireString(
"<base64>");
332 (void) ConcatenateString(&canonical_content,base64);
333 base64=DestroyString(base64);
334 (void) ConcatenateString(&canonical_content,
"</base64>");
335 return(canonical_content);
341 canonical_content=AcquireString((
char *) NULL);
342 extent=MaxTextExtent;
343 for (p=utf8; *p !=
'\0'; p++)
345 if ((i+MaxTextExtent) > (ssize_t) extent)
347 extent+=MaxTextExtent;
348 canonical_content=(
char *) ResizeQuantumMemory(canonical_content,extent,
349 sizeof(*canonical_content));
350 if (canonical_content == (
char *) NULL)
351 return(canonical_content);
357 i+=FormatLocaleString(canonical_content+i,extent,
"&");
362 i+=FormatLocaleString(canonical_content+i,extent,
"<");
367 i+=FormatLocaleString(canonical_content+i,extent,
">");
372 i+=FormatLocaleString(canonical_content+i,extent,
""");
377 if (pedantic == MagickFalse)
379 canonical_content[i++]=(char) (*p);
382 i+=FormatLocaleString(canonical_content+i,extent,
"
");
387 if (pedantic == MagickFalse)
389 canonical_content[i++]=(char) (*p);
392 i+=FormatLocaleString(canonical_content+i,extent,
"	");
397 i+=FormatLocaleString(canonical_content+i,extent,
"
");
402 canonical_content[i++]=(char) (*p);
407 canonical_content[i]=
'\0';
408 utf8=(
unsigned char *) RelinquishMagickMemory(utf8);
409 return(canonical_content);
436 *DestroyXMLTree_(XMLTreeInfo *,
const size_t);
438static char **DestroyXMLTreeAttributes(
char **attributes)
446 if ((attributes == (
char **) NULL) || (attributes == sentinel))
447 return((
char **) NULL);
448 for (i=0; attributes[i] != (
char *) NULL; i+=2)
453 if (attributes[i] != (
char *) NULL)
454 attributes[i]=DestroyString(attributes[i]);
455 if (attributes[i+1] != (
char *) NULL)
456 attributes[i+1]=DestroyString(attributes[i+1]);
458 attributes=(
char **) RelinquishMagickMemory(attributes);
459 return((
char **) NULL);
462static void DestroyXMLTreeChild(XMLTreeInfo *xml_info,
469 child=xml_info->child;
470 while (child != (XMLTreeInfo *) NULL)
474 node->child=(XMLTreeInfo *) NULL;
475 (void) DestroyXMLTree_(node,depth+1);
479static void DestroyXMLTreeOrdered(XMLTreeInfo *xml_info,
486 ordered=xml_info->ordered;
487 while (ordered != (XMLTreeInfo *) NULL)
490 ordered=node->ordered;
491 node->ordered=(XMLTreeInfo *) NULL;
492 (void) DestroyXMLTree_(node,depth+1);
496static void DestroyXMLTreeRoot(XMLTreeInfo *xml_info)
508 assert(xml_info != (XMLTreeInfo *) NULL);
509 assert((xml_info->signature == MagickCoreSignature) ||
510 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
511 if (IsEventLogging() != MagickFalse)
512 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
513 if (xml_info->parent != (XMLTreeInfo *) NULL)
518 root=(XMLTreeRoot *) xml_info;
519 for (i=NumberPredefinedEntities; root->entities[i] != (
char *) NULL; i+=2)
520 root->entities[i+1]=DestroyString(root->entities[i+1]);
521 root->entities=(
char **) RelinquishMagickMemory(root->entities);
522 for (i=0; root->attributes[i] != (
char **) NULL; i++)
524 attributes=root->attributes[i];
525 if (attributes[0] != (
char *) NULL)
526 attributes[0]=DestroyString(attributes[0]);
527 for (j=1; attributes[j] != (
char *) NULL; j+=3)
529 if (attributes[j] != (
char *) NULL)
530 attributes[j]=DestroyString(attributes[j]);
531 if (attributes[j+1] != (
char *) NULL)
532 attributes[j+1]=DestroyString(attributes[j+1]);
533 if (attributes[j+2] != (
char *) NULL)
534 attributes[j+2]=DestroyString(attributes[j+2]);
536 attributes=(
char **) RelinquishMagickMemory(attributes);
538 if (root->attributes[0] != (
char **) NULL)
539 root->attributes=(
char ***) RelinquishMagickMemory(root->attributes);
540 if (root->processing_instructions[0] != (
char **) NULL)
542 for (i=0; root->processing_instructions[i] != (
char **) NULL; i++)
544 for (j=0; root->processing_instructions[i][j] != (
char *) NULL; j++)
545 root->processing_instructions[i][j]=DestroyString(
546 root->processing_instructions[i][j]);
547 root->processing_instructions[i][j+1]=DestroyString(
548 root->processing_instructions[i][j+1]);
549 root->processing_instructions[i]=(
char **) RelinquishMagickMemory(
550 root->processing_instructions[i]);
552 root->processing_instructions=(
char ***) RelinquishMagickMemory(
553 root->processing_instructions);
557static XMLTreeInfo *DestroyXMLTree_(XMLTreeInfo *xml_info,
560 assert(xml_info != (XMLTreeInfo *) NULL);
561 assert((xml_info->signature == MagickCoreSignature) ||
562 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
563 if (IsEventLogging() != MagickFalse)
564 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
565 if (depth > MagickMaxRecursionDepth)
566 ThrowFatalException(ResourceLimitFatalError,
567 "MemoryAllocationFailed");
568 DestroyXMLTreeChild(xml_info,depth+1);
569 DestroyXMLTreeOrdered(xml_info,depth+1);
570 DestroyXMLTreeRoot(xml_info);
571 xml_info->attributes=DestroyXMLTreeAttributes(xml_info->attributes);
572 xml_info->content=DestroyString(xml_info->content);
573 xml_info->tag=DestroyString(xml_info->tag);
574 xml_info=(XMLTreeInfo *) RelinquishMagickMemory(xml_info);
575 return((XMLTreeInfo *) NULL);
578MagickExport XMLTreeInfo *DestroyXMLTree(XMLTreeInfo *xml_info)
580 return(DestroyXMLTree_(xml_info,0));
607MagickPrivate
char *FileToXML(
const char *filename,
const size_t extent)
628 assert(filename != (
const char *) NULL);
631 if (LocaleCompare(filename,
"-") != 0)
632 file=open_utf8(filename,O_RDONLY | O_BINARY,0);
634 return((
char *) NULL);
635 offset=(MagickOffsetType) lseek(file,0,SEEK_END);
637 if ((file == fileno(stdin)) || (offset < 0) ||
638 (offset != (MagickOffsetType) ((ssize_t) offset)))
649 offset=(MagickOffsetType) lseek(file,0,SEEK_SET);
650 quantum=(size_t) MagickMaxBufferExtent;
651 if ((fstat(file,&file_stats) == 0) && (file_stats.st_size > 0))
652 quantum=(size_t) MagickMin(file_stats.st_size,MagickMaxBufferExtent);
653 xml=(
char *) AcquireQuantumMemory(quantum,
sizeof(*xml));
654 for (i=0; xml != (
char *) NULL; i+=count)
656 count=read(file,xml+i,quantum);
663 if (~((
size_t) i) < (quantum+1))
665 xml=(
char *) RelinquishMagickMemory(xml);
668 xml=(
char *) ResizeQuantumMemory(xml,i+quantum+1,
sizeof(*xml));
669 if ((
size_t) (i+count) >= extent)
672 if (LocaleCompare(filename,
"-") != 0)
673 file=close_utf8(file);
674 if (xml == (
char *) NULL)
675 return((
char *) NULL);
678 xml=(
char *) RelinquishMagickMemory(xml);
679 return((
char *) NULL);
681 length=(size_t) MagickMin(i+count,extent);
685 length=(size_t) MagickMin(offset,(MagickOffsetType) extent);
687 if (~length >= (MaxTextExtent-1))
688 xml=(
char *) AcquireQuantumMemory(length+MaxTextExtent,
sizeof(*xml));
689 if (xml == (
char *) NULL)
691 file=close_utf8(file);
692 return((
char *) NULL);
694 map=MapBlob(file,ReadMode,0,length);
695 if (map != (
char *) NULL)
697 (void) memcpy(xml,map,length);
698 (void) UnmapBlob(map,length);
702 (void) lseek(file,0,SEEK_SET);
703 for (i=0; i < length; i+=count)
705 count=read(file,xml+i,(
size_t) MagickMin(length-i,(
size_t) MagickMaxBufferExtent));
715 file=close_utf8(file)-1;
716 xml=(
char *) RelinquishMagickMemory(xml);
717 return((
char *) NULL);
721 if (LocaleCompare(filename,
"-") != 0)
722 file=close_utf8(file);
724 xml=(
char *) RelinquishMagickMemory(xml);
750MagickExport XMLTreeInfo *GetNextXMLTreeTag(XMLTreeInfo *xml_info)
752 assert(xml_info != (XMLTreeInfo *) NULL);
753 assert((xml_info->signature == MagickCoreSignature) ||
754 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
755 if (IsEventLogging() != MagickFalse)
756 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
757 return(xml_info->next);
785MagickExport
const char *GetXMLTreeAttribute(XMLTreeInfo *xml_info,
795 assert(xml_info != (XMLTreeInfo *) NULL);
796 assert((xml_info->signature == MagickCoreSignature) ||
797 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
798 if (IsEventLogging() != MagickFalse)
799 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
800 if (xml_info->attributes == (
char **) NULL)
801 return((
const char *) NULL);
803 while ((xml_info->attributes[i] != (
char *) NULL) &&
804 (strcmp(xml_info->attributes[i],tag) != 0))
806 if (xml_info->attributes[i] != (
char *) NULL)
807 return(xml_info->attributes[i+1]);
808 root=(XMLTreeRoot*) xml_info;
809 while (root->root.parent != (XMLTreeInfo *) NULL)
810 root=(XMLTreeRoot *) root->root.parent;
812 while ((root->attributes[i] != (
char **) NULL) &&
813 (strcmp(root->attributes[i][0],xml_info->tag) != 0))
815 if (root->attributes[i] == (
char **) NULL)
816 return((
const char *) NULL);
818 while ((root->attributes[i][j] != (
char *) NULL) &&
819 (strcmp(root->attributes[i][j],tag) != 0))
821 if (root->attributes[i][j] == (
char *) NULL)
822 return((
const char *) NULL);
823 return(root->attributes[i][j+1]);
852MagickExport MagickBooleanType GetXMLTreeAttributes(
const XMLTreeInfo *xml_info,
853 SplayTreeInfo *attributes)
858 assert(xml_info != (XMLTreeInfo *) NULL);
859 assert((xml_info->signature == MagickCoreSignature) ||
860 (((
const XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
861 assert(attributes != (SplayTreeInfo *) NULL);
862 if (IsEventLogging() != MagickFalse)
863 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
864 if (xml_info->attributes == (
char **) NULL)
867 while (xml_info->attributes[i] != (
char *) NULL)
869 (void) AddValueToSplayTree(attributes,
870 ConstantString(xml_info->attributes[i]),
871 ConstantString(xml_info->attributes[i+1]));
900MagickExport XMLTreeInfo *GetXMLTreeChild(XMLTreeInfo *xml_info,
const char *tag)
905 assert(xml_info != (XMLTreeInfo *) NULL);
906 assert((xml_info->signature == MagickCoreSignature) ||
907 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
908 if (IsEventLogging() != MagickFalse)
909 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
910 child=xml_info->child;
911 if (tag != (
const char *) NULL)
912 while ((child != (XMLTreeInfo *) NULL) && (strcmp(child->tag,tag) != 0))
913 child=child->sibling;
940MagickExport
const char *GetXMLTreeContent(XMLTreeInfo *xml_info)
942 assert(xml_info != (XMLTreeInfo *) NULL);
943 assert((xml_info->signature == MagickCoreSignature) ||
944 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
945 if (IsEventLogging() != MagickFalse)
946 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
947 return(xml_info->content);
972MagickExport XMLTreeInfo *GetXMLTreeOrdered(XMLTreeInfo *xml_info)
974 assert(xml_info != (XMLTreeInfo *) NULL);
975 assert((xml_info->signature == MagickCoreSignature) ||
976 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
977 if (IsEventLogging() != MagickFalse)
978 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
979 return(xml_info->ordered);
1007MagickExport XMLTreeInfo *GetXMLTreePath(XMLTreeInfo *xml_info,
const char *path)
1011 subnode[MaxTextExtent],
1024 assert(xml_info != (XMLTreeInfo *) NULL);
1025 assert((xml_info->signature == MagickCoreSignature) ||
1026 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
1027 if (IsEventLogging() != MagickFalse)
1028 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
1030 components=GetPathComponents(path,&number_components);
1031 if (components == (
char **) NULL)
1032 return((XMLTreeInfo *) NULL);
1033 for (i=0; i < (ssize_t) number_components; i++)
1035 GetPathComponent(components[i],SubimagePath,subnode);
1036 GetPathComponent(components[i],CanonicalPath,tag);
1037 node=GetXMLTreeChild(node,tag);
1038 if (node == (XMLTreeInfo *) NULL)
1040 for (j=(ssize_t) StringToLong(subnode)-1; j > 0; j--)
1042 node=GetXMLTreeOrdered(node);
1043 if (node == (XMLTreeInfo *) NULL)
1046 if (node == (XMLTreeInfo *) NULL)
1048 components[i]=DestroyString(components[i]);
1050 for ( ; i < (ssize_t) number_components; i++)
1051 components[i]=DestroyString(components[i]);
1052 components=(
char **) RelinquishMagickMemory(components);
1080MagickExport
const char **GetXMLTreeProcessingInstructions(
1081 XMLTreeInfo *xml_info,
const char *target)
1089 assert(xml_info != (XMLTreeInfo *) NULL);
1090 assert((xml_info->signature == MagickCoreSignature) ||
1091 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
1092 if (IsEventLogging() != MagickFalse)
1093 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
1094 root=(XMLTreeRoot *) xml_info;
1095 while (root->root.parent != (XMLTreeInfo *) NULL)
1096 root=(XMLTreeRoot *) root->root.parent;
1098 while ((root->processing_instructions[i] != (
char **) NULL) &&
1099 (strcmp(root->processing_instructions[i][0],target) != 0))
1101 if (root->processing_instructions[i] == (
char **) NULL)
1102 return((
const char **) sentinel);
1103 return((
const char **) (root->processing_instructions[i]+1));
1128MagickExport XMLTreeInfo *GetXMLTreeSibling(XMLTreeInfo *xml_info)
1130 assert(xml_info != (XMLTreeInfo *) NULL);
1131 assert((xml_info->signature == MagickCoreSignature) ||
1132 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
1133 if (IsEventLogging() != MagickFalse)
1134 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
1135 return(xml_info->sibling);
1160MagickExport
const char *GetXMLTreeTag(XMLTreeInfo *xml_info)
1162 assert(xml_info != (XMLTreeInfo *) NULL);
1163 assert((xml_info->signature == MagickCoreSignature) ||
1164 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
1165 if (IsEventLogging() != MagickFalse)
1166 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
1167 return(xml_info->tag);
1198MagickExport XMLTreeInfo *InsertTagIntoXMLTree(XMLTreeInfo *xml_info,
1199 XMLTreeInfo *child,
const size_t offset)
1206 child->ordered=(XMLTreeInfo *) NULL;
1207 child->sibling=(XMLTreeInfo *) NULL;
1208 child->next=(XMLTreeInfo *) NULL;
1209 child->offset=offset;
1210 child->parent=xml_info;
1211 if (xml_info->child == (XMLTreeInfo *) NULL)
1213 xml_info->child=child;
1216 head=xml_info->child;
1217 if (head->offset > offset)
1219 child->ordered=head;
1220 xml_info->child=child;
1225 while ((node->ordered != (XMLTreeInfo *) NULL) &&
1226 (node->ordered->offset <= offset))
1228 child->ordered=node->ordered;
1229 node->ordered=child;
1231 previous=(XMLTreeInfo *) NULL;
1233 while ((node != (XMLTreeInfo *) NULL) && (strcmp(node->tag,child->tag) != 0))
1238 if ((node != (XMLTreeInfo *) NULL) && (node->offset <= offset))
1240 while ((node->next != (XMLTreeInfo *) NULL) &&
1241 (node->next->offset <= offset))
1243 child->next=node->next;
1248 if ((previous != (XMLTreeInfo *) NULL) && (node != (XMLTreeInfo *) NULL))
1249 previous->sibling=node->sibling;
1251 previous=(XMLTreeInfo *) NULL;
1253 while ((node != (XMLTreeInfo *) NULL) && (node->offset <= offset))
1258 child->sibling=node;
1259 if (previous != (XMLTreeInfo *) NULL)
1260 previous->sibling=child;
1291static char *ConvertUTF16ToUTF8(
const char *content,
size_t *length)
1309 utf8=(
char *) AcquireQuantumMemory(*length+1,
sizeof(*utf8));
1310 if (utf8 == (
char *) NULL)
1311 return((
char *) NULL);
1312 encoding=(*content ==
'\xFE') ? 1 : (*content ==
'\xFF') ? 0 : -1;
1318 (void) memcpy(utf8,content,*length*
sizeof(*utf8));
1324 for (i=2; i < (ssize_t) (*length-1); i+=2)
1326 c=(encoding != 0) ? ((content[i] & 0xff) << 8) | (content[i+1] & 0xff) :
1327 ((content[i+1] & 0xff) << 8) | (content[i] & 0xff);
1328 if ((c >= 0xd800) && (c <= 0xdfff) && ((i+=2) < (ssize_t) (*length-1)))
1330 byte=(encoding != 0) ? ((content[i] & 0xff) << 8) |
1331 (content[i+1] & 0xff) : ((content[i+1] & 0xff) << 8) |
1332 (content[i] & 0xff);
1333 c=(((c & 0x3ff) << 10) | (
byte & 0x3ff))+0x10000;
1335 if ((
size_t) (j+MaxTextExtent) > extent)
1337 extent=(size_t) j+MaxTextExtent;
1338 utf8=(
char *) ResizeQuantumMemory(utf8,extent,
sizeof(*utf8));
1339 if (utf8 == (
char *) NULL)
1352 for (bits=0;
byte != 0;
byte/=2)
1355 utf8[j++]=(0xFF << (7-bits)) | (c >> (6*bits));
1359 utf8[j]=0x80 | ((c >> (6*bits)) & 0x3f);
1364 utf8=(
char *) ResizeQuantumMemory(utf8,(*length+1),
sizeof(*utf8));
1365 if (utf8 != (
char *) NULL)
1370static char *ParseEntities(
char *xml,
char **entities,
int state)
1394 for ( ; *xml !=
'\0'; xml++)
1395 while (*xml ==
'\r')
1399 (void) memmove(xml,xml+1,strlen(xml));
1403 while ((*xml !=
'\0') && (*xml !=
'&') && ((*xml !=
'%') ||
1404 (state !=
'%')) && (isspace((
int) ((
unsigned char) *xml)) == 0))
1416 if ((state !=
'c') && (strncmp(xml,
"&#",2) == 0))
1422 c=strtol(xml+2,&entity,10);
1424 c=strtol(xml+3,&entity,16);
1425 if ((c == 0) || (*entity !=
';'))
1441 for (i=0;
byte != 0;
byte/=2)
1444 *xml=(char) ((0xFF << (7-i)) | (c >> (6*i)));
1449 *xml=(char) (0x80 | ((c >> (6*i)) & 0x3F));
1453 (void) memmove(xml,strchr(xml,
';')+1,strlen(strchr(xml,
';')));
1456 if (((*xml ==
'&') && ((state ==
'&') || (state ==
' ') ||
1457 (state ==
'*'))) || ((state ==
'%') && (*xml ==
'%')))
1463 while ((entities[i] != (
char *) NULL) &&
1464 (strncmp(xml+1,entities[i],strlen(entities[i])) != 0))
1466 if (entities[i++] == (
char *) NULL)
1469 if (entities[i] != (
char *) NULL)
1474 length=strlen(entities[i]);
1475 entity=strchr(xml,
';');
1476 if ((entity != (
char *) NULL) &&
1477 ((length-1L) >= (
size_t) (entity-xml)))
1479 offset=(ssize_t) (xml-p);
1480 extent=(size_t) (offset+length+strlen(entity));
1483 p=(
char *) ResizeQuantumMemory(p,extent+1,
sizeof(*p));
1491 extent_xml=(
char *) AcquireQuantumMemory(extent+1,
1492 sizeof(*extent_xml));
1493 if (extent_xml != (
char *) NULL)
1495 memset(extent_xml,0,extent*
sizeof(*extent_xml));
1496 (void) CopyMagickString(extent_xml,p,extent*
1497 sizeof(*extent_xml));
1501 if (p == (
char *) NULL)
1502 ThrowFatalException(ResourceLimitFatalError,
1503 "MemoryAllocationFailed");
1505 entity=strchr(xml,
';');
1507 if (entity != (
char *) NULL)
1508 (void) memmove(xml+length,entity+1,strlen(entity));
1509 (void) memcpy(xml,entities[i],length);
1513 if (((state ==
' ') || (state ==
'*')) &&
1514 (isspace((
int) ((
unsigned char) *xml) != 0)))
1525 for (xml=p; *xml !=
'\0'; xml++)
1530 i=(ssize_t) strspn(xml,accept);
1532 (void) memmove(xml,xml+i,strlen(xml+i)+1);
1533 while ((*xml !=
'\0') && (*xml !=
' '))
1539 if ((xml >= p) && (*xml ==
' '))
1542 return(p == q ? ConstantString(p) : p);
1545static void ParseCharacterContent(XMLTreeRoot *root,
char *xml,
1546 const size_t length,
const char state)
1551 xml_info=root->node;
1552 if ((xml_info == (XMLTreeInfo *) NULL) || (xml_info->tag == (
char *) NULL) ||
1556 xml=ParseEntities(xml,root->entities,state);
1557 if ((xml_info->content != (
char *) NULL) && (*xml_info->content !=
'\0'))
1559 (void) ConcatenateString(&xml_info->content,xml);
1560 xml=DestroyString(xml);
1564 if (xml_info->content != (
char *) NULL)
1565 xml_info->content=DestroyString(xml_info->content);
1566 xml_info->content=xml;
1570static XMLTreeInfo *ParseCloseTag(XMLTreeRoot *root,
char *tag,
1571 ExceptionInfo *exception)
1573 if ((root->node == (XMLTreeInfo *) NULL) ||
1574 (root->node->tag == (
char *) NULL) || (strcmp(tag,root->node->tag) != 0))
1576 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
1577 "ParseError",
"unexpected closing tag </%s>",tag);
1578 return(&root->root);
1580 root->node=root->node->parent;
1581 return((XMLTreeInfo *) NULL);
1584static MagickBooleanType ValidateEntities(
char *tag,
char *xml,
1585 const size_t depth,
char **entities)
1593 if (depth > MagickMaxRecursionDepth)
1594 return(MagickFalse);
1597 while ((*xml !=
'\0') && (*xml !=
'&'))
1601 if (strncmp(xml+1,tag,strlen(tag)) == 0)
1602 return(MagickFalse);
1604 while ((entities[i] != (
char *) NULL) &&
1605 (strncmp(entities[i],xml+1,strlen(entities[i])) == 0))
1607 if ((entities[i] != (
char *) NULL) &&
1608 (ValidateEntities(tag,entities[i+1],depth+1,entities) == 0))
1609 return(MagickFalse);
1613static void ParseProcessingInstructions(XMLTreeRoot *root,
char *xml,
1625 xml+=strcspn(xml,XMLWhitespace);
1629 xml+=strspn(xml+1,XMLWhitespace)+1;
1631 if (strcmp(target,
"xml") == 0)
1633 xml=strstr(xml,
"standalone");
1634 if ((xml != (
char *) NULL) &&
1635 (strncmp(xml+strspn(xml+10,XMLWhitespace
"='\"")+10,
"yes",3) == 0))
1636 root->standalone=MagickTrue;
1639 if (root->processing_instructions[0] == (
char **) NULL)
1641 root->processing_instructions=(
char ***) AcquireMagickMemory(
sizeof(
1642 *root->processing_instructions));
1643 if (root->processing_instructions ==(
char ***) NULL)
1644 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1645 *root->processing_instructions=(
char **) NULL;
1648 while ((root->processing_instructions[i] != (
char **) NULL) &&
1649 (strcmp(target,root->processing_instructions[i][0]) != 0))
1651 if (root->processing_instructions[i] == (
char **) NULL)
1653 root->processing_instructions=(
char ***) ResizeQuantumMemory(
1654 root->processing_instructions,(
size_t) (i+2),
1655 sizeof(*root->processing_instructions));
1656 if (root->processing_instructions == (
char ***) NULL)
1657 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1658 root->processing_instructions[i]=(
char **) AcquireQuantumMemory(3,
1659 sizeof(**root->processing_instructions));
1660 if (root->processing_instructions[i] == (
char **) NULL)
1661 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1662 root->processing_instructions[i+1]=(
char **) NULL;
1663 root->processing_instructions[i][0]=ConstantString(target);
1664 root->processing_instructions[i][1]=(
char *)
1665 root->processing_instructions[i+1];
1666 root->processing_instructions[i+1]=(
char **) NULL;
1667 root->processing_instructions[i][2]=ConstantString(
"");
1670 while (root->processing_instructions[i][j] != (
char *) NULL)
1672 root->processing_instructions[i]=(
char **) ResizeQuantumMemory(
1673 root->processing_instructions[i],(
size_t) (j+3),
1674 sizeof(**root->processing_instructions));
1675 if (root->processing_instructions[i] == (
char **) NULL)
1676 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1677 root->processing_instructions[i][j+2]=(
char *) ResizeQuantumMemory(
1678 root->processing_instructions[i][j+1],(
size_t) (j+1),
1679 sizeof(***root->processing_instructions));
1680 if (root->processing_instructions[i][j+2] == (
char *) NULL)
1681 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1682 (void) CopyMagickString(root->processing_instructions[i][j+2]+j-1,
1683 root->root.tag != (
char *) NULL ?
">" :
"<",2);
1684 root->processing_instructions[i][j]=ConstantString(xml);
1685 root->processing_instructions[i][j+1]=(
char *) NULL;
1688static MagickBooleanType ParseInternalDoctype(XMLTreeRoot *root,
char *xml,
1689 size_t length,ExceptionInfo *exception)
1695 **predefined_entities,
1705 predefined_entities=(
char **) AcquireMagickMemory(
sizeof(sentinel));
1706 if (predefined_entities == (
char **) NULL)
1707 ThrowFatalException(ResourceLimitError,
"MemoryAllocationFailed");
1708 (void) memcpy(predefined_entities,sentinel,
sizeof(sentinel));
1709 for (xml[length]=
'\0'; xml != (
char *) NULL; )
1711 while ((*xml !=
'\0') && (*xml !=
'<') && (*xml !=
'%'))
1715 if ((strlen(xml) > 9) && (strncmp(xml,
"<!ENTITY",8) == 0))
1720 if (strspn(xml+8,XMLWhitespace) == 0)
1722 xml+=strspn(xml+8,XMLWhitespace)+8;
1724 n=xml+strspn(xml,XMLWhitespace
"%");
1725 if ((isalpha((
int) ((
unsigned char) *n)) == 0) && (*n !=
'_'))
1727 xml=n+strcspn(n,XMLWhitespace);
1731 v=xml+strspn(xml+1,XMLWhitespace)+1;
1734 if ((q !=
'"') && (q !=
'\''))
1739 xml=strchr(xml,
'>');
1742 entities=(*c ==
'%') ? predefined_entities : root->entities;
1743 for (i=0; entities[i] != (
char *) NULL; i++) ;
1744 entities=(
char **) ResizeQuantumMemory(entities,(
size_t) (i+3),
1746 if (entities == (
char **) NULL)
1747 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1749 predefined_entities=entities;
1751 root->entities=entities;
1755 if (xml != (
char *) NULL)
1760 entities[i+1]=ParseEntities(v,predefined_entities,
'%');
1761 entities[i+2]=(
char *) NULL;
1762 if (ValidateEntities(n,entities[i+1],0,entities) != MagickFalse)
1766 if (entities[i+1] != v)
1767 entities[i+1]=DestroyString(entities[i+1]);
1768 (void) ThrowMagickException(exception,GetMagickModule(),
1769 OptionWarning,
"ParseError",
"circular entity declaration &%s",n);
1770 predefined_entities=(
char **) RelinquishMagickMemory(
1771 predefined_entities);
1772 return(MagickFalse);
1776 if (strncmp(xml,
"<!ATTLIST",9) == 0)
1781 t=xml+strspn(xml+9,XMLWhitespace)+9;
1784 (void) ThrowMagickException(exception,GetMagickModule(),
1785 OptionWarning,
"ParseError",
"unclosed <!ATTLIST");
1786 predefined_entities=(
char **) RelinquishMagickMemory(
1787 predefined_entities);
1788 return(MagickFalse);
1790 xml=t+strcspn(t,XMLWhitespace
">");
1795 while ((root->attributes[i] != (
char **) NULL) &&
1796 (n != (
char *) NULL) &&
1797 (strcmp(n,root->attributes[i][0]) != 0))
1799 while ((*(n=xml+strspn(xml+1,XMLWhitespace)+1) !=
'\0') &&
1802 xml=n+strcspn(n,XMLWhitespace);
1807 (void) ThrowMagickException(exception,GetMagickModule(),
1808 OptionWarning,
"ParseError",
"malformed <!ATTLIST");
1809 predefined_entities=(
char **) RelinquishMagickMemory(
1810 predefined_entities);
1811 return(MagickFalse);
1813 xml+=strspn(xml+1,XMLWhitespace)+1;
1814 c=(
char *) (strncmp(xml,
"CDATA",5) != 0 ?
"*" :
" ");
1815 if (strncmp(xml,
"NOTATION",8) == 0)
1816 xml+=strspn(xml+8,XMLWhitespace)+8;
1817 xml=(*xml ==
'(') ? strchr(xml,
')') : xml+
1818 strcspn(xml,XMLWhitespace);
1819 if (xml == (
char *) NULL)
1821 (void) ThrowMagickException(exception,GetMagickModule(),
1822 OptionWarning,
"ParseError",
"malformed <!ATTLIST");
1823 predefined_entities=(
char **) RelinquishMagickMemory(
1824 predefined_entities);
1825 return(MagickFalse);
1827 xml+=strspn(xml,XMLWhitespace
")");
1828 if (strncmp(xml,
"#FIXED",6) == 0)
1829 xml+=strspn(xml+6,XMLWhitespace)+6;
1832 xml+=strcspn(xml,XMLWhitespace
">")-1;
1838 if (((*xml ==
'"') || (*xml ==
'\'')) &&
1839 ((xml=strchr(v=xml+1,*xml)) != (
char *) NULL))
1843 (void) ThrowMagickException(exception,GetMagickModule(),
1844 OptionWarning,
"ParseError",
"malformed <!ATTLIST");
1845 predefined_entities=(
char **) RelinquishMagickMemory(
1846 predefined_entities);
1847 return(MagickFalse);
1849 if (root->attributes[i] == (
char **) NULL)
1855 root->attributes=(
char ***) AcquireQuantumMemory(2,
1856 sizeof(*root->attributes));
1858 root->attributes=(
char ***) ResizeQuantumMemory(
1859 root->attributes,(
size_t) (i+2),
1860 sizeof(*root->attributes));
1861 if (root->attributes == (
char ***) NULL)
1862 ThrowFatalException(ResourceLimitFatalError,
1863 "MemoryAllocationFailed");
1864 root->attributes[i]=(
char **) AcquireQuantumMemory(2,
1865 sizeof(**root->attributes));
1866 if (root->attributes[i] == (
char **) NULL)
1867 ThrowFatalException(ResourceLimitFatalError,
1868 "MemoryAllocationFailed");
1869 root->attributes[i][0]=ConstantString(t);
1870 root->attributes[i][1]=(
char *) NULL;
1871 root->attributes[i+1]=(
char **) NULL;
1873 for (j=1; root->attributes[i][j] != (
char *) NULL; j+=3) ;
1874 root->attributes[i]=(
char **) ResizeQuantumMemory(
1875 root->attributes[i],(
size_t) (j+4),
sizeof(**root->attributes));
1876 if (root->attributes[i] == (
char **) NULL)
1877 ThrowFatalException(ResourceLimitFatalError,
1878 "MemoryAllocationFailed");
1879 root->attributes[i][j+3]=(
char *) NULL;
1880 root->attributes[i][j+2]=ConstantString(c);
1881 root->attributes[i][j+1]=(
char *) NULL;
1882 if (v != (
char *) NULL)
1883 root->attributes[i][j+1]=ParseEntities(v,root->entities,*c);
1884 root->attributes[i][j]=ConstantString(n);
1888 if (strncmp(xml,
"<!--", 4) == 0)
1889 xml=strstr(xml+4,
"-->");
1891 if (strncmp(xml,
"<?", 2) == 0)
1895 if (xml != (
char *) NULL)
1897 ParseProcessingInstructions(root,c,(
size_t) (xml-c));
1903 xml=strchr(xml,
'>');
1905 if ((*(xml++) ==
'%') && (root->standalone == MagickFalse))
1908 predefined_entities=(
char **) RelinquishMagickMemory(predefined_entities);
1912static void ParseOpenTag(XMLTreeRoot *root,
char *tag,
char **attributes)
1917 xml_info=root->node;
1918 if (xml_info->tag == (
char *) NULL)
1919 xml_info->tag=ConstantString(tag);
1921 xml_info=AddChildToXMLTree(xml_info,tag,strlen(xml_info->content));
1922 if (xml_info != (XMLTreeInfo *) NULL)
1923 xml_info->attributes=attributes;
1924 root->node=xml_info;
1935static inline MagickBooleanType IsSkipTag(
const char *tag)
1941 while (skip_tags[i] != (
const char *) NULL)
1943 if (LocaleCompare(tag,skip_tags[i]) == 0)
1947 return(MagickFalse);
1950MagickExport XMLTreeInfo *NewXMLTree(
const char *xml,ExceptionInfo *exception)
1981 if ((xml == (
const char *) NULL) || (strlen(xml) == 0))
1983 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
1984 "ParseError",
"root tag missing");
1985 return((XMLTreeInfo *) NULL);
1987 root=(XMLTreeRoot *) NewXMLTreeTag((
char *) NULL);
1989 utf8=ConvertUTF16ToUTF8(xml,&length);
1990 if (utf8 == (
char *) NULL)
1992 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
1993 "ParseError",
"UTF16 to UTF8 failed");
1994 return((XMLTreeInfo *) NULL);
1998 utf8=DestroyString(utf8);
1999 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
2000 "ParseError",
"root tag missing");
2001 return((XMLTreeInfo *) NULL);
2003 terminal=utf8[length-1];
2004 utf8[length-1]=
'\0';
2006 while ((*p !=
'\0') && (*p !=
'<'))
2010 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
2011 "ParseError",
"root tag missing");
2012 utf8=DestroyString(utf8);
2013 return((XMLTreeInfo *) NULL);
2015 attribute=(
char **) NULL;
2020 attributes=(
char **) sentinel;
2023 if ((isalpha((
int) ((
unsigned char) *p)) !=0) || (*p ==
'_') ||
2024 (*p ==
':') || (c <
'\0'))
2029 if (root->node == (XMLTreeInfo *) NULL)
2031 (void) ThrowMagickException(exception,GetMagickModule(),
2032 OptionWarning,
"ParseError",
"root tag missing");
2033 utf8=DestroyString(utf8);
2034 return(&root->root);
2036 p+=(ptrdiff_t) strcspn(p,XMLWhitespace
"/>");
2037 while (isspace((
int) ((
unsigned char) *p)) != 0)
2039 if (((isalpha((
int) ((
unsigned char) *p)) != 0) || (*p ==
'_')) &&
2040 (ignore_depth == 0))
2042 if ((*p !=
'\0') && (*p !=
'/') && (*p !=
'>'))
2048 while ((root->attributes[i] != (
char **) NULL) &&
2049 (strcmp(root->attributes[i][0],tag) != 0))
2051 attribute=root->attributes[i];
2053 for (l=0; (*p !=
'\0') && (*p !=
'/') && (*p !=
'>'); l+=2)
2059 attributes=(
char **) AcquireQuantumMemory(4,
2060 sizeof(*attributes));
2062 attributes=(
char **) ResizeQuantumMemory(attributes,
2063 (
size_t) (l+4),
sizeof(*attributes));
2064 if (attributes == (
char **) NULL)
2066 (void) ThrowMagickException(exception,GetMagickModule(),
2067 ResourceLimitError,
"MemoryAllocationFailed",
"`%s'",
"");
2068 utf8=DestroyString(utf8);
2069 return(&root->root);
2071 attributes[l+2]=(
char *) NULL;
2072 attributes[l+1]=(
char *) NULL;
2074 p+=(ptrdiff_t) strcspn(p,XMLWhitespace
"=/>");
2075 if ((*p !=
'=') && (isspace((
int) ((
unsigned char) *p)) == 0))
2076 attributes[l]=ConstantString(
"");
2080 p+=(ptrdiff_t) strspn(p,XMLWhitespace
"=");
2082 if ((c ==
'"') || (c ==
'\''))
2089 while ((*p !=
'\0') && (*p != c))
2095 attributes[l]=ConstantString(
"");
2096 attributes[l+1]=ConstantString(
"");
2097 (void) DestroyXMLTreeAttributes(attributes);
2098 (void) ThrowMagickException(exception,
2099 GetMagickModule(),OptionWarning,
"ParseError",
2101 utf8=DestroyString(utf8);
2102 return(&root->root);
2105 while ((attribute != (
char **) NULL) &&
2106 (attribute[j] != (
char *) NULL) &&
2107 (strcmp(attribute[j],attributes[l]) != 0))
2109 attributes[l+1]=ParseEntities(attributes[l+1],
2110 root->entities,(attribute != (
char **) NULL) &&
2111 (attribute[j] != (
char *) NULL) ? *attribute[j+2] :
2114 attributes[l]=ConstantString(attributes[l]);
2116 while (isspace((
int) ((
unsigned char) *p)) != 0)
2122 while((*p !=
'\0') && (*p !=
'/') && (*p !=
'>'))
2131 if (((*p !=
'\0') && (*p !=
'>')) ||
2132 ((*p ==
'\0') && (terminal !=
'>')))
2135 (void) DestroyXMLTreeAttributes(attributes);
2136 (void) ThrowMagickException(exception,GetMagickModule(),
2137 OptionWarning,
"ParseError",
"missing >");
2138 utf8=DestroyString(utf8);
2139 return(&root->root);
2141 if ((ignore_depth != 0) || (IsSkipTag(tag) != MagickFalse))
2142 (void) DestroyXMLTreeAttributes(attributes);
2145 ParseOpenTag(root,tag,attributes);
2146 (void) ParseCloseTag(root,tag,exception);
2152 if ((*p ==
'>') || ((*p ==
'\0') && (terminal ==
'>')))
2155 if ((ignore_depth == 0) && (IsSkipTag(tag) == MagickFalse))
2156 ParseOpenTag(root,tag,attributes);
2160 (void) DestroyXMLTreeAttributes(attributes);
2167 (void) DestroyXMLTreeAttributes(attributes);
2168 (void) ThrowMagickException(exception,GetMagickModule(),
2169 OptionWarning,
"ParseError",
"missing >");
2170 utf8=DestroyString(utf8);
2171 return(&root->root);
2182 p+=(ptrdiff_t) strcspn(tag,XMLWhitespace
">")+1;
2184 if ((c ==
'\0') && (terminal !=
'>'))
2186 (void) ThrowMagickException(exception,GetMagickModule(),
2187 OptionWarning,
"ParseError",
"missing >");
2188 utf8=DestroyString(utf8);
2189 return(&root->root);
2192 if ((ignore_depth == 0) &&
2193 (ParseCloseTag(root,tag,exception) != (XMLTreeInfo *) NULL))
2195 utf8=DestroyString(utf8);
2196 return(&root->root);
2198 if (ignore_depth > 0)
2201 if (isspace((
int) ((
unsigned char) *p)) != 0)
2202 p+=(ptrdiff_t) strspn(p,XMLWhitespace);
2205 if (strncmp(p,
"!--",3) == 0)
2211 if ((p == (
char *) NULL) || ((*(p+=2) !=
'>') && (*p !=
'\0')) ||
2212 ((*p ==
'\0') && (terminal !=
'>')))
2214 (void) ThrowMagickException(exception,GetMagickModule(),
2215 OptionWarning,
"ParseError",
"unclosed <!--");
2216 utf8=DestroyString(utf8);
2217 return(&root->root);
2221 if (strncmp(p,
"![CDATA[",8) == 0)
2227 if (p != (
char *) NULL)
2230 if (ignore_depth == 0)
2231 ParseCharacterContent(root,tag+8,(
size_t) (p-tag-10),
'c');
2235 (void) ThrowMagickException(exception,GetMagickModule(),
2236 OptionWarning,
"ParseError",
"unclosed <![CDATA[");
2237 utf8=DestroyString(utf8);
2238 return(&root->root);
2242 if (strncmp(p,
"!DOCTYPE",8) == 0)
2247 for (l=0; (*p !=
'\0') && (((l == 0) && (*p !=
'>')) ||
2248 ((l != 0) && ((*p !=
']') ||
2249 (*(p+strspn(p+1,XMLWhitespace)+1) !=
'>'))));
2250 l=(ssize_t) ((*p ==
'[') ? 1 : l))
2251 p+=(ptrdiff_t) strcspn(p+1,
"[]>")+1;
2252 if ((*p ==
'\0') && (terminal !=
'>'))
2254 (void) ThrowMagickException(exception,GetMagickModule(),
2255 OptionWarning,
"ParseError",
"unclosed <!DOCTYPE");
2256 utf8=DestroyString(utf8);
2257 return(&root->root);
2260 tag=strchr(tag,
'[')+1;
2263 status=ParseInternalDoctype(root,tag,(
size_t) (p-tag),
2265 if (status == MagickFalse)
2267 utf8=DestroyString(utf8);
2268 return(&root->root);
2282 if (p == (
char *) NULL)
2285 }
while ((*p !=
'\0') && (*p !=
'>'));
2286 if ((p == (
char *) NULL) || ((*p ==
'\0') &&
2289 (void) ThrowMagickException(exception,GetMagickModule(),
2290 OptionWarning,
"ParseError",
"unclosed <?");
2291 utf8=DestroyString(utf8);
2292 return(&root->root);
2294 ParseProcessingInstructions(root,tag+1,(
size_t) (p-tag-2));
2298 (void) ThrowMagickException(exception,GetMagickModule(),
2299 OptionWarning,
"ParseError",
"unexpected <");
2300 utf8=DestroyString(utf8);
2301 return(&root->root);
2303 if ((p == (
char *) NULL) || (*p ==
'\0'))
2307 if ((*p !=
'\0') && (*p !=
'<'))
2312 while ((*p !=
'\0') && (*p !=
'<'))
2316 if (ignore_depth == 0)
2317 ParseCharacterContent(root,tag,(
size_t) (p-tag),
'&');
2323 utf8=DestroyString(utf8);
2324 if (root->node == (XMLTreeInfo *) NULL)
2325 return(&root->root);
2326 if (root->node->tag == (
char *) NULL)
2328 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
2329 "ParseError",
"root tag missing");
2330 return(&root->root);
2332 (void) ThrowMagickException(exception,GetMagickModule(),OptionWarning,
2333 "ParseError",
"unclosed tag: `%s'",root->node->tag);
2334 return(&root->root);
2359MagickExport XMLTreeInfo *NewXMLTreeTag(
const char *tag)
2362 *predefined_entities[NumberPredefinedEntities+1] =
2364 "lt;",
"<",
"gt;",
">",
"quot;",
""",
2365 "apos;",
"'",
"amp;",
"&", (
char *) NULL
2371 root=(XMLTreeRoot *) AcquireCriticalMemory(
sizeof(*root));
2372 (void) memset(root,0,
sizeof(*root));
2373 root->root.tag=(
char *) NULL;
2374 if (tag != (
char *) NULL)
2375 root->root.tag=ConstantString(tag);
2376 root->node=(&root->root);
2377 root->root.content=ConstantString(
"");
2378 root->entities=(
char **) AcquireCriticalMemory(
sizeof(predefined_entities));
2379 (void) memcpy(root->entities,predefined_entities,
sizeof(predefined_entities));
2380 root->root.attributes=sentinel;
2381 root->attributes=(
char ***) root->root.attributes;
2382 root->processing_instructions=(
char ***) root->root.attributes;
2383 root->debug=IsEventLogging();
2384 root->signature=MagickCoreSignature;
2385 return(&root->root);
2411MagickExport XMLTreeInfo *PruneTagFromXMLTree(XMLTreeInfo *xml_info)
2416 assert(xml_info != (XMLTreeInfo *) NULL);
2417 assert((xml_info->signature == MagickCoreSignature) ||
2418 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
2419 if (IsEventLogging() != MagickFalse)
2420 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
2421 if (xml_info->next != (XMLTreeInfo *) NULL)
2422 xml_info->next->sibling=xml_info->sibling;
2423 if (xml_info->parent != (XMLTreeInfo *) NULL)
2425 node=xml_info->parent->child;
2426 if (node == xml_info)
2427 xml_info->parent->child=xml_info->ordered;
2430 while (node->ordered != xml_info)
2432 node->ordered=node->ordered->ordered;
2433 node=xml_info->parent->child;
2434 if (strcmp(node->tag,xml_info->tag) != 0)
2436 while (strcmp(node->sibling->tag,xml_info->tag) != 0)
2438 if (node->sibling != xml_info)
2441 node->sibling=(xml_info->next != (XMLTreeInfo *) NULL) ?
2442 xml_info->next : node->sibling->sibling;
2444 while ((node->next != (XMLTreeInfo *) NULL) &&
2445 (node->next != xml_info))
2447 if (node->next != (XMLTreeInfo *) NULL)
2448 node->next=node->next->next;
2451 xml_info->ordered=(XMLTreeInfo *) NULL;
2452 xml_info->sibling=(XMLTreeInfo *) NULL;
2453 xml_info->next=(XMLTreeInfo *) NULL;
2485MagickExport XMLTreeInfo *SetXMLTreeAttribute(XMLTreeInfo *xml_info,
2486 const char *tag,
const char *value)
2492 assert(xml_info != (XMLTreeInfo *) NULL);
2493 assert((xml_info->signature == MagickCoreSignature) ||
2494 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
2495 if (IsEventLogging() != MagickFalse)
2496 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
2498 while ((xml_info->attributes[i] != (
char *) NULL) &&
2499 (strcmp(xml_info->attributes[i],tag) != 0))
2501 if (xml_info->attributes[i] == (
char *) NULL)
2506 if (value == (
const char *) NULL)
2508 if (xml_info->attributes != sentinel)
2509 xml_info->attributes=(
char **) ResizeQuantumMemory(
2510 xml_info->attributes,(
size_t) (i+4),
sizeof(*xml_info->attributes));
2513 xml_info->attributes=(
char **) AcquireQuantumMemory(4,
2514 sizeof(*xml_info->attributes));
2515 if (xml_info->attributes != (
char **) NULL)
2516 xml_info->attributes[1]=ConstantString(
"");
2518 if (xml_info->attributes == (
char **) NULL)
2519 ThrowFatalException(ResourceLimitFatalError,
"UnableToAcquireString");
2520 xml_info->attributes[i]=ConstantString(tag);
2521 xml_info->attributes[i+2]=(
char *) NULL;
2522 (void) strlen(xml_info->attributes[i+1]);
2527 for (j=i; xml_info->attributes[j] != (
char *) NULL; j+=2) ;
2528 if (xml_info->attributes[i+1] != (
char *) NULL)
2529 xml_info->attributes[i+1]=DestroyString(xml_info->attributes[i+1]);
2530 if (value != (
const char *) NULL)
2532 xml_info->attributes[i+1]=ConstantString(value);
2535 if (xml_info->attributes[i] != (
char *) NULL)
2536 xml_info->attributes[i]=DestroyString(xml_info->attributes[i]);
2537 (void) memmove(xml_info->attributes+i,xml_info->attributes+i+2,
2538 (
size_t) (j-i)*
sizeof(*xml_info->attributes));
2539 xml_info->attributes=(
char **) ResizeQuantumMemory(xml_info->attributes,
2540 (
size_t) (j+2),
sizeof(*xml_info->attributes));
2541 if (xml_info->attributes == (
char **) NULL)
2542 ThrowFatalException(ResourceLimitFatalError,
"UnableToAcquireString");
2544 (void) memmove(xml_info->attributes[j+1]+(i/2),xml_info->attributes[j+1]+
2545 (i/2)+1,(
size_t) (((j+2)/2)-(i/2))*
sizeof(**xml_info->attributes));
2575MagickExport XMLTreeInfo *SetXMLTreeContent(XMLTreeInfo *xml_info,
2576 const char *content)
2578 assert(xml_info != (XMLTreeInfo *) NULL);
2579 assert((xml_info->signature == MagickCoreSignature) ||
2580 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
2581 if (IsEventLogging() != MagickFalse)
2582 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
2583 if (xml_info->content != (
char *) NULL)
2584 xml_info->content=DestroyString(xml_info->content);
2585 xml_info->content=(
char *) ConstantString(content);
2612static char *EncodePredefinedEntities(
const char *source,ssize_t offset,
2613 char **destination,
size_t *length,
size_t *extent,MagickBooleanType pedantic)
2619 canonical_content=CanonicalXMLContent(source,pedantic);
2625 content=AcquireString(source);
2626 content[offset]=
'\0';
2627 canonical_content=CanonicalXMLContent(content,pedantic);
2628 content=DestroyString(content);
2630 if (canonical_content == (
char *) NULL)
2631 return(*destination);
2632 if ((*length+strlen(canonical_content)+MaxTextExtent) > *extent)
2634 *extent=(*length)+strlen(canonical_content)+MaxTextExtent;
2635 *destination=(
char *) ResizeQuantumMemory(*destination,*extent,
2636 sizeof(**destination));
2637 if (*destination == (
char *) NULL)
2638 return(*destination);
2640 *length+=FormatLocaleString(*destination+(*length),*extent,
"%s",
2642 canonical_content=DestroyString(canonical_content);
2643 return(*destination);
2646static char *XMLTreeTagToXML(XMLTreeInfo *xml_info,
char **source,
size_t *length,
2647 size_t *extent,
size_t start,
char ***attributes)
2662 content=(
char *)
"";
2663 if (xml_info->parent != (XMLTreeInfo *) NULL)
2664 content=xml_info->parent->content;
2666 *source=EncodePredefinedEntities(content+start,(ssize_t) (xml_info->offset-
2667 start),source,length,extent,MagickFalse);
2668 if ((*length+strlen(xml_info->tag)+MaxTextExtent) > *extent)
2670 *extent=(*length)+strlen(xml_info->tag)+MaxTextExtent;
2671 *source=(
char *) ResizeQuantumMemory(*source,*extent,
sizeof(**source));
2672 if (*source == (
char *) NULL)
2675 *length+=FormatLocaleString(*source+(*length),*extent,
"<%s",xml_info->tag);
2676 for (i=0; xml_info->attributes[i]; i+=2)
2678 attribute=GetXMLTreeAttribute(xml_info,xml_info->attributes[i]);
2679 if (attribute != xml_info->attributes[i+1])
2681 if ((*length+strlen(xml_info->attributes[i])+MaxTextExtent) > *extent)
2683 *extent=(*length)+strlen(xml_info->attributes[i])+MaxTextExtent;
2684 *source=(
char *) ResizeQuantumMemory(*source,*extent,
sizeof(**source));
2685 if (*source == (
char *) NULL)
2686 return((
char *) NULL);
2688 *length+=FormatLocaleString(*source+(*length),*extent,
" %s=\"",
2689 xml_info->attributes[i]);
2690 (void) EncodePredefinedEntities(xml_info->attributes[i+1],-1,source,length,
2692 *length+=FormatLocaleString(*source+(*length),*extent,
"\"");
2695 while ((attributes[i] != (
char **) NULL) &&
2696 (strcmp(attributes[i][0],xml_info->tag) != 0))
2699 while ((attributes[i] != (
char **) NULL) &&
2700 (attributes[i][j] != (
char *) NULL))
2702 if ((attributes[i][j+1] == (
char *) NULL) ||
2703 (GetXMLTreeAttribute(xml_info,attributes[i][j]) != attributes[i][j+1]))
2708 if ((*length+strlen(attributes[i][j])+MaxTextExtent) > *extent)
2710 *extent=(*length)+strlen(attributes[i][j])+MaxTextExtent;
2711 *source=(
char *) ResizeQuantumMemory(*source,*extent,
sizeof(**source));
2712 if (*source == (
char *) NULL)
2713 return((
char *) NULL);
2715 *length+=FormatLocaleString(*source+(*length),*extent,
" %s=\"",
2717 (void) EncodePredefinedEntities(attributes[i][j+1],-1,source,length,extent,
2719 *length+=FormatLocaleString(*source+(*length),*extent,
"\"");
2722 *length+=FormatLocaleString(*source+(*length),*extent,*xml_info->content ?
2724 if (xml_info->child != (XMLTreeInfo *) NULL)
2725 *source=XMLTreeTagToXML(xml_info->child,source,length,extent,0,attributes);
2727 *source=EncodePredefinedEntities(xml_info->content,-1,source,length,extent,
2729 if ((*length+strlen(xml_info->tag)+MaxTextExtent) > *extent)
2731 *extent=(*length)+strlen(xml_info->tag)+MaxTextExtent;
2732 *source=(
char *) ResizeQuantumMemory(*source,*extent,
sizeof(**source));
2733 if (*source == (
char *) NULL)
2734 return((
char *) NULL);
2736 if (*xml_info->content !=
'\0')
2737 *length+=FormatLocaleString(*source+(*length),*extent,
"</%s>",
2739 while ((offset < xml_info->offset) && (content[offset] !=
'\0'))
2741 if (xml_info->ordered != (XMLTreeInfo *) NULL)
2742 content=XMLTreeTagToXML(xml_info->ordered,source,length,extent,offset,
2745 content=EncodePredefinedEntities(content+offset,-1,source,length,extent,
2750MagickExport
char *XMLTreeInfoToXML(XMLTreeInfo *xml_info)
2773 assert(xml_info != (XMLTreeInfo *) NULL);
2774 assert((xml_info->signature == MagickCoreSignature) ||
2775 (((XMLTreeRoot *) xml_info)->signature == MagickCoreSignature));
2776 if (IsEventLogging() != MagickFalse)
2777 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"...");
2778 if (xml_info->tag == (
char *) NULL)
2779 return((
char *) NULL);
2780 xml=AcquireString((
char *) NULL);
2782 extent=MaxTextExtent;
2783 root=(XMLTreeRoot *) xml_info;
2784 while (root->root.parent != (XMLTreeInfo *) NULL)
2785 root=(XMLTreeRoot *) root->root.parent;
2786 parent=xml_info->parent;
2787 if (parent == (XMLTreeInfo *) NULL)
2788 for (i=0; root->processing_instructions[i] != (
char **) NULL; i++)
2793 for (k=2; root->processing_instructions[i][k-1]; k++) ;
2794 p=root->processing_instructions[i][1];
2795 for (j=1; p != (
char *) NULL; j++)
2797 if (root->processing_instructions[i][k][j-1] ==
'>')
2799 p=root->processing_instructions[i][j];
2802 q=root->processing_instructions[i][0];
2803 if ((length+strlen(p)+strlen(q)+MaxTextExtent) > extent)
2805 extent=length+strlen(p)+strlen(q)+MaxTextExtent;
2806 xml=(
char *) ResizeQuantumMemory(xml,extent,
sizeof(*xml));
2807 if (xml == (
char *) NULL)
2810 length+=FormatLocaleString(xml+length,extent,
"<?%s%s%s?>\n",q,
2811 *p !=
'\0' ?
" " :
"",p);
2812 p=root->processing_instructions[i][j];
2815 ordered=xml_info->ordered;
2816 xml_info->parent=(XMLTreeInfo *) NULL;
2817 xml_info->ordered=(XMLTreeInfo *) NULL;
2818 xml=XMLTreeTagToXML(xml_info,&xml,&length,&extent,0,root->attributes);
2819 xml_info->parent=parent;
2820 xml_info->ordered=ordered;
2821 if (parent == (XMLTreeInfo *) NULL)
2822 for (i=0; root->processing_instructions[i] != (
char **) NULL; i++)
2827 for (k=2; root->processing_instructions[i][k-1]; k++) ;
2828 p=root->processing_instructions[i][1];
2829 for (j=1; p != (
char *) NULL; j++)
2831 if (root->processing_instructions[i][k][j-1] ==
'<')
2833 p=root->processing_instructions[i][j];
2836 q=root->processing_instructions[i][0];
2837 if ((length+strlen(p)+strlen(q)+MaxTextExtent) > extent)
2839 extent=length+strlen(p)+strlen(q)+MaxTextExtent;
2840 xml=(
char *) ResizeQuantumMemory(xml,extent,
sizeof(*xml));
2841 if (xml == (
char *) NULL)
2844 length+=FormatLocaleString(xml+length,extent,
"\n<?%s%s%s?>",q,
2845 *p !=
'\0' ?
" " :
"",p);
2846 p=root->processing_instructions[i][j];
2849 return((
char *) ResizeQuantumMemory(xml,length+1,
sizeof(*xml)));