]> granicus.if.org Git - imagemagick/blob - MagickCore/policy.c
(no commit message)
[imagemagick] / MagickCore / policy.c
1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 %                                                                             %
4 %                                                                             %
5 %                  PPPP    OOO   L      IIIII   CCCC  Y   Y                   %
6 %                  P   P  O   O  L        I    C       Y Y                    %
7 %                  PPPP   O   O  L        I    C        Y                     %
8 %                  P      O   O  L        I    C        Y                     %
9 %                  P       OOO   LLLLL  IIIII   CCCC    Y                     %
10 %                                                                             %
11 %                                                                             %
12 %                         MagickCore Policy Methods                           %
13 %                                                                             %
14 %                              Software Design                                %
15 %                                   Cristy                                    %
16 %                                 July 1992                                   %
17 %                                                                             %
18 %                                                                             %
19 %  Copyright 1999-2014 ImageMagick Studio LLC, a non-profit organization      %
20 %  dedicated to making software imaging solutions freely available.           %
21 %                                                                             %
22 %  You may not use this file except in compliance with the License.  You may  %
23 %  obtain a copy of the License at                                            %
24 %                                                                             %
25 %    http://www.imagemagick.org/script/license.php                            %
26 %                                                                             %
27 %  Unless required by applicable law or agreed to in writing, software        %
28 %  distributed under the License is distributed on an "AS IS" BASIS,          %
29 %  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   %
30 %  See the License for the specific language governing permissions and        %
31 %  limitations under the License.                                             %
32 %                                                                             %
33 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
34 %
35 %  We use linked-lists because splay-trees do not currently support duplicate
36 %  key / value pairs (.e.g X11 green compliance and SVG green compliance).
37 %
38 */
39 \f
40 /*
41   Include declarations.
42 */
43 #include "MagickCore/studio.h"
44 #include "MagickCore/client.h"
45 #include "MagickCore/configure.h"
46 #include "MagickCore/configure-private.h"
47 #include "MagickCore/exception.h"
48 #include "MagickCore/exception-private.h"
49 #include "MagickCore/memory_.h"
50 #include "MagickCore/monitor.h"
51 #include "MagickCore/monitor-private.h"
52 #include "MagickCore/option.h"
53 #include "MagickCore/policy.h"
54 #include "MagickCore/policy-private.h"
55 #include "MagickCore/semaphore.h"
56 #include "MagickCore/string_.h"
57 #include "MagickCore/token.h"
58 #include "MagickCore/utility.h"
59 #include "MagickCore/utility-private.h"
60 #include "MagickCore/xml-tree.h"
61 \f
62 /*
63   Define declarations.
64 */
65 #define PolicyFilename  "policy.xml"
66 \f
67 /*
68   Typedef declarations.
69 */
70 struct _PolicyInfo
71 {
72   char
73     *path;
74
75   PolicyDomain
76     domain;
77
78   PolicyRights
79     rights;
80
81   char
82     *name,
83     *pattern,
84     *value;
85
86   MagickBooleanType
87     exempt,
88     stealth,
89     debug;
90
91   SemaphoreInfo
92     *semaphore;
93
94   size_t
95     signature;
96 };
97
98 typedef struct _PolicyMapInfo
99 {
100   const PolicyDomain
101     domain;
102
103   const PolicyRights
104     rights;
105
106   const char
107     *name,
108     *pattern,
109     *value;
110 } PolicyMapInfo;
111 \f
112 /*
113   Static declarations.
114 */
115 static const PolicyMapInfo
116   PolicyMap[] =
117   {
118     { UndefinedPolicyDomain, UndefinedPolicyRights, (const char *) NULL,
119       (const char *) NULL, (const char *) NULL }
120   };
121
122 static LinkedListInfo
123   *policy_list = (LinkedListInfo *) NULL;
124
125 static SemaphoreInfo
126   *policy_semaphore = (SemaphoreInfo *) NULL;
127
128 static volatile MagickBooleanType
129   instantiate_policy = MagickFalse;
130 \f
131 /*
132   Forward declarations.
133 */
134 static MagickBooleanType
135   InitializePolicyList(ExceptionInfo *),
136   LoadPolicyLists(const char *,ExceptionInfo *);
137 \f
138 /*
139 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
140 %                                                                             %
141 %                                                                             %
142 %                                                                             %
143 +   G e t P o l i c y I n f o                                                 %
144 %                                                                             %
145 %                                                                             %
146 %                                                                             %
147 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
148 %
149 %  GetPolicyInfo() searches the policy list for the specified name and if found
150 %  returns attributes for that policy.
151 %
152 %  The format of the GetPolicyInfo method is:
153 %
154 %      PolicyInfo *GetPolicyInfo(const char *name,ExceptionInfo *exception)
155 %
156 %  A description of each parameter follows:
157 %
158 %    o name: the policy name.
159 %
160 %    o exception: return any errors or warnings in this structure.
161 %
162 */
163 static PolicyInfo *GetPolicyInfo(const char *name,ExceptionInfo *exception)
164 {
165   char
166     policyname[MaxTextExtent];
167
168   register PolicyInfo
169     *p;
170
171   register char
172     *q;
173
174   assert(exception != (ExceptionInfo *) NULL);
175   if ((policy_list == (LinkedListInfo *) NULL) ||
176       (instantiate_policy == MagickFalse))
177     if (InitializePolicyList(exception) == MagickFalse)
178       return((PolicyInfo *) NULL);
179   if ((policy_list == (LinkedListInfo *) NULL) ||
180       (IsLinkedListEmpty(policy_list) != MagickFalse))
181     return((PolicyInfo *) NULL);
182   if ((name == (const char *) NULL) || (LocaleCompare(name,"*") == 0))
183     return((PolicyInfo *) GetValueFromLinkedList(policy_list,0));
184   /*
185     Strip names of whitespace.
186   */
187   (void) CopyMagickString(policyname,name,MaxTextExtent);
188   for (q=policyname; *q != '\0'; q++)
189   {
190     if (isspace((int) ((unsigned char) *q)) == 0)
191       continue;
192     (void) CopyMagickString(q,q+1,MaxTextExtent);
193     q--;
194   }
195   /*
196     Search for policy tag.
197   */
198   LockSemaphoreInfo(policy_semaphore);
199   ResetLinkedListIterator(policy_list);
200   p=(PolicyInfo *) GetNextValueInLinkedList(policy_list);
201   while (p != (PolicyInfo *) NULL)
202   {
203     if (LocaleCompare(policyname,p->name) == 0)
204       break;
205     p=(PolicyInfo *) GetNextValueInLinkedList(policy_list);
206   }
207   if (p != (PolicyInfo *) NULL)
208     (void) InsertValueInLinkedList(policy_list,0,
209       RemoveElementByValueFromLinkedList(policy_list,p));
210   UnlockSemaphoreInfo(policy_semaphore);
211   return(p);
212 }
213 \f
214 /*
215 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
216 %                                                                             %
217 %                                                                             %
218 %                                                                             %
219 %   G e t P o l i c y I n f o L i s t                                         %
220 %                                                                             %
221 %                                                                             %
222 %                                                                             %
223 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
224 %
225 %  GetPolicyInfoList() returns any policies that match the specified pattern.
226 %
227 %  The format of the GetPolicyInfoList function is:
228 %
229 %      const PolicyInfo **GetPolicyInfoList(const char *pattern,
230 %        size_t *number_policies,ExceptionInfo *exception)
231 %
232 %  A description of each parameter follows:
233 %
234 %    o pattern: Specifies a pointer to a text string containing a pattern.
235 %
236 %    o number_policies:  returns the number of policies in the list.
237 %
238 %    o exception: return any errors or warnings in this structure.
239 %
240 */
241 MagickExport const PolicyInfo **GetPolicyInfoList(const char *pattern,
242   size_t *number_policies,ExceptionInfo *exception)
243 {
244   const PolicyInfo
245     **policies;
246
247   register const PolicyInfo
248     *p;
249
250   register ssize_t
251     i;
252
253   /*
254     Allocate policy list.
255   */
256   assert(pattern != (char *) NULL);
257   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern);
258   assert(number_policies != (size_t *) NULL);
259   *number_policies=0;
260   p=GetPolicyInfo("*",exception);
261   if (p == (const PolicyInfo *) NULL)
262     return((const PolicyInfo **) NULL);
263   policies=(const PolicyInfo **) AcquireQuantumMemory((size_t)
264     GetNumberOfElementsInLinkedList(policy_list)+1UL,sizeof(*policies));
265   if (policies == (const PolicyInfo **) NULL)
266     return((const PolicyInfo **) NULL);
267   /*
268     Generate policy list.
269   */
270   LockSemaphoreInfo(policy_semaphore);
271   ResetLinkedListIterator(policy_list);
272   p=(const PolicyInfo *) GetNextValueInLinkedList(policy_list);
273   for (i=0; p != (const PolicyInfo *) NULL; )
274   {
275     if ((p->stealth == MagickFalse) &&
276         (GlobExpression(p->name,pattern,MagickFalse) != MagickFalse))
277       policies[i++]=p;
278     p=(const PolicyInfo *) GetNextValueInLinkedList(policy_list);
279   }
280   UnlockSemaphoreInfo(policy_semaphore);
281   policies[i]=(PolicyInfo *) NULL;
282   *number_policies=(size_t) i;
283   return(policies);
284 }
285 \f
286 /*
287 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
288 %                                                                             %
289 %                                                                             %
290 %                                                                             %
291 %   G e t P o l i c y L i s t                                                 %
292 %                                                                             %
293 %                                                                             %
294 %                                                                             %
295 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
296 %
297 %  GetPolicyList() returns any policies that match the specified pattern.
298 %
299 %  The format of the GetPolicyList function is:
300 %
301 %      char **GetPolicyList(const char *pattern,size_t *number_policies,
302 %        ExceptionInfo *exception)
303 %
304 %  A description of each parameter follows:
305 %
306 %    o pattern: a pointer to a text string containing a pattern.
307 %
308 %    o number_policies:  returns the number of policies in the list.
309 %
310 %    o exception: return any errors or warnings in this structure.
311 %
312 */
313 MagickExport char **GetPolicyList(const char *pattern,
314   size_t *number_policies,ExceptionInfo *exception)
315 {
316   char
317     **policies;
318
319   register const PolicyInfo
320     *p;
321
322   register ssize_t
323     i;
324
325   /*
326     Allocate policy list.
327   */
328   assert(pattern != (char *) NULL);
329   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern);
330   assert(number_policies != (size_t *) NULL);
331   *number_policies=0;
332   p=GetPolicyInfo("*",exception);
333   if (p == (const PolicyInfo *) NULL)
334     return((char **) NULL);
335   policies=(char **) AcquireQuantumMemory((size_t)
336     GetNumberOfElementsInLinkedList(policy_list)+1UL,sizeof(*policies));
337   if (policies == (char **) NULL)
338     return((char **) NULL);
339   /*
340     Generate policy list.
341   */
342   LockSemaphoreInfo(policy_semaphore);
343   ResetLinkedListIterator(policy_list);
344   p=(const PolicyInfo *) GetNextValueInLinkedList(policy_list);
345   for (i=0; p != (const PolicyInfo *) NULL; )
346   {
347     if ((p->stealth == MagickFalse) &&
348         (GlobExpression(p->name,pattern,MagickFalse) != MagickFalse))
349       policies[i++]=ConstantString(p->name);
350     p=(const PolicyInfo *) GetNextValueInLinkedList(policy_list);
351   }
352   UnlockSemaphoreInfo(policy_semaphore);
353   policies[i]=(char *) NULL;
354   *number_policies=(size_t) i;
355   return(policies);
356 }
357 \f
358 /*
359 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
360 %                                                                             %
361 %                                                                             %
362 %                                                                             %
363 %   G e t P o l i c y V a l u e                                               %
364 %                                                                             %
365 %                                                                             %
366 %                                                                             %
367 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
368 %
369 %  GetPolicyValue() returns the value associated with the named policy.
370 %
371 %  The format of the GetPolicyValue method is:
372 %
373 %      char *GetPolicyValue(const char *name)
374 %
375 %  A description of each parameter follows:
376 %
377 %    o policy_info:  The policy info.
378 %
379 */
380 MagickExport char *GetPolicyValue(const char *name)
381 {
382   const char
383     *value;
384
385   const PolicyInfo
386     *policy_info;
387
388   ExceptionInfo
389     *exception;
390
391   assert(name != (const char *) NULL);
392   (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",name);
393   exception=AcquireExceptionInfo();
394   policy_info=GetPolicyInfo(name,exception);
395   exception=DestroyExceptionInfo(exception);
396   if (policy_info == (PolicyInfo *) NULL)
397     return((char *) NULL);
398   value=policy_info->value;
399   if ((value == (const char *) NULL) || (*value == '\0'))
400     return((char *) NULL);
401   return(ConstantString(value));
402 }
403 \f
404 /*
405 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
406 %                                                                             %
407 %                                                                             %
408 %                                                                             %
409 +   I n i t i a l i z e P o l i c y L i s t                                   %
410 %                                                                             %
411 %                                                                             %
412 %                                                                             %
413 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
414 %
415 %  InitializePolicyList() initializes the policy list.
416 %
417 %  The format of the InitializePolicyList method is:
418 %
419 %      MagickBooleanType InitializePolicyList(ExceptionInfo *exception)
420 %
421 %  A description of each parameter follows.
422 %
423 %    o exception: return any errors or warnings in this structure.
424 %
425 */
426 static MagickBooleanType InitializePolicyList(ExceptionInfo *exception)
427 {
428   if ((policy_list == (LinkedListInfo *) NULL) &&
429       (instantiate_policy == MagickFalse))
430     {
431       if (policy_semaphore == (SemaphoreInfo *) NULL)
432         AcquireSemaphoreInfo(&policy_semaphore);
433       LockSemaphoreInfo(policy_semaphore);
434       if ((policy_list == (LinkedListInfo *) NULL) &&
435           (instantiate_policy == MagickFalse))
436         {
437           (void) LoadPolicyLists(PolicyFilename,exception);
438           instantiate_policy=MagickTrue;
439         }
440       UnlockSemaphoreInfo(policy_semaphore);
441     }
442   return(policy_list != (LinkedListInfo *) NULL ? MagickTrue : MagickFalse);
443 }
444 \f
445 /*
446 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
447 %                                                                             %
448 %                                                                             %
449 %                                                                             %
450 %   I s R i g h t s A u t h o r i z e d                                       %
451 %                                                                             %
452 %                                                                             %
453 %                                                                             %
454 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
455 %
456 %  IsRightsAuthorized() returns MagickTrue if the policy authorizes the
457 %  requested rights for the specified domain.
458 %
459 %  The format of the IsRightsAuthorized method is:
460 %
461 %      MagickBooleanType IsRightsAuthorized(const PolicyDomain domain,
462 %        const PolicyRights rights,const char *pattern)
463 %
464 %  A description of each parameter follows:
465 %
466 %    o domain: the policy domain.
467 %
468 %    o rights: the policy rights.
469 %
470 %    o pattern: the coder, delegate, filter, or path pattern.
471 %
472 */
473 MagickExport MagickBooleanType IsRightsAuthorized(const PolicyDomain domain,
474   const PolicyRights rights,const char *pattern)
475 {
476   const PolicyInfo
477     *policy_info;
478
479   ExceptionInfo
480     *exception;
481
482   MagickBooleanType
483     authorized;
484
485   register PolicyInfo
486     *p;
487
488   (void) LogMagickEvent(PolicyEvent,GetMagickModule(),
489     "Domain: %s; rights=%s; pattern=\"%s\" ...",
490     CommandOptionToMnemonic(MagickPolicyDomainOptions,domain),
491     CommandOptionToMnemonic(MagickPolicyRightsOptions,rights),pattern);
492   exception=AcquireExceptionInfo();
493   policy_info=GetPolicyInfo("*",exception);
494   exception=DestroyExceptionInfo(exception);
495   if (policy_info == (PolicyInfo *) NULL)
496     return(MagickTrue);
497   authorized=MagickTrue;
498   LockSemaphoreInfo(policy_semaphore);
499   ResetLinkedListIterator(policy_list);
500   p=(PolicyInfo *) GetNextValueInLinkedList(policy_list);
501   while ((p != (PolicyInfo *) NULL) && (authorized != MagickFalse))
502   {
503     if ((p->domain == domain) &&
504         (GlobExpression(pattern,p->pattern,MagickFalse) != MagickFalse))
505       {
506         if (((rights & ReadPolicyRights) != 0) &&
507             ((p->rights & ReadPolicyRights) == 0))
508           authorized=MagickFalse;
509         if (((rights & WritePolicyRights) != 0) &&
510             ((p->rights & WritePolicyRights) == 0))
511           authorized=MagickFalse;
512         if (((rights & ExecutePolicyRights) != 0) &&
513             ((p->rights & ExecutePolicyRights) == 0))
514           authorized=MagickFalse;
515       }
516     p=(PolicyInfo *) GetNextValueInLinkedList(policy_list);
517   }
518   UnlockSemaphoreInfo(policy_semaphore);
519   return(authorized);
520 }
521 \f
522 /*
523 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
524 %                                                                             %
525 %                                                                             %
526 %                                                                             %
527 %  L i s t P o l i c y I n f o                                                %
528 %                                                                             %
529 %                                                                             %
530 %                                                                             %
531 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
532 %
533 %  ListPolicyInfo() lists policies to the specified file.
534 %
535 %  The format of the ListPolicyInfo method is:
536 %
537 %      MagickBooleanType ListPolicyInfo(FILE *file,ExceptionInfo *exception)
538 %
539 %  A description of each parameter follows.
540 %
541 %    o file:  List policy names to this file handle.
542 %
543 %    o exception: return any errors or warnings in this structure.
544 %
545 */
546 MagickExport MagickBooleanType ListPolicyInfo(FILE *file,
547   ExceptionInfo *exception)
548 {
549   const char
550     *path,
551     *domain;
552
553   const PolicyInfo
554     **policy_info;
555
556   register ssize_t
557     i;
558
559   size_t
560     number_policies;
561
562   /*
563     List name and attributes of each policy in the list.
564   */
565   if (file == (const FILE *) NULL)
566     file=stdout;
567   policy_info=GetPolicyInfoList("*",&number_policies,exception);
568   if (policy_info == (const PolicyInfo **) NULL)
569     return(MagickFalse);
570   path=(const char *) NULL;
571   for (i=0; i < (ssize_t) number_policies; i++)
572   {
573     if (policy_info[i]->stealth != MagickFalse)
574       continue;
575     if (((path == (const char *) NULL) ||
576          (LocaleCompare(path,policy_info[i]->path) != 0)) &&
577          (policy_info[i]->path != (char *) NULL))
578       (void) FormatLocaleFile(file,"\nPath: %s\n",policy_info[i]->path);
579     path=policy_info[i]->path;
580     domain=CommandOptionToMnemonic(MagickPolicyDomainOptions,
581       policy_info[i]->domain);
582     (void) FormatLocaleFile(file,"  Policy: %s\n",domain);
583     if ((policy_info[i]->domain == ResourcePolicyDomain) ||
584         (policy_info[i]->domain == SystemPolicyDomain))
585       {
586         if (policy_info[i]->name != (char *) NULL)
587           (void) FormatLocaleFile(file,"    name: %s\n",policy_info[i]->name);
588         if (policy_info[i]->value != (char *) NULL)
589           (void) FormatLocaleFile(file,"    value: %s\n",policy_info[i]->value);
590       }
591     else
592       {
593         (void) FormatLocaleFile(file,"    rights: ");
594         if (policy_info[i]->rights == NoPolicyRights)
595           (void) FormatLocaleFile(file,"None ");
596         if ((policy_info[i]->rights & ReadPolicyRights) != 0)
597           (void) FormatLocaleFile(file,"Read ");
598         if ((policy_info[i]->rights & WritePolicyRights) != 0)
599           (void) FormatLocaleFile(file,"Write ");
600         if ((policy_info[i]->rights & ExecutePolicyRights) != 0)
601           (void) FormatLocaleFile(file,"Execute ");
602         (void) FormatLocaleFile(file,"\n");
603         if (policy_info[i]->pattern != (char *) NULL)
604           (void) FormatLocaleFile(file,"    pattern: %s\n",
605             policy_info[i]->pattern);
606       }
607   }
608   policy_info=(const PolicyInfo **) RelinquishMagickMemory((void *)
609     policy_info);
610   (void) fflush(file);
611   return(MagickTrue);
612 }
613 \f
614 /*
615 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
616 %                                                                             %
617 %                                                                             %
618 %                                                                             %
619 +   L o a d P o l i c y L i s t                                               %
620 %                                                                             %
621 %                                                                             %
622 %                                                                             %
623 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
624 %
625 %  LoadPolicyList() loads the policy configuration file which provides a mapping
626 %  between policy attributes and a policy domain.
627 %
628 %  The format of the LoadPolicyList method is:
629 %
630 %      MagickBooleanType LoadPolicyList(const char *xml,const char *filename,
631 %        const size_t depth,ExceptionInfo *exception)
632 %
633 %  A description of each parameter follows:
634 %
635 %    o xml:  The policy list in XML format.
636 %
637 %    o filename:  The policy list filename.
638 %
639 %    o depth: depth of <include /> statements.
640 %
641 %    o exception: return any errors or warnings in this structure.
642 %
643 */
644 static MagickBooleanType LoadPolicyList(const char *xml,const char *filename,
645   const size_t depth,ExceptionInfo *exception)
646 {
647   char
648     keyword[MaxTextExtent],
649     *token;
650
651   PolicyInfo
652     *policy_info;
653
654   const char
655     *q;
656
657   MagickBooleanType
658     status;
659
660   /*
661     Load the policy map file.
662   */
663   (void) LogMagickEvent(ConfigureEvent,GetMagickModule(),
664     "Loading policy file \"%s\" ...",filename);
665   if (xml == (char *) NULL)
666     return(MagickFalse);
667   if (policy_list == (LinkedListInfo *) NULL)
668     {
669       policy_list=NewLinkedList(0);
670       if (policy_list == (LinkedListInfo *) NULL)
671         {
672           ThrowFileException(exception,ResourceLimitError,
673             "MemoryAllocationFailed",filename);
674           return(MagickFalse);
675         }
676     }
677   status=MagickTrue;
678   policy_info=(PolicyInfo *) NULL;
679   token=AcquireString(xml);
680   for (q=(const char *) xml; *q != '\0'; )
681   {
682     /*
683       Interpret XML.
684     */
685     GetMagickToken(q,&q,token);
686     if (*token == '\0')
687       break;
688     (void) CopyMagickString(keyword,token,MaxTextExtent);
689     if (LocaleNCompare(keyword,"<!DOCTYPE",9) == 0)
690       {
691         /*
692           Docdomain element.
693         */
694         while ((LocaleNCompare(q,"]>",2) != 0) && (*q != '\0'))
695           GetMagickToken(q,&q,token);
696         continue;
697       }
698     if (LocaleNCompare(keyword,"<!--",4) == 0)
699       {
700         /*
701           Comment element.
702         */
703         while ((LocaleNCompare(q,"->",2) != 0) && (*q != '\0'))
704           GetMagickToken(q,&q,token);
705         continue;
706       }
707     if (LocaleCompare(keyword,"<include") == 0)
708       {
709         /*
710           Include element.
711         */
712         while (((*token != '/') && (*(token+1) != '>')) && (*q != '\0'))
713         {
714           (void) CopyMagickString(keyword,token,MaxTextExtent);
715           GetMagickToken(q,&q,token);
716           if (*token != '=')
717             continue;
718           GetMagickToken(q,&q,token);
719           if (LocaleCompare(keyword,"file") == 0)
720             {
721               if (depth > 200)
722                 (void) ThrowMagickException(exception,GetMagickModule(),
723                   ConfigureError,"IncludeElementNestedTooDeeply","`%s'",token);
724               else
725                 {
726                   char
727                     path[MaxTextExtent],
728                     *xml;
729
730                   GetPathComponent(filename,HeadPath,path);
731                   if (*path != '\0')
732                     (void) ConcatenateMagickString(path,DirectorySeparator,
733                       MaxTextExtent);
734                   if (*token == *DirectorySeparator)
735                     (void) CopyMagickString(path,token,MaxTextExtent);
736                   else
737                     (void) ConcatenateMagickString(path,token,MaxTextExtent);
738                   xml=FileToString(path,~0UL,exception);
739                   if (xml != (char *) NULL)
740                     {
741                       status=LoadPolicyList(xml,path,depth+1,exception);
742                       xml=(char *) RelinquishMagickMemory(xml);
743                     }
744                 }
745             }
746         }
747         continue;
748       }
749     if (LocaleCompare(keyword,"<policy") == 0)
750       {
751         /*
752           Policy element.
753         */
754         policy_info=(PolicyInfo *) AcquireMagickMemory(sizeof(*policy_info));
755         if (policy_info == (PolicyInfo *) NULL)
756           ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
757         (void) ResetMagickMemory(policy_info,0,sizeof(*policy_info));
758         policy_info->path=ConstantString(filename);
759         policy_info->exempt=MagickFalse;
760         policy_info->signature=MagickSignature;
761         continue;
762       }
763     if (policy_info == (PolicyInfo *) NULL)
764       continue;
765     if (LocaleCompare(keyword,"/>") == 0)
766       {
767         status=AppendValueToLinkedList(policy_list,policy_info);
768         if (status == MagickFalse)
769           (void) ThrowMagickException(exception,GetMagickModule(),
770             ResourceLimitError,"MemoryAllocationFailed","`%s'",
771             policy_info->name);
772         policy_info=(PolicyInfo *) NULL;
773         continue;
774       }
775     GetMagickToken(q,(const char **) NULL,token);
776     if (*token != '=')
777       continue;
778     GetMagickToken(q,&q,token);
779     GetMagickToken(q,&q,token);
780     switch (*keyword)
781     {
782       case 'D':
783       case 'd':
784       {
785         if (LocaleCompare((char *) keyword,"domain") == 0)
786           {
787             policy_info->domain=(PolicyDomain) ParseCommandOption(
788               MagickPolicyDomainOptions,MagickTrue,token);
789             break;
790           }
791         break;
792       }
793       case 'N':
794       case 'n':
795       {
796         if (LocaleCompare((char *) keyword,"name") == 0)
797           {
798             policy_info->name=ConstantString(token);
799             break;
800           }
801         break;
802       }
803       case 'P':
804       case 'p':
805       {
806         if (LocaleCompare((char *) keyword,"pattern") == 0)
807           {
808             policy_info->pattern=ConstantString(token);
809             break;
810           }
811         break;
812       }
813       case 'R':
814       case 'r':
815       {
816         if (LocaleCompare((char *) keyword,"rights") == 0)
817           {
818             policy_info->rights=(PolicyRights) ParseCommandOption(
819               MagickPolicyRightsOptions,MagickTrue,token);
820             break;
821           }
822         break;
823       }
824       case 'S':
825       case 's':
826       {
827         if (LocaleCompare((char *) keyword,"stealth") == 0)
828           {
829             policy_info->stealth=IsStringTrue(token);
830             break;
831           }
832         break;
833       }
834       case 'V':
835       case 'v':
836       {
837         if (LocaleCompare((char *) keyword,"value") == 0)
838           {
839             policy_info->value=ConstantString(token);
840             break;
841           }
842         break;
843       }
844       default:
845         break;
846     }
847   }
848   token=(char *) RelinquishMagickMemory(token);
849   return(status);
850 }
851 \f
852 /*
853 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
854 %                                                                             %
855 %                                                                             %
856 %                                                                             %
857 %  L o a d P o l i c y L i s t s                                              %
858 %                                                                             %
859 %                                                                             %
860 %                                                                             %
861 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
862 %
863 %  LoadPolicyList() loads one or more policy configuration file which provides a
864 %  mapping between policy attributes and a policy name.
865 %
866 %  The format of the LoadPolicyLists method is:
867 %
868 %      MagickBooleanType LoadPolicyLists(const char *filename,
869 %        ExceptionInfo *exception)
870 %
871 %  A description of each parameter follows:
872 %
873 %    o filename: the font file name.
874 %
875 %    o exception: return any errors or warnings in this structure.
876 %
877 */
878 static MagickBooleanType LoadPolicyLists(const char *filename,
879   ExceptionInfo *exception)
880 {
881   const StringInfo
882     *option;
883
884   LinkedListInfo
885     *options;
886
887   MagickStatusType
888     status;
889
890   register ssize_t
891     i;
892
893   /*
894     Load external policy map.
895   */
896   if (policy_list == (LinkedListInfo *) NULL)
897     {
898       policy_list=NewLinkedList(0);
899       if (policy_list == (LinkedListInfo *) NULL)
900         {
901           ThrowFileException(exception,ResourceLimitError,
902             "MemoryAllocationFailed",filename);
903           return(MagickFalse);
904         }
905     }
906   status=MagickTrue;
907   options=GetConfigureOptions(filename,exception);
908   option=(const StringInfo *) GetNextValueInLinkedList(options);
909   while (option != (const StringInfo *) NULL)
910   {
911     status&=LoadPolicyList((const char *) GetStringInfoDatum(option),
912       GetStringInfoPath(option),0,exception);
913     option=(const StringInfo *) GetNextValueInLinkedList(options);
914   }
915   options=DestroyConfigureOptions(options);
916   /*
917     Load built-in policy map.
918   */
919   for (i=0; i < (ssize_t) (sizeof(PolicyMap)/sizeof(*PolicyMap)); i++)
920   {
921     PolicyInfo
922       *policy_info;
923
924     register const PolicyMapInfo
925       *p;
926
927     p=PolicyMap+i;
928     policy_info=(PolicyInfo *) AcquireMagickMemory(sizeof(*policy_info));
929     if (policy_info == (PolicyInfo *) NULL)
930       {
931         (void) ThrowMagickException(exception,GetMagickModule(),
932           ResourceLimitError,"MemoryAllocationFailed","`%s'",p->name);
933         continue;
934       }
935     (void) ResetMagickMemory(policy_info,0,sizeof(*policy_info));
936     policy_info->path=(char *) "[built-in]";
937     policy_info->domain=p->domain;
938     policy_info->rights=p->rights;
939     policy_info->name=(char *) p->name;
940     policy_info->pattern=(char *) p->pattern;
941     policy_info->value=(char *) p->value;
942     policy_info->exempt=MagickTrue;
943     policy_info->signature=MagickSignature;
944     status&=AppendValueToLinkedList(policy_list,policy_info);
945     if (status == MagickFalse)
946       (void) ThrowMagickException(exception,GetMagickModule(),
947         ResourceLimitError,"MemoryAllocationFailed","`%s'",policy_info->name);
948   }
949   return(status != 0 ? MagickTrue : MagickFalse);
950 }
951 \f
952 /*
953 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
954 %                                                                             %
955 %                                                                             %
956 %                                                                             %
957 +   P o l i c y C o m p o n e n t G e n e s i s                               %
958 %                                                                             %
959 %                                                                             %
960 %                                                                             %
961 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
962 %
963 %  PolicyComponentGenesis() instantiates the policy component.
964 %
965 %  The format of the PolicyComponentGenesis method is:
966 %
967 %      MagickBooleanType PolicyComponentGenesis(void)
968 %
969 */
970 MagickPrivate MagickBooleanType PolicyComponentGenesis(void)
971 {
972   AcquireSemaphoreInfo(&policy_semaphore);
973   return(MagickTrue);
974 }
975 \f
976 /*
977 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
978 %                                                                             %
979 %                                                                             %
980 %                                                                             %
981 +   P o l i c y C o m p o n e n t T e r m i n u s                             %
982 %                                                                             %
983 %                                                                             %
984 %                                                                             %
985 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
986 %
987 %  PolicyComponentTerminus() destroys the policy component.
988 %
989 %  The format of the PolicyComponentTerminus method is:
990 %
991 %      PolicyComponentTerminus(void)
992 %
993 */
994
995 static void *DestroyPolicyElement(void *policy_info)
996 {
997   register PolicyInfo
998     *p;
999
1000   p=(PolicyInfo *) policy_info;
1001   if (p->exempt == MagickFalse)
1002     {
1003       if (p->value != (char *) NULL)
1004         p->value=DestroyString(p->value);
1005       if (p->pattern != (char *) NULL)
1006         p->pattern=DestroyString(p->pattern);
1007       if (p->name != (char *) NULL)
1008         p->name=DestroyString(p->name);
1009       if (p->path != (char *) NULL)
1010         p->path=DestroyString(p->path);
1011     }
1012   p=(PolicyInfo *) RelinquishMagickMemory(p);
1013   return((void *) NULL);
1014 }
1015
1016 MagickPrivate void PolicyComponentTerminus(void)
1017 {
1018   if (policy_semaphore == (SemaphoreInfo *) NULL)
1019     AcquireSemaphoreInfo(&policy_semaphore);
1020   LockSemaphoreInfo(policy_semaphore);
1021   if (policy_list != (LinkedListInfo *) NULL)
1022     policy_list=DestroyLinkedList(policy_list,DestroyPolicyElement);
1023   instantiate_policy=MagickFalse;
1024   UnlockSemaphoreInfo(policy_semaphore);
1025   DestroySemaphoreInfo(&policy_semaphore);
1026 }