-------------------------------------------------------------------------------------------------- ------ This file is a copy of some part of PGF/Tikz. ------ It has been copied here to provide : ------ - compatibility with older PGF versions ------ - availability of PGF contributions by Christian Feuersaenger ------ which are necessary or helpful for pgfplots. ------ ------ For reasons of simplicity, I have copied the whole file, including own contributions AND ------ PGF parts. The copyrights are as they appear in PGF. ------ ------ Note that pgfplots has compatible licenses. ------ ------ This copy has been modified in the following ways: ------ - nested \input commands have been updated ------ -- -- Support for the contents of this file will NOT be done by the PGF/TikZ team. -- Please contact the author and/or maintainer of pgfplots (Christian Feuersaenger) if you need assistance in conjunction -- with the deployment of this patch or partial content of PGF. Note that the author and/or maintainer of pgfplots has no obligation to fix anything: -- This file comes without any warranty as the rest of pgfplots; there is no obligation for help. ---------------------------------------------------------------------------------------------------- -- Date of this copy: Mi 6. Jan 11:32:04 CET 2016 --- -- Copyright 2011 by Christophe Jorssen and Mark Wibrow -- Copyright 2014 by Christian Feuersaenger -- -- This file may be distributed and/or modified -- -- 1. under the LaTeX Project Public License and/or -- 2. under the GNU Public License. -- -- See the file doc/generic/pgf/licenses/LICENSE for more details. -- -- $Id: parser.lua,v 1.3 2015/11/28 17:20:49 cfeuersaenger Exp $ -- -- usage: -- -- pgfluamathparser = require("pgfplotsoldpgfsupp.luamath.parser") -- -- local result = pgfluamathparser.pgfmathparse("1+ 2*4^2") -- -- This LUA class has a direct backend in \pgfuselibrary{luamath}, see the documentation of that TeX package. local pgfluamathparser = pgfluamathparser or {} pgfluamathfunctions = require("pgfplotsoldpgfsupp.luamath.functions") -- lpeg is always present in luatex local lpeg = require("lpeg") local S, P, R = lpeg.S, lpeg.P, lpeg.R local C, Cc, Ct = lpeg.C, lpeg.Cc, lpeg.Ct local Cf, Cg, Cs = lpeg.Cf, lpeg.Cg, lpeg.Cs local V = lpeg.V local match = lpeg.match local space_pattern = S(" \n\r\t")^0 local tex_unit = P('pt') + P('mm') + P('cm') + P('in') + -- while valid units, the font-depending ones need special attention... move them to the TeX side. For now. -- P('ex') + P('em') + P('bp') + P('pc') + P('dd') + P('cc') + P('sp'); local one_digit_pattern = R("09") local positive_integer_pattern = one_digit_pattern^1 -- FIXME : it might be a better idea to remove '-' from all number_patterns! Instead, rely on the prefix operator 'neg' to implement negative numbers. -- Is that wise? It is certainly less efficient... local integer_pattern = S("+-")^-1 * positive_integer_pattern -- Valid positive decimals are |xxx.xxx|, |.xxx| and |xxx.| local positive_integer_or_decimal_pattern = positive_integer_pattern * ( P(".") * one_digit_pattern^0)^-1 + (P(".") * one_digit_pattern^1) local integer_or_decimal_pattern = S("+-")^-1 * positive_integer_or_decimal_pattern local fpu_pattern = R"05" * P"Y" * positive_integer_or_decimal_pattern * P"e" * S("+-")^-1 * R("09")^1 * P"]" local unbounded_pattern = P"inf" + P"INF" + P"nan" + P"NaN" + P"Inf" local number_pattern = C(unbounded_pattern + fpu_pattern + integer_or_decimal_pattern * (S"eE" * integer_pattern + C(tex_unit))^-1) local underscore_pattern = P("_") local letter_pattern = R("az","AZ") local alphanum__pattern = letter_pattern + one_digit_pattern + underscore_pattern local identifier_pattern = letter_pattern^1 * alphanum__pattern^0 local openparen_pattern = P("(") * space_pattern local closeparen_pattern = P(")") local opencurlybrace_pattern = P("{") local closecurlybrace_pattern = P("}") local openbrace_pattern = P("[") local closebrace_pattern = P("]") -- hm. what about '\\' or '\%' ? -- accept \pgf@x, \count0, \dimen42, \c@pgf@counta, \wd0, \ht0, \dp 0 local controlsequence_pattern = P"\\" * C( (R("az","AZ") + P"@")^1) * space_pattern* C( R"09"^0 ) -- local string = P('"') * C((1 - P('"'))^0) * P('"') local comma_pattern = P(",") * space_pattern ---------------- local TermOp = C(S("+-")) * space_pattern local EqualityOp = C( P"==" + P"!=" ) * space_pattern local RelationalOp = C( P"<=" + P">=" + P"<" + P">" ) * space_pattern local FactorOp = C(S("*/")) * space_pattern -- Grammar local Exp, Term, Factor = V"Exp", V"Term", V"Factor" local Prefix = V"Prefix" local Postfix = V"Postfix" local function eval (v1, op, v2) if (op == "+") then return v1 + v2 elseif (op == "-") then return v1 - v2 elseif (op == "*") then return v1 * v2 elseif (op == "/") then return v1 / v2 else error("This function must not be invoked for operator "..op) end end local pgfStringToFunctionMap = pgfluamathfunctions.stringToFunctionMap local function function_eval(name, ... ) local f = pgfStringToFunctionMap[name] if not f then error("Function '" .. name .. "' is undefined (did not find pgfluamathfunctions."..name .." (looked into pgfluamathfunctions.stringToFunctionMap))") end -- FIXME: validate signature return f(...) end local func = (C(identifier_pattern) * space_pattern * openparen_pattern * Exp * (comma_pattern * Exp)^0 * closeparen_pattern) / function_eval; local functionWithoutArg = identifier_pattern / function_eval -- this is what can occur as exponent after '^'. -- I have the impression that the priorities could be implemented in a better way than this... but it seems to work. local pow_exponent = -- allows 2^-4, 2^1e4, 2^2 -- FIXME : why not 2^1e2 ? Cg(C(integer_or_decimal_pattern) -- 2^pi, 2^multiply(2,2) + Cg(func+functionWithoutArg) -- 2^(2+2) + openparen_pattern * Exp * closeparen_pattern ) local function prefix_eval(op, x) if op == "-" then return pgfluamathfunctions.neg(x) elseif op == "!" then return pgfluamathfunctions.notPGF(x) else error("This function must not be invoked for operator "..op) end end local prefix_operator = C( S"-!" ) local prefix_operator_pattern = (prefix_operator * space_pattern * Cg(Prefix) ) / prefix_eval -- apparently, we need to distinghuish between ! and != : local postfix_operator = C( S"r!" - P"!=" ) + C(P"^") * space_pattern * pow_exponent local ternary_eval = pgfluamathfunctions.ifthenelse local factorial_eval = pgfluamathfunctions.factorial local deg = pgfluamathfunctions.deg local pow_eval = pgfluamathfunctions.pow -- @param prefix the argument before the postfix operator. -- @param op either nil or the postfix operator -- @param arg either nil or the (mandatory) argument for 'op' local function postfix_eval(prefix, op, arg) local result if op == nil then result = prefix elseif op == "r" then if arg then error("parser setup error: expected nil argument") end result = deg(prefix) elseif op == "!" then if arg then error("parser setup error: expected nil argument") end result = factorial_eval(prefix) elseif op == "^" then if not arg then error("parser setup error: ^ with its argument") end result = pow_eval(prefix, arg) else error("Parser setup error: " .. tostring(op) .. " unexpected in this context") end return result end local function equality_eval(v1, op, v2) local fct if (op == "==") then fct = pgfluamathfunctions.equal elseif (op == "!=") then fct = pgfluamathfunctions.notequal else error("This function must not be invoked for operator "..op) end return fct(v1,v2) end local function relational_eval(v1, op, v2) local fct if (op == "<") then fct = pgfluamathfunctions.less elseif (op == ">") then fct = pgfluamathfunctions.greater elseif (op == ">=") then fct = pgfluamathfunctions.notless elseif (op == "<=") then fct = pgfluamathfunctions.notgreater else error("This function must not be invoked for operator "..op) end return fct(v1,v2) end -- @return either the box property or nil -- @param cs "wd", "ht", or "dp" -- @param intSuffix some integer local function get_tex_box(cs, intSuffix) -- assume get_tex_box is only called when a dimension is required. local result pgfluamathparser.units_declared = true local box =tex.box[tonumber(intSuffix)] if not box then error("There is no box " .. intSuffix) end if cs == "wd" then result = box.width / 65536 elseif cs == "ht" then result = box.height / 65536 elseif cs == "dp" then result = box.depth / 65536 else result = nil end return result end local function controlsequence_eval(cs, intSuffix) local result if intSuffix and #intSuffix >0 then if cs == "count" then result= pgfluamathparser.get_tex_count(intSuffix) elseif cs == "dimen" then result= pgfluamathparser.get_tex_dimen(intSuffix) else result = get_tex_box(cs,intSuffix) if not result then -- this can happen - we cannot expand \chardef'ed boxes here. -- this will be done by the TeX part error('I do not know/support the TeX register "\\' .. cs .. '"') end end else result = pgfluamathparser.get_tex_register(cs) end return result end pgfluamathparser.units_declared = false function pgfluamathparser.get_tex_register(register) -- register is a string which could be a count or a dimen. if pcall(tex.getcount, register) then return tex.count[register] elseif pcall(tex.getdimen, register) then pgfluamathparser.units_declared = true return tex.dimen[register] / 65536 -- return in points. else error('I do not know the TeX register "' .. register .. '"') return nil end end function pgfluamathparser.get_tex_count(count) -- count is expected to be a number return tex.count[tonumber(count)] end function pgfluamathparser.get_tex_dimen(dimen) -- dimen is expected to be a number pgfluamathparser.units_declared = true return tex.dimen[tonumber(dimen)] / 65536 end function pgfluamathparser.get_tex_sp(dimension) -- dimension should be a string pgfluamathparser.units_declared = true return tex.sp(dimension) / 65536 end local initialRule = V"initial" local Summand = V"Summand" local Relational = V"Relational" local Equality = V"Equality" local LogicalOr = V"LogicalOr" local LogicalAnd = V"LogicalAnd" local pgftonumber = pgfluamathfunctions.tonumber local tonumber_withunit = pgfluamathparser.get_tex_sp local function number_optional_units_eval(x, unit) if not unit then return pgftonumber(x) else return tonumber_withunit(x) end end -- @param scale the number. -- @param controlsequence either nil in which case just the number must be returned or a control sequence -- @see controlsequence_eval local function scaled_controlsequence_eval(scale, controlsequence, intSuffix) if controlsequence==nil then return scale else return scale * controlsequence_eval(controlsequence, intSuffix) end end -- Grammar -- -- for me: -- - use '/' to evaluate all expressions which contain a _constant_ number of captures. -- - use Cf to evaluate expressions which contain a _dynamic_ number of captures -- -- see unittest_luamathparser.tex for tons of examples local G = P{ "initialRule", initialRule = space_pattern* Exp * -1; -- ternary operator (or chained ternary operators): -- FIXME : is this chaining a good idea!? Exp = Cf( LogicalOr * Cg(P"?" * space_pattern * LogicalOr * P":" *space_pattern * LogicalOr )^0, ternary_eval) ; LogicalOr = Cf(LogicalAnd * (P"||" * space_pattern * LogicalAnd)^0, pgfluamathfunctions.orPGF); LogicalAnd = Cf(Equality * (P"&&" * space_pattern * Equality)^0, pgfluamathfunctions.andPGF); Equality = Cf(Relational * Cg(EqualityOp * Relational)^0, equality_eval); Relational = Cf(Summand * Cg(RelationalOp * Summand)^0, relational_eval); Summand = Cf(Term * Cg(TermOp * Term)^0, eval) ; Term = Cf(Prefix * Cg(FactorOp * Prefix)^0, eval); Prefix = prefix_operator_pattern + Postfix; -- this calls 'postfix_eval' with nil arguments if it is no postfix operation.. but that does not hurt (right?) Postfix = Factor * (postfix_operator * space_pattern)^-1 / postfix_eval; Factor = ( number_pattern / number_optional_units_eval * -- this construction will evaluate number_pattern with 'number_optional_units_eval' FIRST. -- also accept '0.5 \pgf@x' here: space_pattern *controlsequence_pattern^-1 / scaled_controlsequence_eval + func + functionWithoutArg + openparen_pattern * Exp * closeparen_pattern + controlsequence_pattern / controlsequence_eval ) *space_pattern ; } -- does not reset units_declared. local function pgfmathparseinternal(str) local result = match(G,str) if result == nil then error("The string '" .. str .. "' is no valid PGF math expression. Please check for syntax errors.") end return result end -- This is the math parser function in this module. -- -- @param str a string like "1+1" which is accepted by the PGF math language -- @return the result of the expression. -- -- Throws an error if the string is no valid expression. function pgfluamathparser.pgfmathparse(str) pgfluamathparser.units_declared = false return pgfmathparseinternal(str) end local pgfmathparse = pgfluamathparser.pgfmathparse local tostringfixed = pgfluamathfunctions.tostringfixed local tostringfpu = pgfluamathfunctions.toTeXstring local tmpFunctionArgumentPrefix = "tmpVar" local stackOfLocalFunctions = {} -- This is a backend for PGF's 'declare function'. -- \tikzset{declare function={mu(\x,\i)=\x^\i;}} -- will boil down to -- pgfluamathparser.declareExpressionFunction("mu", 2, "#1^#2") -- -- The local function will be pushed on a stack of known local functions and is -- available until popLocalExpressionFunction() is called. TeX will call this using -- \aftergroup. -- -- @param name the name of the new function -- @param numArgs the number of arguments -- @param expression an expression containing #1, ... #n where n is numArgs -- -- ATTENTION: local functions behave DIFFERENTLY in LUA! -- In LUA, local variables are not expanded whereas TeX expands them. -- The difference is -- -- declare function={mu1(\x,\i)=\x^\i;} -- \pgfmathparse{mu1(-5,2)} --> -25 -- \pgfluamathparse{mu1(-5,2)} --> 25 -- -- x = -5 -- \pgfmathparse{mu1(x,2)} --> 25 -- \pgfluamathparse{mu1(x,2)} --> 25 -- -- In an early prototype, I simulated TeX's expansion to fix the first case (successfully). -- BUT: that "simulated expansion" broke the second case because LUA will evaluate "x" and hand -5 to the local function. -- I decided to keep it as is. Perhaps we should fix PGF's expansion approach in TeX (which is ugly anyway) function pgfluamathparser.pushLocalExpressionFunction(name, numArgs, expression) -- now we have "tmpVar1^tmpVar2" instead of "#1^#2" local normalizedExpr = expression:gsub("#", tmpFunctionArgumentPrefix) local restores = {} local tmpVars = {} for i=1,numArgs do local tmpVar = tmpFunctionArgumentPrefix .. tostring(i) tmpVars[i] = tmpVar end local newFunction = function(...) local args = table.pack(...) -- define "tmpVar1" ... "tmpVarN" to return args[i]. -- Of course, we need to restore "tmpVar" after we return! for i=1,numArgs do local tmpVar = tmpVars[i] local value = args[i] restores[i] = pgfStringToFunctionMap[tmpVar] pgfStringToFunctionMap[tmpVar] = function () return value end end -- parse our expression. -- FIXME : this here is an attempt to mess around with "units_declared". -- It would be better to call pgfmathparse and introduce some -- semaphore to check if pgfmathparse is a nested call-- in this case, it should -- not reset units_declared. But there is no "finally" block and pcall is crap (looses stack trace). local success,result = pcall(pgfmathparseinternal, normalizedExpr) -- remove 'tmpVar1', ... from the function table: for i=1,numArgs do local tmpVar = tmpVars[i] pgfStringToFunctionMap[tmpVar] = restores[i] end if success==false then error(result) end return result end table.insert(stackOfLocalFunctions, name) pgfStringToFunctionMap[name] = newFunction end function pgfluamathparser.popLocalExpressionFunction() local name = stackOfLocalFunctions[#stackOfLocalFunctions] pgfStringToFunctionMap[name] = nil -- this removes the last element: table.remove(stackOfLocalFunctions) end -- A Utility function which simplifies the interaction with the TeX code -- @param expression the input expression (string) -- @param outputFormatChoice 0 if the result should be a fixed point number, 1 if it should be in FPU format -- @param showErrorMessage (boolean) true if any error should be displayed, false if errors should simply result in an invocation of TeX's parser (the default) -- -- it defines \pgfmathresult and \ifpgfmathunitsdeclared function pgfluamathparser.texCallParser(expression, outputFormatChoice, showErrorMessage) local success, result if showErrorMessage then result = pgfmathparse(expression) success = true else success, result = pcall(pgfmathparse, expression) end if success and result then local result_str if outputFormatChoice == 0 then -- luamath/output format=fixed result_str = tostringfixed(result) else -- luamath/output format=fixed result_str = tostringfpu(result) end tex.sprint("\\def\\pgfmathresult{" .. result_str .. "}") if pgfluamathparser.units_declared then tex.sprint("\\pgfmathunitsdeclaredtrue") else tex.sprint("\\pgfmathunitsdeclaredfalse") end else tex.sprint("\\def\\pgfmathresult{}") tex.sprint("\\pgfmathunitsdeclaredfalse") end end return pgfluamathparser