44#include "MagickCore/studio.h"
45#include "MagickCore/accelerate-private.h"
46#include "MagickCore/annotate.h"
47#include "MagickCore/artifact.h"
48#include "MagickCore/attribute.h"
49#include "MagickCore/cache.h"
50#include "MagickCore/cache-view.h"
51#include "MagickCore/channel.h"
52#include "MagickCore/color.h"
53#include "MagickCore/color-private.h"
54#include "MagickCore/colorspace-private.h"
55#include "MagickCore/composite.h"
56#include "MagickCore/decorate.h"
57#include "MagickCore/distort.h"
58#include "MagickCore/draw.h"
59#include "MagickCore/effect.h"
60#include "MagickCore/enhance.h"
61#include "MagickCore/exception.h"
62#include "MagickCore/exception-private.h"
63#include "MagickCore/fx.h"
64#include "MagickCore/fx-private.h"
65#include "MagickCore/gem.h"
66#include "MagickCore/gem-private.h"
67#include "MagickCore/geometry.h"
68#include "MagickCore/layer.h"
69#include "MagickCore/list.h"
70#include "MagickCore/log.h"
71#include "MagickCore/image.h"
72#include "MagickCore/image-private.h"
73#include "MagickCore/magick.h"
74#include "MagickCore/memory_.h"
75#include "MagickCore/memory-private.h"
76#include "MagickCore/monitor.h"
77#include "MagickCore/monitor-private.h"
78#include "MagickCore/option.h"
79#include "MagickCore/pixel.h"
80#include "MagickCore/pixel-accessor.h"
81#include "MagickCore/policy.h"
82#include "MagickCore/property.h"
83#include "MagickCore/quantum.h"
84#include "MagickCore/quantum-private.h"
85#include "MagickCore/random_.h"
86#include "MagickCore/random-private.h"
87#include "MagickCore/resample.h"
88#include "MagickCore/resample-private.h"
89#include "MagickCore/resize.h"
90#include "MagickCore/resource_.h"
91#include "MagickCore/splay-tree.h"
92#include "MagickCore/statistic.h"
93#include "MagickCore/statistic-private.h"
94#include "MagickCore/string_.h"
95#include "MagickCore/thread-private.h"
96#include "MagickCore/threshold.h"
97#include "MagickCore/timer-private.h"
98#include "MagickCore/token.h"
99#include "MagickCore/transform.h"
100#include "MagickCore/transform-private.h"
101#include "MagickCore/utility.h"
104#define MaxTokenLen 100
106#define TableExtend 0.1
107#define InitNumOprStack 50
108#define MinValStackSize 100
109#define InitNumUserSymbols 50
111typedef long double fxFltType;
167 {oAddEq,
"+=", 12, 1},
168 {oSubtractEq,
"-=", 12, 1},
169 {oMultiplyEq,
"*=", 13, 1},
170 {oDivideEq,
"/=", 13, 1},
171 {oPlusPlus,
"++", 12, 0},
172 {oSubSub,
"--", 12, 0},
174 {oSubtract,
"-", 12, 2},
175 {oMultiply,
"*", 13, 2},
176 {oDivide,
"/", 13, 2},
177 {oModulus,
"%", 13, 2},
178 {oUnaryPlus,
"+", 14, 1},
179 {oUnaryMinus,
"-", 14, 1},
180 {oLshift,
"<<", 11, 2},
181 {oRshift,
">>", 11, 2},
183 {oNotEq,
"!=", 9, 2},
184 {oLtEq,
"<=", 10, 2},
185 {oGtEq,
">=", 10, 2},
188 {oLogAnd,
"&&", 6, 2},
189 {oLogOr,
"||", 5, 2},
190 {oLogNot,
"!", 16, 1},
191 {oBitAnd,
"&", 8, 2},
193 {oBitNot,
"~", 16, 1},
197 {oOpenParen,
"(", 0, 0},
198 {oCloseParen,
")", 0, 0},
199 {oOpenBracket,
"[", 0, 0},
200 {oCloseBracket,
"]", 0, 0},
201 {oOpenBrace,
"{", 0, 0},
202 {oCloseBrace,
"}", 0, 0},
203 {oAssign,
"=", 3, 1},
204 {oNull,
"onull", 17, 0}
232 {cEpsilon, MagickEpsilon,
"epsilon"},
233 {cE, 2.7182818284590452354,
"e"},
234 {cOpaque, 1.0,
"opaque"},
235 {cPhi, MagickPHI,
"phi"},
236 {cPi, MagickPI,
"pi"},
237 {cQuantumRange, QuantumRange,
"quantumrange"},
238 {cQuantumScale, QuantumScale,
"quantumscale"},
239 {cTransparent, 0.0,
"transparent"},
240 {cMaxRgb, QuantumRange,
"MaxRGB"},
241 {cNull, 0.0,
"cnull"}
244#define FirstFunc ((FunctionE) (oNull+1))
248#if defined(MAGICKCORE_HAVE_ACOSH)
252#if defined(MAGICKCORE_HAVE_J1)
256#if defined(MAGICKCORE_HAVE_ASINH)
260#if defined(MAGICKCORE_HAVE_ATANH)
272#if defined(MAGICKCORE_HAVE_ERF)
282#if defined(MAGICKCORE_HAVE_J0)
285#if defined(MAGICKCORE_HAVE_J1)
288#if defined(MAGICKCORE_HAVE_J1)
339#if defined(MAGICKCORE_HAVE_ACOSH)
340 {fAcosh,
"acosh" , 1},
343#if defined(MAGICKCORE_HAVE_J1)
347#if defined(MAGICKCORE_HAVE_ASINH)
348 {fAsinh,
"asinh" , 1},
351#if defined(MAGICKCORE_HAVE_ATANH)
352 {fAtanh,
"atanh" , 1},
354 {fAtan2,
"atan2" , 2},
357 {fChannel,
"channel", 5},
358 {fClamp,
"clamp" , 1},
361 {fDebug,
"debug" , 1},
363#if defined(MAGICKCORE_HAVE_ERF)
367 {fFloor,
"floor" , 1},
368 {fGauss,
"gauss" , 1},
370 {fHypot,
"hypot" , 2},
372 {fIsnan,
"isnan" , 1},
373#if defined(MAGICKCORE_HAVE_J0)
376#if defined(MAGICKCORE_HAVE_J1)
379#if defined(MAGICKCORE_HAVE_J1)
383 {fLogtwo,
"logtwo", 1},
391 {fRound,
"round" , 1},
397 {fSquish,
"squish", 1},
400 {fTrunc,
"trunc" , 1},
404 {fWhile,
"while", 2},
417#define FirstImgAttr ((ImgAttrE) (fNull+1))
461 {aDepth,
"depth", MagickTrue},
462 {aExtent,
"extent", MagickFalse},
463 {aKurtosis,
"kurtosis", MagickTrue},
464 {aMaxima,
"maxima", MagickTrue},
465 {aMean,
"mean", MagickTrue},
466 {aMedian,
"median", MagickTrue},
467 {aMinima,
"minima", MagickTrue},
468 {aPage,
"page", MagickFalse},
469 {aPageX,
"page.x", MagickFalse},
470 {aPageY,
"page.y", MagickFalse},
471 {aPageWid,
"page.width", MagickFalse},
472 {aPageHt,
"page.height", MagickFalse},
473 {aPrintsize,
"printsize", MagickFalse},
474 {aPrintsizeX,
"printsize.x", MagickFalse},
475 {aPrintsizeY,
"printsize.y", MagickFalse},
476 {aQuality,
"quality", MagickFalse},
477 {aRes,
"resolution", MagickFalse},
478 {aResX,
"resolution.x", MagickFalse},
479 {aResY,
"resolution.y", MagickFalse},
480 {aSkewness,
"skewness", MagickTrue},
481 {aStdDev,
"standard_deviation", MagickTrue},
482 {aH,
"h", MagickFalse},
483 {aN,
"n", MagickFalse},
484 {aT,
"t", MagickFalse},
485 {aW,
"w", MagickFalse},
486 {aZ,
"z", MagickFalse},
487 {aNull,
"anull", MagickFalse},
488 {aNull,
"anull", MagickFalse},
489 {aNull,
"anull", MagickFalse},
490 {aNull,
"anull", MagickFalse}
493#define FirstSym ((SymbolE) (aNull+1))
524static const SymbolT Symbols[] = {
526 {sIntensity,
"intensity"},
527 {sLightness,
"lightness"},
529 {sLuminance,
"luminance"},
530 {sSaturation,
"saturation"},
551#define FirstCont (sNull+1)
578 {rGotoChk,
"gotochk", 0},
579 {rIfZeroGoto,
"ifzerogoto", 1},
580 {rIfNotZeroGoto,
"ifnotzerogoto", 1},
581 {rCopyFrom,
"copyfrom", 0},
582 {rCopyTo,
"copyto", 1},
583 {rZerStk,
"zerstk", 0},
587#define NULL_ADDRESS -2
603#define NO_CHAN_QUAL ((PixelChannel) (-1))
604#define THIS_CHANNEL ((PixelChannel) (-2))
605#define HUE_CHANNEL ((PixelChannel) (-3))
606#define SAT_CHANNEL ((PixelChannel) (-4))
607#define LIGHT_CHANNEL ((PixelChannel) (-5))
608#define INTENSITY_CHANNEL ((PixelChannel) (-6))
611 {
"r", RedPixelChannel},
612 {
"g", GreenPixelChannel},
613 {
"b", BluePixelChannel},
614 {
"c", CyanPixelChannel},
615 {
"m", MagentaPixelChannel},
616 {
"y", YellowPixelChannel},
617 {
"k", BlackPixelChannel},
618 {
"a", AlphaPixelChannel},
619 {
"o", AlphaPixelChannel},
620 {
"hue", HUE_CHANNEL},
621 {
"saturation", SAT_CHANNEL},
622 {
"lightness", LIGHT_CHANNEL},
623 {
"intensity", INTENSITY_CHANNEL},
624 {
"all", CompositePixelChannel},
625 {
"this", THIS_CHANNEL},
649static const char * sElementTypes[] = {
706 fxFltType * ValStack;
707 fxFltType * UserSymVals;
715 MagickBooleanType NeedStats;
716 MagickBooleanType GotStats;
717 MagickBooleanType NeedHsl;
718 MagickBooleanType DebugOpt;
719 MagickBooleanType ContainsDebug;
722 char ShortExp[MagickPathExtent];
724 char token[MagickPathExtent];
735 OperatorE * OperatorStack;
741 **magick_restrict random_infos;
753static MagickBooleanType TranslateStatementList
754 (
FxInfo * pfx,
const char * strLimit,
char * chLimit);
756static MagickBooleanType TranslateExpression
757 (
FxInfo * pfx,
const char * strLimit,
char * chLimit, MagickBooleanType * needPopAll);
759static MagickBooleanType GetFunction (
FxInfo * pfx, FunctionE fe);
761static inline MagickBooleanType ChanIsVirtual (PixelChannel pc)
763 if (pc==HUE_CHANNEL || pc==SAT_CHANNEL || pc==LIGHT_CHANNEL || pc==INTENSITY_CHANNEL)
769static MagickBooleanType InitFx (
FxInfo * pfx,
const Image * img,
775 pfx->ImgListLen = GetImageListLength (img);
776 pfx->ImgNum = GetImageIndexInList (img);
777 pfx->image = (
Image *)img;
779 pfx->NeedStats = MagickFalse;
780 pfx->GotStats = MagickFalse;
781 pfx->NeedHsl = MagickFalse;
782 pfx->DebugOpt = IsStringTrue (GetImageArtifact (img,
"fx:debug"));
783 pfx->statistics = NULL;
786 pfx->exception = exception;
787 pfx->precision = GetMagickPrecision ();
788 pfx->random_infos = AcquireRandomInfoTLS ();
789 pfx->ContainsDebug = MagickFalse;
790 pfx->runType = (CalcAllStats) ? rtEntireImage : rtCornerOnly;
791 pfx->Imgs = (
ImgT *)AcquireQuantumMemory (pfx->ImgListLen, sizeof (
ImgT));
793 (void) ThrowMagickException (
794 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
796 (
unsigned long) pfx->ImgListLen);
800 next = GetFirstImageInList (img);
801 for ( ; next != (
Image *) NULL; next=next->next)
803 ImgT * pimg = &pfx->Imgs[i];
804 pimg->View = AcquireVirtualCacheView (next, pfx->exception);
806 (void) ThrowMagickException (
807 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
811 for ( ; i > 0; i--) {
812 pimg = &pfx->Imgs[i-1];
813 pimg->View = DestroyCacheView (pimg->View);
815 pfx->Imgs=(
ImgT *) RelinquishMagickMemory (pfx->Imgs);
821 pfx->Images = ImageListToArray (img, pfx->exception);
826static MagickBooleanType DeInitFx (
FxInfo * pfx)
830 if (pfx->Images) pfx->Images = (
Image**) RelinquishMagickMemory (pfx->Images);
833 for (i = (ssize_t)GetImageListLength(pfx->image); i > 0; i--) {
834 ImgT * pimg = &pfx->Imgs[i-1];
835 pimg->View = DestroyCacheView (pimg->View);
837 pfx->Imgs=(
ImgT *) RelinquishMagickMemory (pfx->Imgs);
839 pfx->random_infos = DestroyRandomInfoTLS (pfx->random_infos);
841 if (pfx->statistics) {
842 for (i = (ssize_t)GetImageListLength(pfx->image); i > 0; i--) {
843 pfx->statistics[i-1]=(
ChannelStatistics *) RelinquishMagickMemory (pfx->statistics[i-1]);
852static ElementTypeE TypeOfOpr (
int op)
854 if (op < oNull)
return etOperator;
855 if (op == oNull)
return etConstant;
856 if (op <= fNull)
return etFunction;
857 if (op <= aNull)
return etImgAttr;
858 if (op <= sNull)
return etSymbol;
859 if (op <= rNull)
return etControl;
861 return (ElementTypeE) 0;
864static char * SetPtrShortExp (
FxInfo * pfx,
char * pExp,
size_t len)
871 *pfx->ShortExp =
'\0';
874 slen = CopyMagickString (pfx->ShortExp, pExp, len);
876 (void) CopyMagickString (pfx->ShortExp+MaxLen,
"...", 4);
878 p = strchr (pfx->ShortExp,
'\n');
879 if (p) (void) CopyMagickString (p,
"...", 4);
880 p = strchr (pfx->ShortExp,
'\r');
881 if (p) (void) CopyMagickString (p,
"...", 4);
883 return pfx->ShortExp;
886static char * SetShortExp (
FxInfo * pfx)
888 return SetPtrShortExp (pfx, pfx->pex, MaxTokenLen-1);
891static int FindUserSymbol (
FxInfo * pfx,
char * name)
898 lenName = strlen (name);
899 for (i=0; i < pfx->usedUserSymbols; i++) {
901 if (lenName == pus->len && LocaleNCompare (name, pus->pex, lenName)==0)
break;
903 if (i == pfx->usedUserSymbols)
return NULL_ADDRESS;
907static MagickBooleanType ExtendUserSymbols (
FxInfo * pfx)
909 pfx->numUserSymbols = (int) ceil (pfx->numUserSymbols * (1 + TableExtend));
910 pfx->UserSymbols = (
UserSymbolT*) ResizeMagickMemory (pfx->UserSymbols, (
size_t) pfx->numUserSymbols *
sizeof(
UserSymbolT));
911 if (!pfx->UserSymbols) {
912 (void) ThrowMagickException (
913 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
915 pfx->numUserSymbols);
922static int AddUserSymbol (
FxInfo * pfx,
char * pex,
size_t len)
925 if (++pfx->usedUserSymbols >= pfx->numUserSymbols) {
926 if (!ExtendUserSymbols (pfx))
return -1;
928 pus = &pfx->UserSymbols[pfx->usedUserSymbols-1];
932 return pfx->usedUserSymbols-1;
935static void DumpTables (FILE * fh)
939 for (i=0; i <= rNull; i++) {
940 const char * str =
"";
941 if ( i < oNull) str = Operators[i].str;
942 if (i >= (
int) FirstFunc && i < fNull) str = Functions[i-(int) FirstFunc].str;
943 if (i >= (
int) FirstImgAttr && i < aNull) str = ImgAttrs[i-(
int) FirstImgAttr].str;
944 if (i >= (
int) FirstSym && i < sNull) str = Symbols[i-(
int) FirstSym].str;
945 if (i >= (
int) FirstCont && i < rNull) str = Controls[i-(
int) FirstCont].str;
946 if (i==0 ) fprintf (stderr,
"Operators:\n ");
947 else if (i==oNull) fprintf (stderr,
"\nFunctions:\n ");
948 else if (i==fNull) fprintf (stderr,
"\nImage attributes:\n ");
949 else if (i==aNull) fprintf (stderr,
"\nSymbols:\n ");
950 else if (i==sNull) fprintf (stderr,
"\nControls:\n ");
951 fprintf (fh,
" %s", str);
956static char * NameOfUserSym (
FxInfo * pfx,
int ndx,
char * buf)
959 assert (ndx >= 0 && ndx < pfx->usedUserSymbols);
960 pus = &pfx->UserSymbols[ndx];
961 (void) CopyMagickString (buf, pus->pex, pus->len+1);
965static void DumpUserSymbols (
FxInfo * pfx, FILE * fh)
967 char UserSym[MagickPathExtent];
969 fprintf (fh,
"UserSymbols (%i)\n", pfx->usedUserSymbols);
970 for (i=0; i < pfx->usedUserSymbols; i++) {
971 fprintf (fh,
" %i: '%s'\n", i, NameOfUserSym (pfx, i, UserSym));
975static MagickBooleanType BuildRPN (
FxInfo * pfx)
977 pfx->numUserSymbols = InitNumUserSymbols;
978 pfx->usedUserSymbols = 0;
979 pfx->UserSymbols = (
UserSymbolT*) AcquireMagickMemory ((
size_t) pfx->numUserSymbols *
sizeof(
UserSymbolT));
980 if (!pfx->UserSymbols) {
981 (void) ThrowMagickException (
982 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
984 pfx->numUserSymbols);
988 pfx->numElements = RpnInit;
989 pfx->usedElements = 0;
990 pfx->Elements = NULL;
992 pfx->Elements = (
ElementT*) AcquireMagickMemory ((
size_t) pfx->numElements *
sizeof(
ElementT));
994 if (!pfx->Elements) {
995 (void) ThrowMagickException (
996 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1002 pfx->usedOprStack = 0;
1003 pfx->maxUsedOprStack = 0;
1004 pfx->numOprStack = InitNumOprStack;
1005 pfx->OperatorStack = (OperatorE*) AcquireMagickMemory ((
size_t) pfx->numOprStack *
sizeof(OperatorE));
1006 if (!pfx->OperatorStack) {
1007 (void) ThrowMagickException (
1008 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1009 "OperatorStack",
"%i",
1017static MagickBooleanType AllocFxRt (
FxInfo * pfx,
fxRtT * pfxrt)
1021 pfxrt->random_info = AcquireRandomInfo ();
1022 pfxrt->thisPixel = NULL;
1024 nRnd = 20 + 10 * (int) GetPseudoRandomValue (pfxrt->random_info);
1025 for (i=0; i < nRnd; i++) (
void) GetPseudoRandomValue (pfxrt->random_info);;
1027 pfxrt->usedValStack = 0;
1028 pfxrt->numValStack = 2 * pfx->maxUsedOprStack;
1029 if (pfxrt->numValStack < MinValStackSize) pfxrt->numValStack = MinValStackSize;
1030 pfxrt->ValStack = (fxFltType*) AcquireMagickMemory ((
size_t) pfxrt->numValStack *
sizeof(fxFltType));
1031 if (!pfxrt->ValStack) {
1032 (void) ThrowMagickException (
1033 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1035 pfxrt->numValStack);
1039 pfxrt->UserSymVals = NULL;
1041 if (pfx->usedUserSymbols) {
1042 pfxrt->UserSymVals = (fxFltType*) AcquireMagickMemory ((
size_t) pfx->usedUserSymbols *
sizeof(fxFltType));
1043 if (!pfxrt->UserSymVals) {
1044 (void) ThrowMagickException (
1045 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1046 "UserSymVals",
"%i",
1047 pfx->usedUserSymbols);
1050 for (i = 0; i < pfx->usedUserSymbols; i++) pfxrt->UserSymVals[i] = (fxFltType) 0;
1056static MagickBooleanType ExtendRPN (
FxInfo * pfx)
1058 pfx->numElements = (int) ceil (pfx->numElements * (1 + TableExtend));
1059 pfx->Elements = (
ElementT*) ResizeMagickMemory (pfx->Elements, (
size_t) pfx->numElements *
sizeof(
ElementT));
1060 if (!pfx->Elements) {
1061 (void) ThrowMagickException (
1062 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1070static inline MagickBooleanType OprInPlace (
int op)
1072 return (op >= oAddEq && op <= oSubSub ? MagickTrue : MagickFalse);
1075static const char * OprStr (
int oprNum)
1078 if (oprNum < 0) str =
"bad OprStr";
1079 else if (oprNum <= oNull) str = Operators[oprNum].str;
1080 else if (oprNum <= fNull) str = Functions[oprNum-(int) FirstFunc].str;
1081 else if (oprNum <= aNull) str = ImgAttrs[oprNum-(int) FirstImgAttr].str;
1082 else if (oprNum <= sNull) str = Symbols[oprNum-(int) FirstSym].str;
1083 else if (oprNum <= rNull) str = Controls[oprNum-(int) FirstCont].str;
1090static MagickBooleanType DumpRPN (
FxInfo * pfx, FILE * fh)
1094 fprintf (fh,
"DumpRPN:");
1095 fprintf (fh,
" numElements=%i", pfx->numElements);
1096 fprintf (fh,
" usedElements=%i", pfx->usedElements);
1097 fprintf (fh,
" maxUsedOprStack=%i", pfx->maxUsedOprStack);
1098 fprintf (fh,
" ImgListLen=%g", (
double) pfx->ImgListLen);
1099 fprintf (fh,
" NeedStats=%s", pfx->NeedStats ?
"yes" :
"no");
1100 fprintf (fh,
" GotStats=%s", pfx->GotStats ?
"yes" :
"no");
1101 fprintf (fh,
" NeedHsl=%s\n", pfx->NeedHsl ?
"yes" :
"no");
1102 if (pfx->runType==rtEntireImage) fprintf (stderr,
"EntireImage");
1103 else if (pfx->runType==rtCornerOnly) fprintf (stderr,
"CornerOnly");
1107 for (i=0; i < pfx->usedElements; i++) {
1108 ElementT * pel = &pfx->Elements[i];
1109 pel->number_dest = 0;
1111 for (i=0; i < pfx->usedElements; i++) {
1112 ElementT * pel = &pfx->Elements[i];
1113 if (pel->operator_index == rGoto || pel->operator_index == rGotoChk || pel->operator_index == rIfZeroGoto || pel->operator_index == rIfNotZeroGoto) {
1114 if (pel->element_index >= 0 && pel->element_index < pfx->numElements) {
1115 ElementT * pelDest = &pfx->Elements[pel->element_index];
1116 pelDest->number_dest++;
1120 for (i=0; i < pfx->usedElements; i++) {
1121 char UserSym[MagickPathExtent];
1123 ElementT * pel = &pfx->Elements[i];
1124 const char * str = OprStr (pel->operator_index);
1125 const char *sRelAbs =
"";
1127 if (pel->operator_index == fP || pel->operator_index == fUP || pel->operator_index == fVP || pel->operator_index == fSP)
1128 sRelAbs = pel->is_relative ?
"[]" :
"{}";
1130 if (pel->type == etColourConstant)
1131 fprintf (fh,
" %i: %s vals=%.*Lg,%.*Lg,%.*Lg '%s%s' nArgs=%i ndx=%i %s",
1132 i, sElementTypes[pel->type],
1133 pfx->precision, pel->val, pfx->precision, pel->val1, pfx->precision, pel->val2,
1134 str, sRelAbs, pel->number_args, pel->element_index,
1135 pel->do_push ?
"push" :
"NO push");
1137 fprintf (fh,
" %i: %s val=%.*Lg '%s%s' nArgs=%i ndx=%i %s",
1138 i, sElementTypes[pel->type], pfx->precision, pel->val, str, sRelAbs,
1139 pel->number_args, pel->element_index,
1140 pel->do_push ?
"push" :
"NO push");
1142 if (pel->img_attr_qual != aNull)
1143 fprintf (fh,
" ia=%s", OprStr((
int) pel->img_attr_qual));
1145 if (pel->channel_qual != NO_CHAN_QUAL) {
1146 if (pel->channel_qual == THIS_CHANNEL) fprintf (stderr,
" ch=this");
1147 else fprintf (stderr,
" ch=%i", pel->channel_qual);
1150 if (pel->operator_index == rCopyTo) {
1151 fprintf (fh,
" CopyTo ==> %s", NameOfUserSym (pfx, pel->element_index, UserSym));
1152 }
else if (pel->operator_index == rCopyFrom) {
1153 fprintf (fh,
" CopyFrom <== %s", NameOfUserSym (pfx, pel->element_index, UserSym));
1154 }
else if (OprInPlace (pel->operator_index)) {
1155 fprintf (fh,
" <==> %s", NameOfUserSym (pfx, pel->element_index, UserSym));
1157 if (pel->number_dest > 0) fprintf (fh,
" <==dest(%i)", pel->number_dest);
1163static void DestroyRPN (
FxInfo * pfx)
1165 pfx->numOprStack = 0;
1166 pfx->usedOprStack = 0;
1167 if (pfx->OperatorStack) pfx->OperatorStack = (OperatorE*) RelinquishMagickMemory (pfx->OperatorStack);
1169 pfx->numElements = 0;
1170 pfx->usedElements = 0;
1171 if (pfx->Elements) pfx->Elements = (
ElementT*) RelinquishMagickMemory (pfx->Elements);
1173 pfx->usedUserSymbols = 0;
1174 if (pfx->UserSymbols) pfx->UserSymbols = (
UserSymbolT*) RelinquishMagickMemory (pfx->UserSymbols);
1177static void DestroyFxRt (
fxRtT * pfxrt)
1179 pfxrt->usedValStack = 0;
1180 if (pfxrt->ValStack) pfxrt->ValStack = (fxFltType*) RelinquishMagickMemory (pfxrt->ValStack);
1181 if (pfxrt->UserSymVals) pfxrt->UserSymVals = (fxFltType*) RelinquishMagickMemory (pfxrt->UserSymVals);
1183 pfxrt->random_info = DestroyRandomInfo (pfxrt->random_info);
1186static size_t GetToken (
FxInfo * pfx)
1197 char * p = pfx->pex;
1201 if (!isalpha((
int)*p))
return 0;
1208 if (LocaleNCompare (p,
"icc-", 4) == 0) {
1211 while (isalpha ((
int)*p)) { len++; p++; }
1212 }
else if (LocaleNCompare (p,
"device-", 7) == 0) {
1215 while (isalpha ((
int)*p)) { len++; p++; }
1217 while (isalpha ((
int)*p)) { len++; p++; }
1218 if (*p ==
'_') { len++; p++; }
1219 while (isalpha ((
int)*p)) { len++; p++; }
1220 while (isdigit ((
int)*p)) { len++; p++; }
1222 if (len >= MaxTokenLen) {
1223 (void) ThrowMagickException (
1224 pfx->exception, GetMagickModule(), OptionError,
1225 "GetToken: too long",
"%g at '%s'",
1226 (double) len, SetShortExp(pfx));
1230 (void) CopyMagickString (pfx->token, pfx->pex, (len+1<MaxTokenLen)?len+1:MaxTokenLen);
1233 pfx->lenToken = strlen (pfx->token);
1237static MagickBooleanType TokenMaybeUserSymbol (
FxInfo * pfx)
1239 char * p = pfx->token;
1242 if (!isalpha ((
int)*p++))
return MagickFalse;
1245 if (i < 2)
return MagickFalse;
1249static MagickBooleanType AddElement (
FxInfo * pfx, fxFltType val,
int oprNum)
1253 assert (oprNum <= rNull);
1255 if (++pfx->usedElements >= pfx->numElements) {
1256 if (!ExtendRPN (pfx))
return MagickFalse;
1259 pel = &pfx->Elements[pfx->usedElements-1];
1260 pel->type = TypeOfOpr (oprNum);
1262 pel->val1 = (fxFltType) 0;
1263 pel->val2 = (fxFltType) 0;
1264 pel->operator_index = oprNum;
1265 pel->do_push = MagickTrue;
1266 pel->element_index = 0;
1267 pel->channel_qual = NO_CHAN_QUAL;
1268 pel->img_attr_qual = aNull;
1269 pel->number_dest = 0;
1270 pel->exp_start = NULL;
1273 if (oprNum <= oNull) pel->number_args = Operators[oprNum].number_args;
1274 else if (oprNum <= fNull) pel->number_args = Functions[oprNum-(int) FirstFunc].number_args;
1275 else if (oprNum <= aNull) pel->number_args = 0;
1276 else if (oprNum <= sNull) pel->number_args = 0;
1277 else pel->number_args = Controls[oprNum-(int) FirstCont].number_args;
1282static MagickBooleanType AddAddressingElement (
FxInfo * pfx,
int oprNum,
int EleNdx)
1285 if (!AddElement (pfx, (fxFltType) 0, oprNum))
return MagickFalse;
1286 pel = &pfx->Elements[pfx->usedElements-1];
1287 pel->element_index = EleNdx;
1288 if (oprNum == rGoto || oprNum == rGotoChk || oprNum == rIfZeroGoto || oprNum == rIfNotZeroGoto
1289 || oprNum == rZerStk)
1291 pel->do_push = MagickFalse;
1301static MagickBooleanType AddColourElement (
FxInfo * pfx, fxFltType val0, fxFltType val1, fxFltType val2)
1304 if (!AddElement (pfx, val0, oNull))
return MagickFalse;
1305 pel = &pfx->Elements[pfx->usedElements-1];
1308 pel->type = etColourConstant;
1312static inline void SkipSpaces (
FxInfo * pfx)
1314 while (isspace ((
int)*pfx->pex)) pfx->pex++;
1317static inline char PeekChar (
FxInfo * pfx)
1323static inline MagickBooleanType PeekStr (
FxInfo * pfx,
const char * str)
1327 return (LocaleNCompare (pfx->pex, str, strlen(str))==0 ? MagickTrue : MagickFalse);
1330static MagickBooleanType ExpectChar (
FxInfo * pfx,
char c)
1332 if (PeekChar (pfx) != c) {
1333 (void) ThrowMagickException (
1334 pfx->exception, GetMagickModule(), OptionError,
1335 "Expected char",
"'%c' at '%s'", c, SetShortExp (pfx));
1342static int MaybeXYWH (
FxInfo * pfx, ImgAttrE * pop)
1349 if (*pop != aPage && *pop != aPrintsize && *pop != aRes)
return 0;
1351 if (PeekChar (pfx) !=
'.')
return 0;
1353 if (!ExpectChar (pfx,
'.'))
return 0;
1355 (void) GetToken (pfx);
1356 if (LocaleCompare (
"x", pfx->token)==0) ret=1;
1357 else if (LocaleCompare (
"y", pfx->token)==0) ret=2;
1358 else if (LocaleCompare (
"width", pfx->token)==0) ret=3;
1359 else if (LocaleCompare (
"height", pfx->token)==0) ret=4;
1362 (void) ThrowMagickException (
1363 pfx->exception, GetMagickModule(), OptionError,
1364 "Invalid 'x' or 'y' or 'width' or 'height' token=",
"'%s' at '%s'",
1365 pfx->token, SetShortExp(pfx));
1367 if (*pop == aPage) (*pop) = (ImgAttrE) ((
int) *pop + ret);
1370 (void) ThrowMagickException (
1371 pfx->exception, GetMagickModule(), OptionError,
1372 "Invalid 'width' or 'height' token=",
"'%s' at '%s'",
1373 pfx->token, SetShortExp(pfx));
1375 (*pop) = (ImgAttrE) ((
int) *pop + ret);
1378 pfx->pex+=pfx->lenToken;
1383static MagickBooleanType ExtendOperatorStack (
FxInfo * pfx)
1385 pfx->numOprStack = (int) ceil (pfx->numOprStack * (1 + TableExtend));
1386 pfx->OperatorStack = (OperatorE*) ResizeMagickMemory (pfx->OperatorStack, (
size_t) pfx->numOprStack *
sizeof(OperatorE));
1387 if (!pfx->OperatorStack) {
1388 (void) ThrowMagickException (
1389 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
1397static MagickBooleanType PushOperatorStack (
FxInfo * pfx,
int op)
1399 if (++pfx->usedOprStack >= pfx->numOprStack) {
1400 if (!ExtendOperatorStack (pfx))
1403 pfx->OperatorStack[pfx->usedOprStack-1] = (OperatorE) op;
1405 if (pfx->maxUsedOprStack < pfx->usedOprStack)
1406 pfx->maxUsedOprStack = pfx->usedOprStack;
1410static OperatorE GetLeadingOp (
FxInfo * pfx)
1412 OperatorE op = oNull;
1414 if (*pfx->pex ==
'-') op = oUnaryMinus;
1415 else if (*pfx->pex ==
'+') op = oUnaryPlus;
1416 else if (*pfx->pex ==
'~') op = oBitNot;
1417 else if (*pfx->pex ==
'!') op = oLogNot;
1418 else if (*pfx->pex ==
'(') op = oOpenParen;
1423static inline MagickBooleanType OprIsUnaryPrefix (OperatorE op)
1425 return (op == oUnaryMinus || op == oUnaryPlus || op == oBitNot || op == oLogNot ? MagickTrue : MagickFalse);
1428static MagickBooleanType TopOprIsUnaryPrefix (
FxInfo * pfx)
1430 if (!pfx->usedOprStack)
return MagickFalse;
1432 return OprIsUnaryPrefix (pfx->OperatorStack[pfx->usedOprStack-1]);
1435static MagickBooleanType PopOprOpenParen (
FxInfo * pfx, OperatorE op)
1438 if (!pfx->usedOprStack)
return MagickFalse;
1440 if (pfx->OperatorStack[pfx->usedOprStack-1] != op)
return MagickFalse;
1442 pfx->usedOprStack--;
1447static int GetCoordQualifier (
FxInfo * pfx,
int op)
1451 if (op != fU && op != fV && op != fS)
return -1;
1453 (void) GetToken (pfx);
1455 if (pfx->lenToken != 1) {
1458 if (*pfx->token !=
'p' && *pfx->token !=
'P')
return -1;
1459 if (!GetFunction (pfx, fP))
return -1;
1464static PixelChannel GetChannelQualifier (
FxInfo * pfx,
int op)
1466 if (op == fU || op == fV || op == fP ||
1467 op == fUP || op == fVP ||
1468 op == fS || (op >= (
int) FirstImgAttr && op <= aNull)
1471 const ChannelT * pch = &Channels[0];
1472 (void) GetToken (pfx);
1475 if (LocaleCompare (pch->str, pfx->token)==0) {
1477 if (op >= (
int) FirstImgAttr && op <= (int) ((OperatorE)aNull) &&
1478 ChanIsVirtual (pch->pixel_channel)
1481 (void) ThrowMagickException (
1482 pfx->exception, GetMagickModule(), OptionError,
1483 "Can't have image attribute with channel qualifier at",
"'%s' at '%s'",
1484 pfx->token, SetShortExp(pfx));
1485 return NO_CHAN_QUAL;
1488 pfx->pex += pfx->lenToken;
1489 return pch->pixel_channel;
1494 return NO_CHAN_QUAL;
1497static ImgAttrE GetImgAttrToken (
FxInfo * pfx)
1499 ImgAttrE ia = aNull;
1501 for (ia = FirstImgAttr; ia < aNull; ia=(ImgAttrE) (ia+1)) {
1502 iaStr = ImgAttrs[ia-(int) FirstImgAttr].str;
1503 if (LocaleCompare (iaStr, pfx->token)==0) {
1504 pfx->pex += strlen(pfx->token);
1505 if (ImgAttrs[ia-(
int) FirstImgAttr].need_stats != MagickFalse) pfx->NeedStats = MagickTrue;
1506 MaybeXYWH (pfx, &ia);
1511 if (ia == aPage || ia == aPrintsize || ia == aRes) {
1512 (void) ThrowMagickException (
1513 pfx->exception, GetMagickModule(), OptionError,
1514 "Attribute",
"'%s' needs qualifier at '%s'",
1515 iaStr, SetShortExp(pfx));
1521static ImgAttrE GetImgAttrQualifier (
FxInfo * pfx,
int op)
1523 ImgAttrE ia = aNull;
1524 if (op == (OperatorE)fU || op == (OperatorE)fV || op == (OperatorE)fP || op == (OperatorE)fS) {
1525 (void) GetToken (pfx);
1526 if (pfx->lenToken == 0) {
1529 ia = GetImgAttrToken (pfx);
1534static MagickBooleanType IsQualifier (
FxInfo * pfx)
1536 if (PeekChar (pfx) ==
'.') {
1543static ssize_t GetProperty (
FxInfo * pfx, fxFltType *val)
1549 if (PeekStr (pfx,
"%[")) {
1552 char sProperty [MagickPathExtent];
1553 char * p = pfx->pex + 2;
1557 if (*p ==
'[') level++;
1558 else if (*p ==
']') {
1559 if (level == 0)
break;
1564 if (!*p || level != 0) {
1565 (void) ThrowMagickException (
1566 pfx->exception, GetMagickModule(), OptionError,
1567 "After '%[' expected ']' at",
"'%s'",
1572 len = (size_t) (p - pfx->pex + 1);
1573 if (len > MaxTokenLen) {
1574 (void) ThrowMagickException (
1575 pfx->exception, GetMagickModule(), OptionError,
1576 "Too much text between '%[' and ']' at",
"'%s'",
1581 (void) CopyMagickString (sProperty, pfx->pex, len+1);
1582 sProperty[len] =
'\0';
1586 text = InterpretImageProperties (pfx->image->image_info, pfx->image,
1587 sProperty, pfx->exception);
1588 if (!text || !*text) {
1589 text = DestroyString(text);
1590 (void) ThrowMagickException (
1591 pfx->exception, GetMagickModule(), OptionError,
1592 "Unknown property",
"'%s' at '%s'",
1593 sProperty, SetShortExp(pfx));
1597 *val = strtold (text, &tailptr);
1598 if (text == tailptr) {
1599 text = DestroyString(text);
1600 (void) ThrowMagickException (
1601 pfx->exception, GetMagickModule(), OptionError,
1602 "Property",
"'%s' text '%s' is not a number at '%s'",
1603 sProperty, text, SetShortExp(pfx));
1607 text = DestroyString(text);
1609 return ((ssize_t) len);
1615static inline ssize_t GetConstantColour (
FxInfo * pfx, fxFltType *v0, fxFltType *v1, fxFltType *v2)
1626 *dummy_exception = AcquireExceptionInfo ();
1636 char ColSp[MagickPathExtent];
1637 (void) CopyMagickString (ColSp, pfx->token, MaxTokenLen);
1638 p = ColSp + pfx->lenToken - 1;
1639 if (*p ==
'a' || *p ==
'A') *p =
'\0';
1641 (void) GetPixelInfo (pfx->image, &colour);
1645 IsGray = (LocaleCompare (ColSp,
"gray") == 0) ? MagickTrue : MagickFalse;
1646 IsIcc = (LocaleCompare (ColSp,
"icc-color") == 0) ? MagickTrue : MagickFalse;
1647 IsDev = (LocaleNCompare (ColSp,
"device-", 7) == 0) ? MagickTrue : MagickFalse;
1651 if (!QueryColorCompliance (pfx->token, AllCompliance, &colour, dummy_exception) || IsGray) {
1652 ssize_t type = ParseCommandOption (MagickColorspaceOptions, MagickFalse, ColSp);
1653 if (type >= 0 || IsIcc || IsDev) {
1654 char * q = pfx->pex + pfx->lenToken;
1655 while (isspace((
int) ((
unsigned char) *q))) q++;
1658 char sFunc[MagickPathExtent];
1659 while (*q && *q !=
')') q++;
1661 (void) ThrowMagickException (
1662 pfx->exception, GetMagickModule(), OptionError,
1663 "constant color missing ')'",
"at '%s'",
1665 dummy_exception = DestroyExceptionInfo (dummy_exception);
1668 lenfun = (size_t) (q - pfx->pex + 1);
1669 if (lenfun > MaxTokenLen) {
1670 (void) ThrowMagickException (
1671 pfx->exception, GetMagickModule(), OptionError,
1672 "lenfun too long",
"'%lu' at '%s'",
1673 (
unsigned long) lenfun, SetShortExp(pfx));
1674 dummy_exception = DestroyExceptionInfo (dummy_exception);
1677 (void) CopyMagickString (sFunc, pfx->pex, lenfun+1);
1678 if (QueryColorCompliance (sFunc, AllCompliance, &colour, dummy_exception)) {
1679 *v0 = QuantumScale*colour.red;
1680 *v1 = QuantumScale*colour.green;
1681 *v2 = QuantumScale*colour.blue;
1682 dummy_exception = DestroyExceptionInfo (dummy_exception);
1683 return (ssize_t)lenfun;
1686 (void) ThrowMagickException (
1687 pfx->exception, GetMagickModule(), OptionError,
1688 "colorspace but not a valid color with '(...)' at",
"'%s'",
1690 dummy_exception = DestroyExceptionInfo (dummy_exception);
1695 dummy_exception = DestroyExceptionInfo (dummy_exception);
1700 *v0 = QuantumScale*colour.red;
1701 *v1 = QuantumScale*colour.green;
1702 *v2 = QuantumScale*colour.blue;
1704 dummy_exception = DestroyExceptionInfo (dummy_exception);
1705 return (ssize_t)strlen (pfx->token);
1708static inline ssize_t GetHexColour (
FxInfo * pfx, fxFltType *v0, fxFltType *v1, fxFltType *v2)
1717 if (*pfx->pex !=
'#')
return 0;
1721 while (isxdigit ((
int)*p)) p++;
1722 if (isalpha ((
int)*p)) {
1723 (void) ThrowMagickException (
1724 pfx->exception, GetMagickModule(), OptionError,
1725 "Bad hex number at",
"'%s'",
1730 len = (size_t) (p - pfx->pex);
1731 if (len < 1)
return 0;
1732 if (len >= MaxTokenLen) {
1733 (void) ThrowMagickException (
1734 pfx->exception, GetMagickModule(), OptionError,
1735 "Hex colour too long at",
"'%s'",
1739 (void) CopyMagickString (pfx->token, pfx->pex, len+1);
1741 (void) GetPixelInfo (pfx->image, &colour);
1743 if (!QueryColorCompliance (pfx->token, AllCompliance, &colour, pfx->exception)) {
1744 (void) ThrowMagickException (
1745 pfx->exception, GetMagickModule(), OptionError,
1746 "QueryColorCompliance rejected",
"'%s' at '%s'",
1747 pfx->token, SetShortExp(pfx));
1751 *v0 = QuantumScale*colour.red;
1752 *v1 = QuantumScale*colour.green;
1753 *v2 = QuantumScale*colour.blue;
1755 return (ssize_t) len;
1758static MagickBooleanType GetFunction (
FxInfo * pfx, FunctionE fe)
1762 const char * funStr = Functions[fe-(int) FirstFunc].str;
1763 int nArgs = Functions[fe-(int) FirstFunc].number_args;
1765 char expChLimit =
')';
1766 const char *strLimit =
",)";
1767 OperatorE pushOp = oOpenParen;
1774 int ndx0 = NULL_ADDRESS, ndx1 = NULL_ADDRESS, ndx2 = NULL_ADDRESS, ndx3 = NULL_ADDRESS;
1776 MagickBooleanType coordQual = MagickFalse;
1777 PixelChannel chQual = NO_CHAN_QUAL;
1778 ImgAttrE iaQual = aNull;
1780 pfx->pex += pfx->lenToken;
1783 char p = PeekChar (pfx);
1785 (void) ExpectChar (pfx,
'{');
1786 pushOp = oOpenBrace;
1790 }
else if (p==
'[') {
1791 (void) ExpectChar (pfx,
'[');
1792 pushOp = oOpenBracket;
1801 }
else if (fe == fU) {
1802 char p = PeekChar (pfx);
1804 (void) ExpectChar (pfx,
'[');
1805 pushOp = oOpenBracket;
1814 }
else if (fe == fV || fe == fS) {
1816 pushOp = oOpenBracket;
1820 if (!ExpectChar (pfx,
'('))
return MagickFalse;
1822 if (!PushOperatorStack (pfx, (
int) pushOp))
return MagickFalse;
1824 pExpStart = pfx->pex;
1825 ndx0 = pfx->usedElements;
1827 (void) AddAddressingElement (pfx, rGoto, NULL_ADDRESS);
1831 if (TranslateStatementList (pfx, strLimit, &chLimit)) {
1835 (void) ThrowMagickException (
1836 pfx->exception, GetMagickModule(), OptionError,
1837 "For function",
"'%s' expected ')' at '%s'",
1838 funStr, SetShortExp(pfx));
1842 if (!chLimit)
break;
1843 if (fe == fP || fe == fS|| fe == fIf) {
1844 (void) AddElement (pfx, (fxFltType) 0, oNull);
1849 if (strchr (strLimit, chLimit)==NULL) {
1850 (void) ThrowMagickException (
1851 pfx->exception, GetMagickModule(), OptionError,
1852 "For function",
"'%s' expected one of '%s' after expression but found '%c' at '%s'",
1853 funStr, strLimit, chLimit ? chLimit :
' ', SetShortExp(pfx));
1862 if (ndx1 != NULL_ADDRESS) {
1863 (void) ThrowMagickException (
1864 pfx->exception, GetMagickModule(), OptionError,
1865 "For function",
"'%s' required argument is missing at '%s'",
1866 funStr, SetShortExp(pfx));
1869 ndx1 = pfx->usedElements;
1870 if (fe==fWhile || fe==fIf) {
1871 (void) AddAddressingElement (pfx, rIfZeroGoto, NULL_ADDRESS);
1872 }
else if (fe==fDo) {
1873 (void) AddAddressingElement (pfx, rIfZeroGoto, NULL_ADDRESS);
1874 }
else if (fe==fFor) {
1875 pfx->Elements[pfx->usedElements-1].do_push = MagickFalse;
1879 if (ndx2 != NULL_ADDRESS) {
1880 (void) ThrowMagickException (
1881 pfx->exception, GetMagickModule(), OptionError,
1882 "For function",
"'%s' required argument is missing at '%s'",
1883 funStr, SetShortExp(pfx));
1886 ndx2 = pfx->usedElements;
1888 pfx->Elements[pfx->usedElements-1].do_push = MagickFalse;
1889 (void) AddAddressingElement (pfx, rGotoChk, ndx0);
1890 }
else if (fe==fDo) {
1891 pfx->Elements[pfx->usedElements-1].do_push = MagickFalse;
1892 (void) AddAddressingElement (pfx, rGotoChk, ndx0 + 1);
1893 }
else if (fe==fFor) {
1894 (void) AddAddressingElement (pfx, rIfZeroGoto, NULL_ADDRESS);
1895 pfx->Elements[pfx->usedElements-1].do_push = MagickTrue;
1896 (void) AddAddressingElement (pfx, rZerStk, NULL_ADDRESS);
1897 }
else if (fe==fIf) {
1898 (void) AddAddressingElement (pfx, rGoto, NULL_ADDRESS);
1902 if (ndx3 != NULL_ADDRESS) {
1903 (void) ThrowMagickException (
1904 pfx->exception, GetMagickModule(), OptionError,
1905 "For function",
"'%s' required argument is missing at '%s'",
1906 funStr, SetShortExp(pfx));
1910 pfx->Elements[pfx->usedElements-1].do_push = MagickFalse;
1911 (void) AddAddressingElement (pfx, rGotoChk, ndx1);
1913 ndx3 = pfx->usedElements;
1918 if (chLimit == expChLimit) {
1919 lenExp = (size_t) (pfx->pex - pExpStart - 1);
1923 if (chLimit && chLimit != expChLimit && chLimit !=
',' ) {
1924 (void) ThrowMagickException (
1925 pfx->exception, GetMagickModule(), OptionError,
1926 "For function",
"'%s' expected '%c', found '%c' at '%s'",
1927 funStr, expChLimit, chLimit ? chLimit :
' ', SetShortExp(pfx));
1931 if (fe == fP || fe == fS || fe == fU || fe == fChannel) {
1932 while (FndArgs < Functions[fe-(
int) FirstFunc].number_args) {
1933 (void) AddElement (pfx, (fxFltType) 0, oNull);
1938 if (FndArgs > Functions[fe-(
int) FirstFunc].number_args)
1941 (void) ThrowMagickException (
1942 pfx->exception, GetMagickModule(), OptionError,
1943 "For function",
"'%s' expected up to %i arguments, found '%i' at '%s'",
1944 funStr, Functions[fe-(int) FirstFunc].number_args, FndArgs, SetShortExp(pfx));
1946 (void) ThrowMagickException (
1947 pfx->exception, GetMagickModule(), OptionError,
1948 "For function",
"'%s' expected %i arguments, found '%i' at '%s'",
1949 funStr, Functions[fe-(int) FirstFunc].number_args, FndArgs, SetShortExp(pfx));
1953 if (FndArgs < Functions[fe-(
int) FirstFunc].number_args) {
1954 (void) ThrowMagickException (
1955 pfx->exception, GetMagickModule(), OptionError,
1956 "For function",
"'%s' expected %i arguments, found too few (%i) at '%s'",
1957 funStr, Functions[fe-(int) FirstFunc].number_args, FndArgs, SetShortExp(pfx));
1960 if (fe != fS && fe != fV && FndArgs == 0 && Functions[fe-(
int) FirstFunc].number_args == 0) {
1962 chLimit = expChLimit;
1963 if (!ExpectChar (pfx,
')'))
return MagickFalse;
1966 if (chLimit != expChLimit) {
1967 (void) ThrowMagickException (
1968 pfx->exception, GetMagickModule(), OptionError,
1969 "For function",
"'%s', arguments don't end with '%c' at '%s'",
1970 funStr, expChLimit, SetShortExp(pfx));
1973 if (!PopOprOpenParen (pfx, pushOp)) {
1974 (void) ThrowMagickException (
1975 pfx->exception, GetMagickModule(), OptionError,
1976 "Bug: For function",
"'%s' tos not '%s' at '%s'",
1977 funStr, Operators[pushOp].str, SetShortExp(pfx));
1981 if (IsQualifier (pfx)) {
1983 if (fe == fU || fe == fV || fe == fS) {
1985 coordQual = (GetCoordQualifier (pfx, (
int) fe) == 1) ? MagickTrue : MagickFalse;
1990 ElementT * pel = &pfx->Elements[pfx->usedElements-1];
1991 if (pel->operator_index != fP) {
1992 (void) ThrowMagickException (
1993 pfx->exception, GetMagickModule(), OptionError,
1994 "Bug: For function",
"'%s' last element not 'p' at '%s'",
1995 funStr, SetShortExp(pfx));
1998 chQual = pel->channel_qual;
1999 expChLimit = (pel->is_relative) ?
']' :
'}';
2000 pfx->usedElements--;
2001 if (fe == fU) fe = fUP;
2002 else if (fe == fV) fe = fVP;
2003 else if (fe == fS) fe = fSP;
2004 funStr = Functions[fe-(int) FirstFunc].str;
2008 if ( chQual == NO_CHAN_QUAL &&
2009 (fe == fP || fe == fS || fe == fSP || fe == fU || fe == fUP || fe == fV || fe == fVP)
2012 chQual = GetChannelQualifier (pfx, (
int) fe);
2015 if (chQual == NO_CHAN_QUAL && (fe == fU || fe == fV || fe == fS)) {
2017 iaQual = GetImgAttrQualifier (pfx, (
int) fe);
2019 if (IsQualifier (pfx) && chQual == NO_CHAN_QUAL && iaQual != aNull) {
2020 chQual = GetChannelQualifier (pfx, (
int) fe);
2022 if (coordQual && iaQual != aNull) {
2023 (void) ThrowMagickException (
2024 pfx->exception, GetMagickModule(), OptionError,
2025 "For function",
"'%s', can't have qualifiers 'p' and image attribute '%s' at '%s'",
2026 funStr, pfx->token, SetShortExp(pfx));
2029 if (!coordQual && chQual == NO_CHAN_QUAL && iaQual == aNull) {
2030 (void) ThrowMagickException (
2031 pfx->exception, GetMagickModule(), OptionError,
2032 "For function",
"'%s', bad qualifier '%s' at '%s'",
2033 funStr, pfx->token, SetShortExp(pfx));
2036 if (!coordQual && chQual == CompositePixelChannel && iaQual == aNull) {
2037 (void) ThrowMagickException (
2038 pfx->exception, GetMagickModule(), OptionError,
2039 "For function",
"'%s', bad composite qualifier '%s' at '%s'",
2040 funStr, pfx->token, SetShortExp(pfx));
2044 if (chQual == HUE_CHANNEL || chQual == SAT_CHANNEL || chQual == LIGHT_CHANNEL) {
2045 pfx->NeedHsl = MagickTrue;
2047 if (iaQual >= FirstImgAttr && iaQual < aNull) {
2048 (void) ThrowMagickException (
2049 pfx->exception, GetMagickModule(), OptionError,
2050 "Can't have image attribute with HLS qualifier at",
"'%s'",
2057 if (iaQual != aNull && chQual != NO_CHAN_QUAL) {
2058 if (ImgAttrs[iaQual-(
int) FirstImgAttr].need_stats == MagickFalse) {
2059 (void) ThrowMagickException (
2060 pfx->exception, GetMagickModule(), OptionError,
2061 "Can't have image attribute ",
"'%s' with channel qualifier '%s' at '%s'",
2062 ImgAttrs[iaQual-(int) FirstImgAttr].str,
2063 pfx->token, SetShortExp(pfx));
2066 if (ChanIsVirtual (chQual)) {
2067 (void) ThrowMagickException (
2068 pfx->exception, GetMagickModule(), OptionError,
2069 "Can't have statistical image attribute ",
"'%s' with virtual channel qualifier '%s' at '%s'",
2070 ImgAttrs[iaQual-(int) FirstImgAttr].str,
2071 pfx->token, SetShortExp(pfx));
2078 pfx->Elements[ndx1].element_index = ndx2+1;
2079 }
else if (fe==fDo) {
2080 pfx->Elements[ndx0].element_index = ndx1+1;
2081 pfx->Elements[ndx1].element_index = ndx2+1;
2082 }
else if (fe==fFor) {
2083 pfx->Elements[ndx2].element_index = ndx3;
2084 }
else if (fe==fIf) {
2085 pfx->Elements[ndx1].element_index = ndx2 + 1;
2086 pfx->Elements[ndx2].element_index = ndx3;
2088 if (fe == fU && iaQual == aNull) {
2089 ElementT * pel = &pfx->Elements[pfx->usedElements-1];
2090 if (pel->type == etConstant && pel->val == 0.0) {
2091 pfx->usedElements--;
2095 (void) AddElement (pfx, (fxFltType) 0, (int) fe);
2096 if (fe == fP || fe == fU || fe == fU0 || fe == fUP ||
2097 fe == fV || fe == fVP || fe == fS || fe == fSP)
2099 ElementT * pel = &pfx->Elements[pfx->usedElements-1];
2100 pel->is_relative = (expChLimit ==
']' ? MagickTrue : MagickFalse);
2101 if (chQual >= 0) pel->channel_qual = chQual;
2102 if (iaQual != aNull && (fe == fU || fe == fV || fe == fS)) {
2104 pel->img_attr_qual = iaQual;
2109 if (pExpStart && lenExp) {
2110 ElementT * pel = &pfx->Elements[pfx->usedElements-1];
2111 pel->exp_start = pExpStart;
2112 pel->exp_len = lenExp;
2116 pfx->ContainsDebug = MagickTrue;
2121static MagickBooleanType IsStealth (
int op)
2123 return (op == fU0 || op == fUP || op == fSP || op == fVP ||
2124 (op >= FirstCont && op <= rNull) ? MagickTrue : MagickFalse
2128static MagickBooleanType GetOperand (
2129 FxInfo * pfx, MagickBooleanType * UserSymbol, MagickBooleanType * NewUserSymbol,
int * UserSymNdx,
2130 MagickBooleanType * needPopAll)
2133 *NewUserSymbol = *UserSymbol = MagickFalse;
2134 *UserSymNdx = NULL_ADDRESS;
2137 if (!*pfx->pex)
return MagickFalse;
2138 (void) GetToken (pfx);
2140 if (pfx->lenToken==0) {
2144 OperatorE op = GetLeadingOp (pfx);
2145 if (op==oOpenParen) {
2146 char chLimit =
'\0';
2147 if (!PushOperatorStack (pfx, (
int) op))
return MagickFalse;
2149 if (!TranslateExpression (pfx,
")", &chLimit, needPopAll)) {
2150 (void) ThrowMagickException (
2151 pfx->exception, GetMagickModule(), OptionError,
2152 "Empty expression in parentheses at",
"'%s'",
2156 if (chLimit !=
')') {
2157 (void) ThrowMagickException (
2158 pfx->exception, GetMagickModule(), OptionError,
2159 "'(' but no ')' at",
"'%s'",
2164 if (!PopOprOpenParen (pfx, oOpenParen)) {
2165 (void) ThrowMagickException (
2166 pfx->exception, GetMagickModule(), OptionError,
2167 "Bug: tos not '(' at",
"'%s'",
2172 }
else if (OprIsUnaryPrefix (op)) {
2173 MagickBooleanType operand_ok;
2174 if (!PushOperatorStack (pfx, (
int) op))
return MagickFalse;
2177 if (!*pfx->pex)
return MagickFalse;
2178 if (pfx->teDepth >= MagickMaxRecursionDepth) {
2179 (void) ThrowMagickException (
2180 pfx->exception, GetMagickModule(), OptionError,
2181 "Expression too deeply nested",
"(depth %i exceeds limit %i)",
2182 pfx->teDepth, MagickMaxRecursionDepth);
2186 operand_ok=GetOperand (pfx, UserSymbol, NewUserSymbol, UserSymNdx, needPopAll);
2189 (void) ThrowMagickException (
2190 pfx->exception, GetMagickModule(), OptionError,
2191 "After unary, bad operand at",
"'%s'",
2196 if (*NewUserSymbol) {
2197 (void) ThrowMagickException (
2198 pfx->exception, GetMagickModule(), OptionError,
2199 "After unary, NewUserSymbol at",
"'%s'",
2205 (void) AddAddressingElement (pfx, rCopyFrom, *UserSymNdx);
2206 *UserSymNdx = NULL_ADDRESS;
2208 *UserSymbol = MagickFalse;
2209 *NewUserSymbol = MagickFalse;
2212 (void) GetToken (pfx);
2214 }
else if (*pfx->pex ==
'#') {
2215 fxFltType v0=0, v1=0, v2=0;
2216 ssize_t lenToken = GetHexColour (pfx, &v0, &v1, &v2);
2218 (void) ThrowMagickException (
2219 pfx->exception, GetMagickModule(), OptionError,
2220 "Bad hex number at",
"'%s'",
2223 }
else if (lenToken > 0) {
2224 (void) AddColourElement (pfx, v0, v1, v2);
2235 fxFltType val = strtold (pfx->pex, &tailptr);
2236 if (pfx->pex != tailptr) {
2244 const char Prefixes[] =
"yzafpnum.kMGTPEZY";
2245 const char * pSi = strchr (Prefixes, *tailptr);
2246 if (pSi && *pSi !=
'.') Pow = (double) ((pSi - Prefixes) * 3 - 24);
2247 else if (*tailptr ==
'c') Pow = -2;
2248 else if (*tailptr ==
'h') Pow = 2;
2249 else if (*tailptr ==
'k') Pow = 3;
2251 if (*(++pfx->pex) ==
'i') {
2252 val *= pow (2.0, Pow/0.3);
2255 val *= pow (10.0, Pow);
2259 (void) AddElement (pfx, val, oNull);
2263 val = (fxFltType) 0;
2264 lenOptArt = GetProperty (pfx, &val);
2265 if (lenOptArt < 0)
return MagickFalse;
2266 if (lenOptArt > 0) {
2267 (void) AddElement (pfx, val, oNull);
2268 pfx->pex += lenOptArt;
2275 if (pfx->lenToken > 0) {
2280 for (ce = (ConstantE)0; ce < cNull; ce=(ConstantE) (ce+1)) {
2281 const char * ceStr = Constants[ce].str;
2282 if (LocaleCompare (ceStr, pfx->token)==0) {
2288 (void) AddElement (pfx, Constants[ce].val, oNull);
2289 pfx->pex += pfx->lenToken;
2298 for (fe = FirstFunc; fe < fNull; fe=(FunctionE) (fe+1)) {
2299 const char * feStr = Functions[fe-(int) FirstFunc].str;
2300 if (LocaleCompare (feStr, pfx->token)==0) {
2305 if (fe == fV && pfx->ImgListLen < 2) {
2306 (void) ThrowMagickException (
2307 pfx->exception, GetMagickModule(), OptionError,
2308 "Symbol 'v' but fewer than two images at",
"'%s'",
2313 if (IsStealth ((
int) fe)) {
2314 (void) ThrowMagickException (
2315 pfx->exception, GetMagickModule(), OptionError,
2316 "Function",
"'%s' not permitted at '%s'",
2317 pfx->token, SetShortExp(pfx));
2320 if (fe == fDo || fe == fFor || fe == fIf || fe == fWhile) {
2321 *needPopAll = MagickTrue;
2324 if (fe != fNull)
return (GetFunction (pfx, fe));
2330 ImgAttrE ia = GetImgAttrToken (pfx);
2333 (void) AddElement (pfx, val, (
int) ia);
2335 if (ImgAttrs[ia-(
int) FirstImgAttr].need_stats != MagickFalse) {
2336 if (IsQualifier (pfx)) {
2337 PixelChannel chQual = GetChannelQualifier (pfx, (
int) ia);
2339 if (chQual == NO_CHAN_QUAL) {
2340 (void) ThrowMagickException (
2341 pfx->exception, GetMagickModule(), OptionError,
2342 "Bad channel qualifier at",
"'%s'",
2347 pel = &pfx->Elements[pfx->usedElements-1];
2348 pel->channel_qual = chQual;
2359 for (se = FirstSym; se < sNull; se=(SymbolE) (se+1)) {
2360 const char * seStr = Symbols[se-(int) FirstSym].str;
2361 if (LocaleCompare (seStr, pfx->token)==0) {
2367 (void) AddElement (pfx, val, (
int) se);
2368 pfx->pex += pfx->lenToken;
2370 if (se==sHue || se==sSaturation || se==sLightness) pfx->NeedHsl = MagickTrue;
2378 fxFltType v0, v1, v2;
2379 ssize_t ColLen = GetConstantColour (pfx, &v0, &v1, &v2);
2380 if (ColLen < 0)
return MagickFalse;
2382 (void) AddColourElement (pfx, v0, v1, v2);
2391 const char *artifact;
2392 artifact = GetImageArtifact (pfx->image, pfx->token);
2393 if (artifact != (
const char *) NULL) {
2395 fxFltType val = strtold (artifact, &tailptr);
2396 if (pfx->token == tailptr) {
2397 (void) ThrowMagickException (
2398 pfx->exception, GetMagickModule(), OptionError,
2399 "Artifact",
"'%s' has value '%s', not a number, at '%s'",
2400 pfx->token, artifact, SetShortExp(pfx));
2403 (void) AddElement (pfx, val, oNull);
2404 pfx->pex+=pfx->lenToken;
2411 if (TokenMaybeUserSymbol (pfx)) {
2412 *UserSymbol = MagickTrue;
2413 *UserSymNdx = FindUserSymbol (pfx, pfx->token);
2414 if (*UserSymNdx == NULL_ADDRESS) {
2415 *UserSymNdx = AddUserSymbol (pfx, pfx->pex, pfx->lenToken);
2416 *NewUserSymbol = MagickTrue;
2419 pfx->pex += pfx->lenToken;
2425 (void) ThrowMagickException (
2426 pfx->exception, GetMagickModule(), OptionError,
2427 "Expected operand at",
"'%s'",
2433static inline MagickBooleanType IsRealOperator (OperatorE op)
2435 return (op < oOpenParen || op > oCloseBrace) ? MagickTrue : MagickFalse;
2438static inline MagickBooleanType ProcessTernaryOpr (
FxInfo * pfx,
TernaryT * ptern)
2443 if (pfx->usedOprStack == 0)
2445 if (pfx->OperatorStack[pfx->usedOprStack-1] == oQuery) {
2446 if (ptern->addr_query != NULL_ADDRESS) {
2447 (void) ThrowMagickException (
2448 pfx->exception, GetMagickModule(), OptionError,
2449 "Already have '?' in sub-expression at",
"'%s'",
2453 if (ptern->addr_colon != NULL_ADDRESS) {
2454 (void) ThrowMagickException (
2455 pfx->exception, GetMagickModule(), OptionError,
2456 "Already have ':' in sub-expression at",
"'%s'",
2460 pfx->usedOprStack--;
2461 ptern->addr_query = pfx->usedElements;
2462 (void) AddAddressingElement (pfx, rIfZeroGoto, NULL_ADDRESS);
2465 else if (pfx->OperatorStack[pfx->usedOprStack-1] == oColon) {
2466 if (ptern->addr_query == NULL_ADDRESS) {
2467 (void) ThrowMagickException (
2468 pfx->exception, GetMagickModule(), OptionError,
2469 "Need '?' in sub-expression at",
"'%s'",
2473 if (ptern->addr_colon != NULL_ADDRESS) {
2474 (void) ThrowMagickException (
2475 pfx->exception, GetMagickModule(), OptionError,
2476 "Already have ':' in sub-expression at",
"'%s'",
2480 pfx->usedOprStack--;
2481 ptern->addr_colon = pfx->usedElements;
2482 pfx->Elements[pfx->usedElements-1].do_push = MagickTrue;
2483 (void) AddAddressingElement (pfx, rGoto, NULL_ADDRESS);
2489static MagickBooleanType GetOperator (
2491 MagickBooleanType * Assign, MagickBooleanType * Update, MagickBooleanType * IncrDecr)
2495 MagickBooleanType DoneIt = MagickFalse;
2497 for (op = (OperatorE)0; op != oNull; op=(OperatorE) (op+1)) {
2498 const char * opStr = Operators[op].str;
2499 len = strlen(opStr);
2500 if (LocaleNCompare (opStr, pfx->pex, len)==0) {
2505 if (!IsRealOperator (op)) {
2506 (void) ThrowMagickException (
2507 pfx->exception, GetMagickModule(), OptionError,
2508 "Not a real operator at",
"'%s'",
2514 (void) ThrowMagickException (
2515 pfx->exception, GetMagickModule(), OptionError,
2516 "Expected operator at",
"'%s'",
2521 *Assign = (op==oAssign) ? MagickTrue : MagickFalse;
2522 *Update = OprInPlace ((
int) op);
2523 *IncrDecr = (op == oPlusPlus || op == oSubSub) ? MagickTrue : MagickFalse;
2530 while (pfx->usedOprStack > 0) {
2531 OperatorE top = pfx->OperatorStack[pfx->usedOprStack-1];
2532 int precTop, precNew;
2533 if (top == oOpenParen || top == oAssign || OprInPlace ((
int) top))
break;
2534 precTop = Operators[top].precedence;
2535 precNew = Operators[op].precedence;
2539 if (precTop < precNew)
break;
2540 (void) AddElement (pfx, (fxFltType) 0, (int) top);
2541 pfx->usedOprStack--;
2547 if (op==oCloseParen) {
2548 if (pfx->usedOprStack == 0) {
2549 (void) ThrowMagickException (
2550 pfx->exception, GetMagickModule(), OptionError,
2551 "Found ')' but nothing on stack at",
"'%s'",
2556 if (pfx->OperatorStack[pfx->usedOprStack-1] != oOpenParen) {
2557 (void) ThrowMagickException (
2558 pfx->exception, GetMagickModule(), OptionError,
2559 "Found ')' but no '(' on stack at",
"'%s'",
2563 pfx->usedOprStack--;
2564 DoneIt = MagickTrue;
2568 if (!PushOperatorStack (pfx, (
int) op))
return MagickFalse;
2576static MagickBooleanType ResolveTernaryAddresses (
FxInfo * pfx,
TernaryT * ptern)
2578 if (ptern->addr_query == NULL_ADDRESS && ptern->addr_colon == NULL_ADDRESS)
2581 if (ptern->addr_query != NULL_ADDRESS && ptern->addr_colon != NULL_ADDRESS) {
2582 pfx->Elements[ptern->addr_query].element_index = ptern->addr_colon + 1;
2583 pfx->Elements[ptern->addr_colon].element_index = pfx->usedElements;
2584 ptern->addr_query = NULL_ADDRESS;
2585 ptern->addr_colon = NULL_ADDRESS;
2586 }
else if (ptern->addr_query != NULL_ADDRESS) {
2587 (void) ThrowMagickException (
2588 pfx->exception, GetMagickModule(), OptionError,
2589 "'?' with no corresponding ':'",
"'%s' at '%s'",
2590 pfx->token, SetShortExp(pfx));
2592 }
else if (ptern->addr_colon != NULL_ADDRESS) {
2593 (void) ThrowMagickException (
2594 pfx->exception, GetMagickModule(), OptionError,
2595 "':' with no corresponding '?'",
"'%s' at '%s'",
2596 pfx->token, SetShortExp(pfx));
2602static MagickBooleanType TranslateExpression (
2603 FxInfo * pfx,
const char * strLimit,
char * chLimit, MagickBooleanType * needPopAll)
2607 MagickBooleanType UserSymbol, NewUserSymbol;
2608 int UserSymNdx0, UserSymNdx1;
2611 Assign = MagickFalse,
2612 Update = MagickFalse,
2613 IncrDecr = MagickFalse;
2618 ternary.addr_query = NULL_ADDRESS;
2619 ternary.addr_colon = NULL_ADDRESS;
2621 if (pfx->teDepth >= MagickMaxRecursionDepth) {
2622 (void) ThrowMagickException(pfx->exception, GetMagickModule(), OptionError,
2623 "Expression too deeply nested",
"(depth %i exceeds limit %i)",
2624 pfx->teDepth, MagickMaxRecursionDepth);
2632 StartEleNdx = pfx->usedElements-1;
2633 if (StartEleNdx < 0) StartEleNdx = 0;
2642 if (strchr(strLimit,*pfx->pex)!=NULL) {
2643 *chLimit = *pfx->pex;
2650 if (!GetOperand (pfx, &UserSymbol, &NewUserSymbol, &UserSymNdx0, needPopAll))
return MagickFalse;
2655 while (*pfx->pex && (!*strLimit || (strchr(strLimit,*pfx->pex)==NULL))) {
2656 if (!GetOperator (pfx, &Assign, &Update, &IncrDecr))
return MagickFalse;
2658 if (NewUserSymbol && !Assign) {
2659 (void) ThrowMagickException (
2660 pfx->exception, GetMagickModule(), OptionError,
2661 "Expected assignment after new UserSymbol",
"'%s' at '%s'",
2662 pfx->token, SetShortExp(pfx));
2665 if (!UserSymbol && Assign) {
2666 (void) ThrowMagickException (
2667 pfx->exception, GetMagickModule(), OptionError,
2668 "Attempted assignment to non-UserSymbol",
"'%s' at '%s'",
2669 pfx->token, SetShortExp(pfx));
2672 if (!UserSymbol && Update) {
2673 (void) ThrowMagickException (
2674 pfx->exception, GetMagickModule(), OptionError,
2675 "Attempted update to non-UserSymbol",
"'%s' at '%s'",
2676 pfx->token, SetShortExp(pfx));
2679 if (UserSymbol && (Assign || Update) && !IncrDecr) {
2681 if (!TranslateExpression (pfx, strLimit, chLimit, needPopAll))
return MagickFalse;
2682 if (!*pfx->pex)
break;
2683 if (!*strLimit)
break;
2684 if (strchr(strLimit,*chLimit)!=NULL)
break;
2686 if (UserSymbol && !Assign && !Update && UserSymNdx0 != NULL_ADDRESS) {
2688 (void) AddAddressingElement (pfx, rCopyFrom, UserSymNdx0);
2689 UserSymNdx0 = NULL_ADDRESS;
2690 pel = &pfx->Elements[pfx->usedElements-1];
2691 pel->do_push = MagickTrue;
2695 while (TopOprIsUnaryPrefix (pfx)) {
2696 OperatorE op = pfx->OperatorStack[pfx->usedOprStack-1];
2697 (void) AddElement (pfx, (fxFltType) 0, (int) op);
2698 pfx->usedOprStack--;
2702 if (!ProcessTernaryOpr (pfx, &ternary))
return MagickFalse;
2704 if (ternary.addr_colon != NULL_ADDRESS) {
2705 if (!TranslateExpression (pfx,
",);", chLimit, needPopAll))
return MagickFalse;
2709 UserSymbol = NewUserSymbol = MagickFalse;
2711 if ( (!*pfx->pex) || (*strLimit && (strchr(strLimit,*pfx->pex)!=NULL) ) )
2713 if (IncrDecr)
break;
2715 (void) ThrowMagickException (
2716 pfx->exception, GetMagickModule(), OptionError,
2717 "Expected operand after operator",
"at '%s'",
2723 (void) ThrowMagickException (
2724 pfx->exception, GetMagickModule(), OptionError,
2725 "'++' and '--' must be the final operators in an expression at",
"'%s'",
2730 if (!GetOperand (pfx, &UserSymbol, &NewUserSymbol, &UserSymNdx1, needPopAll)) {
2731 (void) ThrowMagickException (
2732 pfx->exception, GetMagickModule(), OptionError,
2733 "Expected operand at",
"'%s'",
2738 if (NewUserSymbol && !Assign) {
2739 (void) ThrowMagickException (
2740 pfx->exception, GetMagickModule(), OptionError,
2741 "NewUserSymbol",
"'%s' after non-assignment operator at '%s'",
2742 pfx->token, SetShortExp(pfx));
2745 if (UserSymbol && !NewUserSymbol) {
2746 (void) AddAddressingElement (pfx, rCopyFrom, UserSymNdx1);
2747 UserSymNdx1 = NULL_ADDRESS;
2749 UserSymNdx0 = UserSymNdx1;
2752 if (UserSymbol && !Assign && !Update && UserSymNdx0 != NULL_ADDRESS) {
2754 if (NewUserSymbol) {
2755 (void) ThrowMagickException (
2756 pfx->exception, GetMagickModule(), OptionError,
2757 "NewUserSymbol",
"'%s' needs assignment operator at '%s'",
2758 pfx->token, SetShortExp(pfx));
2761 (void) AddAddressingElement (pfx, rCopyFrom, UserSymNdx0);
2762 pel = &pfx->Elements[pfx->usedElements-1];
2763 pel->do_push = MagickTrue;
2766 if (*pfx->pex && !*chLimit && (strchr(strLimit,*pfx->pex)!=NULL)) {
2767 *chLimit = *pfx->pex;
2770 while (pfx->usedOprStack) {
2771 OperatorE op = pfx->OperatorStack[pfx->usedOprStack-1];
2772 if (op == oOpenParen || op == oOpenBracket || op == oOpenBrace) {
2775 if ( (op==oAssign && !Assign) || (OprInPlace((
int) op) && !Update) ) {
2778 pfx->usedOprStack--;
2779 (void) AddElement (pfx, (fxFltType) 0, (int) op);
2780 if (op == oAssign) {
2781 if (UserSymNdx0 < 0) {
2782 (void) ThrowMagickException (
2783 pfx->exception, GetMagickModule(), OptionError,
2784 "Assignment to unknown user symbol at",
"'%s'",
2790 pfx->usedElements--;
2791 (void) AddAddressingElement (pfx, rCopyTo, UserSymNdx0);
2793 }
else if (OprInPlace ((
int) op)) {
2794 if (UserSymNdx0 < 0) {
2795 (void) ThrowMagickException (
2796 pfx->exception, GetMagickModule(), OptionError,
2797 "Operator-in-place to unknown user symbol at",
"'%s'",
2803 pfx->Elements[pfx->usedElements-1].element_index = UserSymNdx0;
2808 if (ternary.addr_query != NULL_ADDRESS) *needPopAll = MagickTrue;
2810 (void) ResolveTernaryAddresses (pfx, &ternary);
2814 if (!pfx->teDepth && *needPopAll) {
2815 (void) AddAddressingElement (pfx, rZerStk, NULL_ADDRESS);
2816 *needPopAll = MagickFalse;
2819 if (pfx->exception->severity >= ErrorException)
2826static MagickBooleanType TranslateStatement (
FxInfo * pfx,
char * strLimit,
char * chLimit)
2828 MagickBooleanType NeedPopAll = MagickFalse;
2832 if (!*pfx->pex)
return MagickFalse;
2834 if (!TranslateExpression (pfx, strLimit, chLimit, &NeedPopAll)) {
2837 if (pfx->usedElements && *chLimit==
';') {
2842 ElementT * pel = &pfx->Elements[pfx->usedElements-1];
2843 if (pel->do_push) pel->do_push = MagickFalse;
2849static MagickBooleanType TranslateStatementList (
FxInfo * pfx,
const char * strLimit,
char * chLimit)
2851#define MAX_SLIMIT 10
2852 char sLimits[MAX_SLIMIT];
2855 if (!*pfx->pex)
return MagickFalse;
2856 (void) CopyMagickString (sLimits, strLimit, MAX_SLIMIT-1);
2858 if (strchr(strLimit,
';')==NULL)
2859 (
void) ConcatenateMagickString (sLimits,
";", MAX_SLIMIT);
2862 if (!TranslateStatement (pfx, sLimits, chLimit))
return MagickFalse;
2864 if (!*pfx->pex)
break;
2866 if (*chLimit !=
';') {
2871 if (pfx->exception->severity >= ErrorException)
2890 for (ch=0; ch <= (int) MaxPixelChannels; ch++) {
2891 cs[ch].mean *= QuantumScale;
2892 cs[ch].median *= QuantumScale;
2893 cs[ch].maxima *= QuantumScale;
2894 cs[ch].minima *= QuantumScale;
2895 cs[ch].standard_deviation *= QuantumScale;
2901static MagickBooleanType CollectStatistics (
FxInfo * pfx)
2903 Image * img = GetFirstImageInList (pfx->image);
2908 if (!pfx->statistics) {
2909 (void) ThrowMagickException (
2910 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
2911 "Statistics",
"%lu",
2912 (
unsigned long) pfx->ImgListLen);
2917 pfx->statistics[imgNum] = CollectOneImgStats (pfx, img);
2919 if (++imgNum == pfx->ImgListLen)
break;
2920 img = GetNextImageInList (img);
2921 assert (img != (
Image *) NULL);
2923 pfx->GotStats = MagickTrue;
2928static inline MagickBooleanType PushVal (
FxInfo * pfx,
fxRtT * pfxrt, fxFltType val,
int addr)
2930 if (pfxrt->usedValStack >=pfxrt->numValStack) {
2931 (void) ThrowMagickException (
2932 pfx->exception, GetMagickModule(), OptionError,
2933 "ValStack overflow at addr=",
"%i",
2938 pfxrt->ValStack[pfxrt->usedValStack++] = val;
2942static inline fxFltType PopVal (
FxInfo * pfx,
fxRtT * pfxrt,
int addr)
2944 if (pfxrt->usedValStack <= 0) {
2945 (void) ThrowMagickException (
2946 pfx->exception, GetMagickModule(), OptionError,
2947 "ValStack underflow at addr=",
"%i",
2949 return (fxFltType) 0;
2952 return pfxrt->ValStack[--pfxrt->usedValStack];
2955static inline fxFltType ImageStat (
2956 FxInfo * pfx, ssize_t ImgNum, PixelChannel channel, ImgAttrE ia)
2960 MagickBooleanType NeedRelinq = MagickFalse;
2964 (void) ThrowMagickException(pfx->exception,GetMagickModule(),
2965 OptionError,
"NoSuchImage",
"%lu",(
unsigned long) ImgNum);
2969 if (pfx->GotStats) {
2970 if ((channel < 0) || (channel > MaxPixelChannels))
2972 (void) ThrowMagickException(pfx->exception,GetMagickModule(),
2973 OptionError,
"NoSuchImageChannel",
"%i",channel);
2974 channel=(PixelChannel) 0;
2976 cs = pfx->statistics[ImgNum];
2977 }
else if (pfx->NeedStats) {
2979 if ((channel < 0) || (channel > MaxPixelChannels))
2981 (void) ThrowMagickException(pfx->exception,GetMagickModule(),
2982 OptionError,
"NoSuchImageChannel",
"%i",channel);
2983 channel=(PixelChannel) 0;
2985 cs = CollectOneImgStats (pfx, pfx->Images[ImgNum]);
2986 NeedRelinq = MagickTrue;
2991 ret = (fxFltType) GetImageDepth (pfx->Images[ImgNum], pfx->exception);
2994 ret = (fxFltType) GetBlobSize (pfx->image);
2998 ret = cs[channel].kurtosis;
3002 ret = cs[channel].maxima;
3006 ret = cs[channel].mean;
3010 ret = cs[channel].median;
3014 ret = cs[channel].minima;
3020 ret = (fxFltType) pfx->Images[ImgNum]->page.x;
3023 ret = (fxFltType) pfx->Images[ImgNum]->page.y;
3026 ret = (fxFltType) pfx->Images[ImgNum]->page.width;
3029 ret = (fxFltType) pfx->Images[ImgNum]->page.height;
3035 ret = (fxFltType) PerceptibleReciprocal (pfx->Images[ImgNum]->resolution.x)
3036 * pfx->Images[ImgNum]->columns;
3039 ret = (fxFltType) PerceptibleReciprocal (pfx->Images[ImgNum]->resolution.y)
3040 * pfx->Images[ImgNum]->rows;
3043 ret = (fxFltType) pfx->Images[ImgNum]->quality;
3049 ret = pfx->Images[ImgNum]->resolution.x;
3052 ret = pfx->Images[ImgNum]->resolution.y;
3056 ret = cs[channel].skewness;
3060 ret = cs[channel].standard_deviation;
3063 ret = (fxFltType) pfx->Images[ImgNum]->rows;
3066 ret = (fxFltType) pfx->ImgListLen;
3069 ret = (fxFltType) ImgNum;
3072 ret = (fxFltType) pfx->Images[ImgNum]->columns;
3075 ret = (fxFltType) GetImageDepth (pfx->Images[ImgNum], pfx->exception);
3078 (void) ThrowMagickException (pfx->exception,GetMagickModule(),OptionError,
3079 "Unknown ia=",
"%i",ia);
3086static inline fxFltType FxGcd (fxFltType x, fxFltType y,
const size_t depth)
3088#define FxMaxFunctionDepth 200
3091 return (FxGcd (y, x, depth+1));
3092 if ((fabs((
double) y) < 0.001) || (depth >= FxMaxFunctionDepth))
3094 return (FxGcd (y, x-y*floor((
double) (x/y)), depth+1));
3097static inline ssize_t ChkImgNum (
FxInfo * pfx, fxFltType f)
3100 ssize_t i = (ssize_t) floor ((
double) f + 0.5);
3101 if (i < 0) i += (ssize_t) pfx->ImgListLen;
3102 if (i < 0 || i >= (ssize_t) pfx->ImgListLen) {
3103 (void) ThrowMagickException (
3104 pfx->exception, GetMagickModule(), OptionError,
3105 "ImgNum",
"%lu bad for ImgListLen %lu",
3106 (
unsigned long) i, (
unsigned long) pfx->ImgListLen);
3112#define WHICH_ATTR_CHAN \
3113 (pel->channel_qual == NO_CHAN_QUAL) ? CompositePixelChannel : \
3114 (pel->channel_qual == THIS_CHANNEL) ? channel : pel->channel_qual
3116#define WHICH_NON_ATTR_CHAN \
3117 (pel->channel_qual == NO_CHAN_QUAL || \
3118 pel->channel_qual == THIS_CHANNEL || \
3119 pel->channel_qual == CompositePixelChannel \
3120 ) ? (channel == CompositePixelChannel ? RedPixelChannel: channel) \
3123static fxFltType GetHslFlt (
FxInfo * pfx, ssize_t ImgNum,
const fxFltType fx,
const fxFltType fy,
3124 PixelChannel channel)
3126 Image * img = pfx->Images[ImgNum];
3128 double red, green, blue;
3129 double hue=0, saturation=0, lightness=0;
3131 MagickBooleanType okay = MagickTrue;
3132 if(!InterpolatePixelChannel (img, pfx->Imgs[ImgNum].View, RedPixelChannel, img->interpolate,
3133 (
double) fx, (
double) fy, &red, pfx->exception)) okay = MagickFalse;
3134 if(!InterpolatePixelChannel (img, pfx->Imgs[ImgNum].View, GreenPixelChannel, img->interpolate,
3135 (
double) fx, (
double) fy, &green, pfx->exception)) okay = MagickFalse;
3136 if(!InterpolatePixelChannel (img, pfx->Imgs[ImgNum].View, BluePixelChannel, img->interpolate,
3137 (
double) fx, (
double) fy, &blue, pfx->exception)) okay = MagickFalse;
3140 (void) ThrowMagickException (
3141 pfx->exception, GetMagickModule(), OptionError,
3142 "GetHslFlt failure",
"%lu %g,%g %i", (
unsigned long) ImgNum,
3143 (
double) fx, (double) fy, channel);
3147 &hue, &saturation, &lightness);
3149 if (channel == HUE_CHANNEL)
return hue;
3150 if (channel == SAT_CHANNEL)
return saturation;
3151 if (channel == LIGHT_CHANNEL)
return lightness;
3156static fxFltType GetHslInt (
FxInfo * pfx, ssize_t ImgNum,
const ssize_t imgx,
const ssize_t imgy, PixelChannel channel)
3158 Image * img = pfx->Images[ImgNum];
3160 double hue=0, saturation=0, lightness=0;
3162 const Quantum * p = GetCacheViewVirtualPixels (pfx->Imgs[ImgNum].View, imgx, imgy, 1, 1, pfx->exception);
3163 if (p == (
const Quantum *) NULL)
3165 (void) ThrowMagickException (pfx->exception,GetMagickModule(),
3166 OptionError,
"GetHslInt failure",
"%lu %li,%li %i",(
unsigned long) ImgNum,
3167 (
long) imgx,(long) imgy,channel);
3172 GetPixelRed (img, p), GetPixelGreen (img, p), GetPixelBlue (img, p),
3173 &hue, &saturation, &lightness);
3175 if (channel == HUE_CHANNEL)
return hue;
3176 if (channel == SAT_CHANNEL)
return saturation;
3177 if (channel == LIGHT_CHANNEL)
return lightness;
3182static inline fxFltType GetIntensity (
FxInfo * pfx, ssize_t ImgNum,
const fxFltType fx,
const fxFltType fy)
3185 quantum_pixel[MaxPixelChannels];
3190 Image * img = pfx->Images[ImgNum];
3192 (void) GetPixelInfo (img, &pixelinf);
3194 if (!InterpolatePixelInfo (img, pfx->Imgs[pfx->ImgNum].View, img->interpolate,
3195 (
double) fx, (
double) fy, &pixelinf, pfx->exception))
3197 (void) ThrowMagickException (
3198 pfx->exception, GetMagickModule(), OptionError,
3199 "GetIntensity failure",
"%lu %g,%g", (
unsigned long) ImgNum,
3200 (
double) fx, (double) fy);
3203 SetPixelViaPixelInfo (img, &pixelinf, quantum_pixel);
3204 return QuantumScale * GetPixelIntensity (img, quantum_pixel);
3207static MagickBooleanType ExecuteRPN (
FxInfo * pfx,
fxRtT * pfxrt, fxFltType *result,
3208 const PixelChannel channel,
const ssize_t imgx,
const ssize_t imgy)
3210 const Quantum * p = pfxrt->thisPixel;
3211 fxFltType regA=0, regB=0, regC=0, regD=0, regE=0;
3212 Image * img = pfx->image;
3214 MagickBooleanType NeedRelinq = MagickFalse;
3215 double hue=0, saturation=0, lightness=0;
3222 if (!p) p = GetCacheViewVirtualPixels (
3223 pfx->Imgs[pfx->ImgNum].View, imgx, imgy, 1, 1, pfx->exception);
3225 if (p == (
const Quantum *) NULL)
3227 (void) ThrowMagickException (pfx->exception,GetMagickModule(),
3228 OptionError,
"Can't get virtual pixels",
"%lu %li,%li",(
unsigned long)
3229 pfx->ImgNum,(
long) imgx,(long) imgy);
3230 return(MagickFalse);
3233 if (pfx->GotStats) {
3234 cs = pfx->statistics[pfx->ImgNum];
3235 }
else if (pfx->NeedStats) {
3236 cs = CollectOneImgStats (pfx, pfx->Images[pfx->ImgNum]);
3237 NeedRelinq = MagickTrue;
3244 GetPixelRed (img, p), GetPixelGreen (img, p), GetPixelBlue (img, p),
3245 &hue, &saturation, &lightness);
3248 for (i=0; i < pfx->usedElements; i++) {
3253 (void) ThrowMagickException (
3254 pfx->exception, GetMagickModule(), OptionError,
3255 "Bad run-time address",
"%i", i);
3257 pel=&pfx->Elements[i];
3258 switch (pel->number_args) {
3262 regA = PopVal (pfx, pfxrt, i);
3265 regB = PopVal (pfx, pfxrt, i);
3266 regA = PopVal (pfx, pfxrt, i);
3269 regC = PopVal (pfx, pfxrt, i);
3270 regB = PopVal (pfx, pfxrt, i);
3271 regA = PopVal (pfx, pfxrt, i);
3274 regD = PopVal (pfx, pfxrt, i);
3275 regC = PopVal (pfx, pfxrt, i);
3276 regB = PopVal (pfx, pfxrt, i);
3277 regA = PopVal (pfx, pfxrt, i);
3280 regE = PopVal (pfx, pfxrt, i);
3281 regD = PopVal (pfx, pfxrt, i);
3282 regC = PopVal (pfx, pfxrt, i);
3283 regB = PopVal (pfx, pfxrt, i);
3284 regA = PopVal (pfx, pfxrt, i);
3287 (void) ThrowMagickException (
3288 pfx->exception, GetMagickModule(), OptionError,
3289 "Too many args:",
"%i", pel->number_args);
3293 switch (pel->operator_index) {
3295 regA = (pfxrt->UserSymVals[pel->element_index] += regA);
3298 regA = (pfxrt->UserSymVals[pel->element_index] -= regA);
3301 regA = (pfxrt->UserSymVals[pel->element_index] *= regA);
3304 regA = (pfxrt->UserSymVals[pel->element_index] *= PerceptibleReciprocal((
double)regA));
3307 regA = pfxrt->UserSymVals[pel->element_index]++;
3310 regA = pfxrt->UserSymVals[pel->element_index]--;
3322 regA *= PerceptibleReciprocal((
double)regB);
3325 regA = fmod ((
double) regA, fabs(floor((
double) regB+0.5)));
3334 if ((
size_t) (regB+0.5) >= (8*
sizeof(
size_t)))
3336 (void) ThrowMagickException ( pfx->exception, GetMagickModule(),
3337 OptionError,
"undefined shift",
"%g", (double) regB);
3338 regA = (fxFltType) 0.0;
3341 regA = (fxFltType) ((
size_t)(regA+0.5) << (
size_t)(regB+0.5));
3344 if ((
size_t) (regB+0.5) >= (8*
sizeof(
size_t)))
3346 (void) ThrowMagickException ( pfx->exception, GetMagickModule(),
3347 OptionError,
"undefined shift",
"%g", (double) regB);
3348 regA = (fxFltType) 0.0;
3351 regA = (fxFltType) ((
size_t)(regA+0.5) >> (
size_t)(regB+0.5));
3354 regA = fabs((
double) (regA-regB)) < MagickEpsilon ? 1.0 : 0.0;
3357 regA = fabs((
double) (regA-regB)) >= MagickEpsilon ? 1.0 : 0.0;
3360 regA = (regA <= regB) ? 1.0 : 0.0;
3363 regA = (regA >= regB) ? 1.0 : 0.0;
3366 regA = (regA < regB) ? 1.0 : 0.0;
3369 regA = (regA > regB) ? 1.0 : 0.0;
3372 regA = (regA<=0) ? 0.0 : (regB > 0) ? 1.0 : 0.0;
3375 regA = (regA>0) ? 1.0 : (regB > 0.0) ? 1.0 : 0.0;
3378 regA = (regA==0) ? 1.0 : 0.0;
3381 regA = (fxFltType) ((
size_t)(regA+0.5) & (
size_t)(regB+0.5));
3384 regA = (fxFltType) ((
size_t)(regA+0.5) | (
size_t)(regB+0.5));
3388 regA = (fxFltType) (~(
size_t)(regA+0.5));
3391 regA = pow ((
double) regA, (
double) regB);
3407 if (pel->type == etColourConstant) {
3408 switch (channel) {
default:
3409 case (PixelChannel) 0:
3412 case (PixelChannel) 1:
3415 case (PixelChannel) 2:
3425 regA = fabs ((
double) regA);
3427#if defined(MAGICKCORE_HAVE_ACOSH)
3429 regA = acosh ((
double) regA);
3433 regA = acos ((
double) regA);
3435#if defined(MAGICKCORE_HAVE_J1)
3437 if (regA==0) regA = 1.0;
3439 fxFltType gamma = 2.0 * j1 ((MagickPI*regA)) / (MagickPI*regA);
3440 regA = gamma * gamma;
3445 regA = (fxFltType) (((ssize_t) regA) & 0x01 ? -1.0 : 1.0);
3447#if defined(MAGICKCORE_HAVE_ASINH)
3449 regA = asinh ((
double) regA);
3453 regA = asin ((
double) regA);
3455#if defined(MAGICKCORE_HAVE_ATANH)
3457 regA = atanh ((
double) regA);
3461 regA = atan2 ((
double) regA, (
double) regB);
3464 regA = atan ((
double) regA);
3467 regA = ceil ((
double) regA);
3471 case (PixelChannel) 0:
break;
3472 case (PixelChannel) 1: regA = regB;
break;
3473 case (PixelChannel) 2: regA = regC;
break;
3474 case (PixelChannel) 3: regA = regD;
break;
3475 case (PixelChannel) 4: regA = regE;
break;
3476 default: regA = 0.0;
3480 if (regA < 0) regA = 0.0;
3481 else if (regA > 1.0) regA = 1.0;
3484 regA = cosh ((
double) regA);
3487 regA = cos ((
double) regA);
3492 (void) fprintf (stderr,
"%s[%g,%g].[%i]: %s=%.*g\n",
3493 img->filename, (
double) imgx, (double) imgy,
3494 channel, SetPtrShortExp (pfx, pel->exp_start, (
size_t) (pel->exp_len+1)),
3495 pfx->precision, (double) regA);
3498 regA = regA / (regB*(regA-1.0) + 1.0);
3500#if defined(MAGICKCORE_HAVE_ERF)
3502 regA = erf ((
double) regA);
3506 regA = exp ((
double) regA);
3509 regA = floor ((
double) regA);
3512 regA = exp((
double) (-regA*regA/2.0))/sqrt(2.0*MagickPI);
3516 regA = FxGcd (regA, regB, 0);
3519 regA = hypot ((
double) regA, (
double) regB);
3522 regA = floor ((
double) regA);
3525 regA = (fxFltType) (!!IsNaN (regA));
3527#if defined(MAGICKCORE_HAVE_J0)
3529 regA = j0 ((
double) regA);
3532#if defined(MAGICKCORE_HAVE_J1)
3534 regA = j1 ((
double) regA);
3537#if defined(MAGICKCORE_HAVE_J1)
3539 if (regA==0) regA = 1.0;
3540 else regA = 2.0 * j1 ((MagickPI*regA))/(MagickPI*regA);
3544 regA = log ((
double) regA);
3547 regA = MagickLog10((
double) regA) / log10(2.0);
3550 regA = MagickLog10 ((
double) regA);
3553 regA = (regA > regB) ? regA : regB;
3556 regA = (regA < regB) ? regA : regB;
3559 regA = regA - floor((
double) (regA*PerceptibleReciprocal((
double) regB)))*regB;
3562 regA = (fxFltType) (regA < MagickEpsilon);
3565 regA = pow ((
double) regA, (
double) regB);
3568#if defined(MAGICKCORE_OPENMP_SUPPORT)
3569 #pragma omp critical (MagickCore_ExecuteRPN)
3571 regA = GetPseudoRandomValue (pfxrt->random_info);
3575 regA = floor ((
double) regA + 0.5);
3578 regA = (regA < 0) ? -1.0 : 1.0;
3581 regA = sin ((
double) (MagickPI*regA)) / (MagickPI*regA);
3584 regA = sinh ((
double) regA);
3587 regA = sin ((
double) regA);
3590 regA = sqrt ((
double) regA);
3593 regA = 1.0 / (1.0 + exp ((
double) -regA));
3596 regA = tanh ((
double) regA);
3599 regA = tan ((
double) regA);
3602 if (regA >= 0) regA = floor ((
double) regA);
3603 else regA = ceil ((
double) regA);
3615 ssize_t ImgNum = ChkImgNum (pfx, regA);
3616 if (ImgNum < 0)
break;
3617 regA = (fxFltType) 0;
3619 Image * pimg = pfx->Images[0];
3620 if (pel->img_attr_qual == aNull) {
3621 if ((
int) pel->channel_qual < 0) {
3622 if (pel->channel_qual == NO_CHAN_QUAL || pel->channel_qual == THIS_CHANNEL) {
3623 if (pfx->ImgNum==0) {
3624 regA = QuantumScale * (double) p[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3626 const Quantum * pv = GetCacheViewVirtualPixels (
3627 pfx->Imgs[0].View, imgx, imgy, 1,1, pfx->exception);
3629 (void) ThrowMagickException (
3630 pfx->exception, GetMagickModule(), OptionError,
3631 "fU can't get cache",
"%lu", (
unsigned long) ImgNum);
3634 regA = QuantumScale * (double) pv[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3636 }
else if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL ||
3637 pel->channel_qual == LIGHT_CHANNEL) {
3638 regA = GetHslInt (pfx, ImgNum, imgx, imgy, pel->channel_qual);
3640 }
else if (pel->channel_qual == INTENSITY_CHANNEL) {
3641 regA = GetIntensity (pfx, 0, (
double) imgx, (
double) imgy);
3645 if (pfx->ImgNum==0) {
3646 regA = QuantumScale * (double) p[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3648 const Quantum * pv = GetCacheViewVirtualPixels (
3649 pfx->Imgs[0].View, imgx, imgy, 1,1, pfx->exception);
3651 (void) ThrowMagickException (
3652 pfx->exception, GetMagickModule(), OptionError,
3653 "fU can't get cache",
"%lu", (
unsigned long) ImgNum);
3656 regA = QuantumScale * (double) pv[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3661 regA = ImageStat (pfx, 0, WHICH_ATTR_CHAN, pel->img_attr_qual);
3665 if (pel->img_attr_qual == aNull) {
3667 if ((
int) pel->channel_qual < 0) {
3668 if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL ||
3669 pel->channel_qual == LIGHT_CHANNEL)
3671 regA = GetHslInt (pfx, ImgNum, imgx, imgy, pel->channel_qual);
3673 }
else if (pel->channel_qual == INTENSITY_CHANNEL)
3675 regA = GetIntensity (pfx, ImgNum, (fxFltType) imgx, (fxFltType) imgy);
3680 pv = GetCacheViewVirtualPixels (
3681 pfx->Imgs[ImgNum].View, imgx, imgy, 1,1, pfx->exception);
3683 (void) ThrowMagickException (
3684 pfx->exception, GetMagickModule(), OptionError,
3685 "fU can't get cache",
"%lu", (
unsigned long) ImgNum);
3688 regA = QuantumScale * (double)
3689 pv[pfx->Images[ImgNum]->channel_map[WHICH_NON_ATTR_CHAN].offset];
3691 regA = ImageStat (pfx, ImgNum, WHICH_ATTR_CHAN, pel->img_attr_qual);
3700 Image * pimg = pfx->Images[0];
3701 if ((
int) pel->channel_qual < 0) {
3702 if (pel->channel_qual == NO_CHAN_QUAL || pel->channel_qual == THIS_CHANNEL) {
3704 if (pfx->ImgNum==0) {
3705 regA = QuantumScale * (double) p[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3707 const Quantum * pv = GetCacheViewVirtualPixels (
3708 pfx->Imgs[0].View, imgx, imgy, 1,1, pfx->exception);
3710 (void) ThrowMagickException (
3711 pfx->exception, GetMagickModule(), OptionError,
3712 "fU0 can't get cache",
"%i", 0);
3715 regA = QuantumScale * (double) pv[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3718 }
else if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL ||
3719 pel->channel_qual == LIGHT_CHANNEL) {
3720 regA = GetHslInt (pfx, 0, imgx, imgy, pel->channel_qual);
3722 }
else if (pel->channel_qual == INTENSITY_CHANNEL) {
3723 regA = GetIntensity (pfx, 0, (fxFltType) imgx, (fxFltType) imgy);
3726 if (pfx->ImgNum==0) {
3727 regA = QuantumScale * (double) p[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3729 const Quantum * pv = GetCacheViewVirtualPixels (
3730 pfx->Imgs[0].View, imgx, imgy, 1,1, pfx->exception);
3732 (void) ThrowMagickException (
3733 pfx->exception, GetMagickModule(), OptionError,
3734 "fU0 can't get cache",
"%i", 0);
3737 regA = QuantumScale * (double) pv[pimg->channel_map[WHICH_NON_ATTR_CHAN].offset];
3744 ssize_t ImgNum = ChkImgNum (pfx, regA);
3747 if (ImgNum < 0)
break;
3749 if (pel->is_relative) {
3757 if ((
int) pel->channel_qual < 0) {
3758 if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL
3759 || pel->channel_qual == LIGHT_CHANNEL) {
3760 regA = GetHslFlt (pfx, ImgNum, fx, fy, pel->channel_qual);
3762 }
else if (pel->channel_qual == INTENSITY_CHANNEL) {
3763 regA = GetIntensity (pfx, ImgNum, fx, fy);
3770 Image * imUP = pfx->Images[ImgNum];
3771 if (! InterpolatePixelChannel (imUP, pfx->Imgs[ImgNum].View, WHICH_NON_ATTR_CHAN,
3772 imUP->interpolate, (
double) fx, (
double) fy, &v, pfx->exception))
3774 (void) ThrowMagickException (
3775 pfx->exception, GetMagickModule(), OptionError,
3776 "fUP can't get interpolate",
"%lu", (
unsigned long) ImgNum);
3779 regA = v * QuantumScale;
3788 if (pel->operator_index == fS) ImgNum = pfx->ImgNum;
3790 if (pel->img_attr_qual == aNull) {
3791 const Quantum * pv = GetCacheViewVirtualPixels (
3792 pfx->Imgs[ImgNum].View, imgx, imgy, 1,1, pfx->exception);
3794 (void) ThrowMagickException (
3795 pfx->exception, GetMagickModule(), OptionError,
3796 "fV can't get cache",
"%lu", (
unsigned long) ImgNum);
3800 if ((
int) pel->channel_qual < 0) {
3801 if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL ||
3802 pel->channel_qual == LIGHT_CHANNEL) {
3803 regA = GetHslInt (pfx, ImgNum, imgx, imgy, pel->channel_qual);
3805 }
else if (pel->channel_qual == INTENSITY_CHANNEL) {
3806 regA = GetIntensity (pfx, ImgNum, (
double) imgx, (
double) imgy);
3811 regA = QuantumScale * (double)
3812 pv[pfx->Images[ImgNum]->channel_map[WHICH_NON_ATTR_CHAN].offset];
3814 regA = ImageStat (pfx, ImgNum, WHICH_ATTR_CHAN, pel->img_attr_qual);
3824 ssize_t ImgNum = pfx->ImgNum;
3825 if (pel->operator_index == fVP) ImgNum = 1;
3826 if (pel->is_relative) {
3833 if ((
int) pel->channel_qual < 0) {
3834 if (pel->channel_qual == HUE_CHANNEL || pel->channel_qual == SAT_CHANNEL ||
3835 pel->channel_qual == LIGHT_CHANNEL) {
3836 regA = GetHslFlt (pfx, ImgNum, fx, fy, pel->channel_qual);
3838 }
else if (pel->channel_qual == INTENSITY_CHANNEL) {
3839 regA = GetIntensity (pfx, ImgNum, fx, fy);
3847 if (! InterpolatePixelChannel (pfx->Images[ImgNum], pfx->Imgs[ImgNum].View,
3848 WHICH_NON_ATTR_CHAN, pfx->Images[ImgNum]->interpolate,
3849 (
double) fx, (
double) fy, &v, pfx->exception)
3852 (void) ThrowMagickException (
3853 pfx->exception, GetMagickModule(), OptionError,
3854 "fSP or fVP can't get interp",
"%lu", (
unsigned long) ImgNum);
3857 regA = v * (fxFltType)QuantumScale;
3865 regA = (fxFltType) GetImageDepth (img, pfx->exception);
3868 regA = (fxFltType) img->extent;
3872 regA = cs[WHICH_ATTR_CHAN].kurtosis;
3876 regA = cs[WHICH_ATTR_CHAN].maxima;
3880 regA = cs[WHICH_ATTR_CHAN].mean;
3884 regA = cs[WHICH_ATTR_CHAN].median;
3888 regA = cs[WHICH_ATTR_CHAN].minima;
3893 regA = (fxFltType) img->page.x;
3896 regA = (fxFltType) img->page.y;
3899 regA = (fxFltType) img->page.width;
3902 regA = (fxFltType) img->page.height;
3907 regA = (fxFltType) PerceptibleReciprocal (img->resolution.x) * img->columns;
3910 regA = (fxFltType) PerceptibleReciprocal (img->resolution.y) * img->rows;
3913 regA = (fxFltType) img->quality;
3918 regA = (fxFltType) img->resolution.x;
3921 regA = (fxFltType) img->resolution.y;
3925 regA = cs[WHICH_ATTR_CHAN].skewness;
3929 regA = cs[WHICH_ATTR_CHAN].standard_deviation;
3932 regA = (fxFltType) img->rows;
3935 regA = (fxFltType) pfx->ImgListLen;
3938 regA = (fxFltType) pfx->ImgNum;
3941 regA = (fxFltType) img->columns;
3944 regA = (fxFltType) GetImageDepth (img, pfx->exception);
3952 regA = GetIntensity (pfx, pfx->ImgNum, (
double) imgx, (
double) imgy);
3959 regA = QuantumScale * (0.212656 * (double) GetPixelRed (img,p) +
3960 0.715158 * (double) GetPixelGreen (img,p) +
3961 0.072186 * (double) GetPixelBlue (img,p));
3967 regA = QuantumScale * (double) GetPixelAlpha (img, p);
3970 regA = QuantumScale * (double) GetPixelBlue (img, p);
3973 regA = QuantumScale * (double) GetPixelCyan (img, p);
3976 regA = QuantumScale * (double) GetPixelGreen (img, p);
3979 regA = (fxFltType) imgx;
3982 regA = (fxFltType) imgy;
3985 regA = QuantumScale * (double) GetPixelBlack (img, p);
3988 regA = QuantumScale * (double) GetPixelGreen (img, p);
3991 regA = QuantumScale * (double) GetPixelAlpha (img, p);
3994 regA = QuantumScale * (double) GetPixelRed (img, p);
3997 regA = QuantumScale * (double) GetPixelYellow (img, p);
4003 assert (pel->element_index >= 0);
4004 i = pel->element_index-1;
4007 assert (pel->element_index >= 0);
4008 i = pel->element_index-1;
4009 if (IsImageTTLExpired(img) != MagickFalse) {
4010 i = pfx->usedElements-1;
4011 (void) ThrowMagickException (pfx->exception, GetMagickModule(),
4012 ResourceLimitFatalError,
"TimeLimitExceeded",
"`%s'", img->filename);
4016 assert (pel->element_index >= 0);
4017 if (fabs((
double) regA) < MagickEpsilon) i = pel->element_index-1;
4019 case rIfNotZeroGoto:
4020 assert (pel->element_index >= 0);
4021 if (fabs((
double) regA) > MagickEpsilon) i = pel->element_index-1;
4024 assert (pel->element_index >= 0);
4025 regA = pfxrt->UserSymVals[pel->element_index];
4028 assert (pel->element_index >= 0);
4029 pfxrt->UserSymVals[pel->element_index] = regA;
4032 pfxrt->usedValStack = 0;
4038 (void) ThrowMagickException (
4039 pfx->exception, GetMagickModule(), OptionError,
4040 "pel->oprNum",
"%i '%s' not yet implemented",
4041 (int)pel->operator_index, OprStr(pel->operator_index));
4045 if (!PushVal (pfx, pfxrt, regA, i))
break;
4048 if (pfxrt->usedValStack > 0) regA = PopVal (pfx, pfxrt, 9999);
4054 if (pfx->exception->severity >= ErrorException)
4057 if (pfxrt->usedValStack != 0) {
4058 (void) ThrowMagickException (
4059 pfx->exception, GetMagickModule(), OptionError,
4060 "ValStack not empty",
"(%i)", pfxrt->usedValStack);
4069MagickPrivate MagickBooleanType FxEvaluateChannelExpression (
4071 const PixelChannel channel,
const ssize_t x,
const ssize_t y,
4075 id = GetOpenMPThreadId();
4079 assert (pfx != NULL);
4080 assert (pfx->image != NULL);
4081 assert (pfx->Images != NULL);
4082 assert (pfx->Imgs != NULL);
4083 assert (pfx->fxrts != NULL);
4085 pfx->fxrts[id].thisPixel = NULL;
4087 if (!ExecuteRPN (pfx, &pfx->fxrts[
id], &ret, channel, x, y)) {
4088 (void) ThrowMagickException (
4089 exception, GetMagickModule(), OptionError,
4090 "ExecuteRPN failed",
" ");
4094 *result = (double) ret;
4099static FxInfo *AcquireFxInfoPrivate (
const Image * images,
const char * expression,
4104 FxInfo * pfx = (
FxInfo*) AcquireCriticalMemory (
sizeof (*pfx));
4106 memset (pfx, 0,
sizeof (*pfx));
4108 if (!InitFx (pfx, images, CalcAllStats, exception)) {
4109 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4113 if (!BuildRPN (pfx)) {
4114 (void) DeInitFx (pfx);
4115 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4119 if ((*expression ==
'@') && (strlen(expression) > 1))
4120 pfx->expression=FileToString(expression,~0UL,exception);
4121 if (pfx->expression == (
char *) NULL)
4122 pfx->expression=ConstantString(expression);
4123 pfx->pex = (
char *) pfx->expression;
4126 if (!TranslateStatementList (pfx,
";", &chLimit)) {
4127 (void) DestroyRPN (pfx);
4128 pfx->expression = DestroyString (pfx->expression);
4130 (void) DeInitFx (pfx);
4131 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4136 (void) ThrowMagickException (
4137 pfx->exception, GetMagickModule(), OptionError,
4138 "Translate expression depth",
"(%i) not 0",
4141 (void) DestroyRPN (pfx);
4142 pfx->expression = DestroyString (pfx->expression);
4144 (void) DeInitFx (pfx);
4145 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4149 if (chLimit !=
'\0' && chLimit !=
';') {
4150 (void) ThrowMagickException (
4151 pfx->exception, GetMagickModule(), OptionError,
4152 "AcquireFxInfo: TranslateExpression did not exhaust input",
"(chLimit=%i) at'%s'",
4153 (int)chLimit, pfx->pex);
4155 (void) DestroyRPN (pfx);
4156 pfx->expression = DestroyString (pfx->expression);
4158 (void) DeInitFx (pfx);
4159 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4163 if (pfx->NeedStats && pfx->runType == rtEntireImage && !pfx->statistics) {
4164 if (!CollectStatistics (pfx)) {
4165 (void) DestroyRPN (pfx);
4166 pfx->expression = DestroyString (pfx->expression);
4168 (void) DeInitFx (pfx);
4169 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4174 if (pfx->DebugOpt) {
4175 DumpTables (stderr);
4176 DumpUserSymbols (pfx, stderr);
4177 (void) DumpRPN (pfx, stderr);
4181 size_t number_threads=(size_t) GetMagickResourceLimit(ThreadResource);
4184 pfx->fxrts = (
fxRtT *)AcquireQuantumMemory (number_threads,
sizeof(
fxRtT));
4186 (void) ThrowMagickException (
4187 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
4189 (
unsigned long) number_threads);
4190 (void) DestroyRPN (pfx);
4191 pfx->expression = DestroyString (pfx->expression);
4193 (void) DeInitFx (pfx);
4194 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4197 for (t=0; t < (ssize_t) number_threads; t++) {
4198 if (!AllocFxRt (pfx, &pfx->fxrts[t])) {
4199 (void) ThrowMagickException (
4200 pfx->exception, GetMagickModule(), ResourceLimitFatalError,
4201 "AllocFxRt t=",
"%g",
4205 for (t2 = t-1; t2 >= 0; t2--) {
4206 DestroyFxRt (&pfx->fxrts[t]);
4209 pfx->fxrts = (
fxRtT *) RelinquishMagickMemory (pfx->fxrts);
4210 (void) DestroyRPN (pfx);
4211 pfx->expression = DestroyString (pfx->expression);
4213 (void) DeInitFx (pfx);
4214 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4224 return AcquireFxInfoPrivate (images, expression, MagickFalse, exception);
4231 assert (pfx != NULL);
4232 assert (pfx->image != NULL);
4233 assert (pfx->Images != NULL);
4234 assert (pfx->Imgs != NULL);
4235 assert (pfx->fxrts != NULL);
4237 for (t=0; t < (ssize_t) GetMagickResourceLimit(ThreadResource); t++) {
4238 DestroyFxRt (&pfx->fxrts[t]);
4240 pfx->fxrts = (
fxRtT *) RelinquishMagickMemory (pfx->fxrts);
4244 pfx->expression = DestroyString (pfx->expression);
4247 (void) DeInitFx (pfx);
4249 pfx = (
FxInfo*) RelinquishMagickMemory(pfx);
4256MagickExport
Image *FxImage(
const Image *image,
const char *expression,
4259#define FxImageTag "FxNew/Image"
4280 assert(image != (
Image *) NULL);
4281 assert(image->signature == MagickCoreSignature);
4282 if (IsEventLogging() != MagickFalse)
4283 (void) LogMagickEvent(TraceEvent,GetMagickModule(),
"%s",image->filename);
4284 if (expression == (
const char *) NULL)
4285 return(CloneImage(image,0,0,MagickTrue,exception));
4286 fx_image=CloneImage(image,0,0,MagickTrue,exception);
4287 if (!fx_image)
return NULL;
4288 if (SetImageStorageClass(fx_image,DirectClass,exception) == MagickFalse) {
4289 fx_image=DestroyImage(fx_image);
4293 pfx = AcquireFxInfoPrivate (image, expression, MagickTrue, exception);
4296 fx_image=DestroyImage(fx_image);
4300 assert (pfx->image != NULL);
4301 assert (pfx->Images != NULL);
4302 assert (pfx->Imgs != NULL);
4303 assert (pfx->fxrts != NULL);
4307 image_view = AcquireVirtualCacheView (image, pfx->exception);
4308 fx_view = AcquireAuthenticCacheView (fx_image, pfx->exception);
4309#if defined(MAGICKCORE_OPENMP_SUPPORT)
4310 #pragma omp parallel for schedule(dynamic) shared(progress,status) \
4311 magick_number_threads(image,fx_image,fx_image->rows, \
4312 pfx->ContainsDebug ? 0 : 1)
4314 for (y=0; y < (ssize_t) fx_image->rows; y++)
4317 id = GetOpenMPThreadId();
4331 if (status == MagickFalse)
4333 p = GetCacheViewVirtualPixels (image_view, 0, y, image->columns, 1, pfx->exception);
4334 q = QueueCacheViewAuthenticPixels (fx_view, 0, y, fx_image->columns, 1, pfx->exception);
4335 if ((p == (
const Quantum *) NULL) || (q == (Quantum *) NULL)) {
4339 for (x=0; x < (ssize_t) fx_image->columns; x++) {
4342 pfx->fxrts[id].thisPixel = (Quantum *)p;
4344 for (i=0; i < (ssize_t) GetPixelChannels(image); i++)
4346 PixelChannel channel = GetPixelChannelChannel (image, i);
4347 PixelTrait traits = GetPixelChannelTraits (image, channel);
4348 PixelTrait fx_traits = GetPixelChannelTraits (fx_image, channel);
4349 if ((traits == UndefinedPixelTrait) ||
4350 (fx_traits == UndefinedPixelTrait))
4352 if ((fx_traits & CopyPixelTrait) != 0) {
4353 SetPixelChannel (fx_image, channel, p[i], q);
4357 if (!ExecuteRPN (pfx, &pfx->fxrts[
id], &result, channel, x, y)) {
4362 q[i] = ClampToQuantum ((MagickRealType) (QuantumRange*result));
4364 p+=(ptrdiff_t) GetPixelChannels (image);
4365 q+=(ptrdiff_t) GetPixelChannels (fx_image);
4367 if (SyncCacheViewAuthenticPixels(fx_view, pfx->exception) == MagickFalse)
4369 if (image->progress_monitor != (MagickProgressMonitor) NULL)
4374#if defined(MAGICKCORE_OPENMP_SUPPORT)
4378 proceed = SetImageProgress (image, FxImageTag, progress, image->rows);
4379 if (proceed == MagickFalse)
4384 fx_view = DestroyCacheView (fx_view);
4385 image_view = DestroyCacheView (image_view);
4389 if (pfx->DebugOpt && pfx->usedUserSymbols) {
4391 char UserSym[MagickPathExtent];
4392 fprintf (stderr,
"User symbols (%i):\n", pfx->usedUserSymbols);
4393 for (t=0; t < (int) GetMagickResourceLimit(ThreadResource); t++) {
4394 for (i = 0; i < (int) pfx->usedUserSymbols; i++) {
4395 fprintf (stderr,
"th=%i us=%i '%s': %.*Lg\n",
4396 t, i, NameOfUserSym (pfx, i, UserSym), pfx->precision, pfx->fxrts[t].UserSymVals[i]);
4401 if ((status == MagickFalse) || (pfx->exception->severity >= ErrorException))
4402 fx_image=DestroyImage(fx_image);
4404 pfx=DestroyFxInfo(pfx);