00000 | /* 00001 | ****************************************************************** 00002 | * C++ Mathematical Expression Toolkit Library * 00003 | * * 00004 | * Author: Arash Partow (1999-2024) * 00005 | * URL: https://www.partow.net/programming/exprtk/index.html * 00006 | * * 00007 | * Copyright notice: * 00008 | * Free use of the C++ Mathematical Expression Toolkit Library is * 00009 | * permitted under the guidelines and in accordance with the most * 00010 | * current version of the MIT License. * 00011 | * https://www.opensource.org/licenses/MIT * 00012 | * SPDX-License-Identifier: MIT * 00013 | * * 00014 | * Example expressions: * 00015 | * (00) (y + x / y) * (x - y / x) * 00016 | * (01) (x^2 / sin(2 * pi / y)) - x / 2 * 00017 | * (02) sqrt(1 - (x^2)) * 00018 | * (03) 1 - sin(2 * x) + cos(pi / y) * 00019 | * (04) a * exp(2 * t) + c * 00020 | * (05) if(((x + 2) == 3) and ((y + 5) <= 9), 1 + w, 2 / z) * 00021 | * (06) (avg(x,y) <= x + y ? x - y : x * y) + 2 * pi / x * 00022 | * (07) z := x + sin(2 * pi / y) * 00023 | * (08) u := 2 * (pi * z) / (w := x + cos(y / pi)) * 00024 | * (09) clamp(-1, sin(2 * pi * x) + cos(y / 2 * pi), +1) * 00025 | * (10) inrange(-2, m, +2) == if(({-2 <= m} and [m <= +2]), 1, 0) * 00026 | * (11) (2sin(x)cos(2y)7 + 1) == (2 * sin(x) * cos(2*y) * 7 + 1) * 00027 | * (12) (x ilike 's*ri?g') and [y < (3 z^7 + w)] * 00028 | * * 00029 | ****************************************************************** 00030 | */ 00031 | 00032 | 00033 | #ifndef INCLUDE_EXPRTK_HPP 00034 | #define INCLUDE_EXPRTK_HPP 00035 | 00036 | 00037 | #include <algorithm> 00038 | #include <cassert> 00039 | #include <cctype> 00040 | #include <cmath> 00041 | #include <cstdio> 00042 | #include <cstdlib> 00043 | #include <cstring> 00044 | #include <deque> 00045 | #include <functional> 00046 | #include <iterator> 00047 | #include <limits> 00048 | #include <list> 00049 | #include <map> 00050 | #include <set> 00051 | #include <stack> 00052 | #include <stdexcept> 00053 | #include <string> 00054 | #include <utility> 00055 | #include <vector> 00056 | 00057 | 00058 | namespace exprtk 00059 | { 00060 | #ifdef exprtk_enable_debugging 00061 | #define exprtk_debug(params) printf params 00062 | #else 00063 | #define exprtk_debug(params) (void)0 00064 | #endif 00065 | 00066 | #define exprtk_error_location \ 00067 | "exprtk.hpp:" + details::to_str(__LINE__) \ 00068 | 00069 | #if __cplusplus >= 201103L 00070 | #define exprtk_override override 00071 | #define exprtk_final final 00072 | #define exprtk_delete = delete 00073 | #else 00074 | #define exprtk_override 00075 | #define exprtk_final 00076 | #define exprtk_delete 00077 | #endif 00078 | 00079 | #if __cplusplus >= 201603L 00080 | #define exprtk_fallthrough [[fallthrough]]; 00081 | #elif (__cplusplus >= 201103L) && (defined(__GNUC__) && !defined(__clang__)) 00082 | #define exprtk_fallthrough [[gnu::fallthrough]]; 00083 | #else 00084 | #ifndef _MSC_VER 00085 | #define exprtk_fallthrough __attribute__ ((fallthrough)); 00086 | #else 00087 | #define exprtk_fallthrough 00088 | #endif 00089 | #endif 00090 | 00091 | namespace details 00092 | { 00093 | typedef char char_t; 00094 | typedef char_t* char_ptr; 00095 | typedef char_t const* char_cptr; 00096 | typedef unsigned char uchar_t; 00097 | typedef uchar_t* uchar_ptr; 00098 | typedef uchar_t const* uchar_cptr; 00099 | typedef unsigned long long int _uint64_t; 00100 | typedef long long int _int64_t; 00101 | 00102 | inline bool is_whitespace(const char_t c) 00103 | { 00104 | return (' ' == c) || ('\n' == c) || 00105 | ('\r' == c) || ('\t' == c) || 00106 | ('\b' == c) || ('\v' == c) || 00107 | ('\f' == c) ; 00108 | } 00109 | 00110 | inline bool is_operator_char(const char_t c) 00111 | { 00112 | return ('+' == c) || ('-' == c) || 00113 | ('*' == c) || ('/' == c) || 00114 | ('^' == c) || ('<' == c) || 00115 | ('>' == c) || ('=' == c) || 00116 | (',' == c) || ('!' == c) || 00117 | ('(' == c) || (')' == c) || 00118 | ('[' == c) || (']' == c) || 00119 | ('{' == c) || ('}' == c) || 00120 | ('%' == c) || (':' == c) || 00121 | ('?' == c) || ('&' == c) || 00122 | ('|' == c) || (';' == c) ; 00123 | } 00124 | 00125 | inline bool is_letter(const char_t c) 00126 | { 00127 | return (('a' <= c) && (c <= 'z')) || 00128 | (('A' <= c) && (c <= 'Z')) ; 00129 | } 00130 | 00131 | inline bool is_digit(const char_t c) 00132 | { 00133 | return ('0' <= c) && (c <= '9'); 00134 | } 00135 | 00136 | inline bool is_letter_or_digit(const char_t c) 00137 | { 00138 | return is_letter(c) || is_digit(c); 00139 | } 00140 | 00141 | inline bool is_left_bracket(const char_t c) 00142 | { 00143 | return ('(' == c) || ('[' == c) || ('{' == c); 00144 | } 00145 | 00146 | inline bool is_right_bracket(const char_t c) 00147 | { 00148 | return (')' == c) || (']' == c) || ('}' == c); 00149 | } 00150 | 00151 | inline bool is_bracket(const char_t c) 00152 | { 00153 | return is_left_bracket(c) || is_right_bracket(c); 00154 | } 00155 | 00156 | inline bool is_sign(const char_t c) 00157 | { 00158 | return ('+' == c) || ('-' == c); 00159 | } 00160 | 00161 | inline bool is_invalid(const char_t c) 00162 | { 00163 | return !is_whitespace (c) && 00164 | !is_operator_char(c) && 00165 | !is_letter (c) && 00166 | !is_digit (c) && 00167 | ('.' != c) && 00168 | ('_' != c) && 00169 | ('$' != c) && 00170 | ('~' != c) && 00171 | ('\'' != c); 00172 | } 00173 | 00174 | inline bool is_valid_string_char(const char_t c) 00175 | { 00176 | return std::isprint(static_cast<uchar_t>(c)) || 00177 | is_whitespace(c); 00178 | } 00179 | 00180 | #ifndef exprtk_disable_caseinsensitivity 00181 | inline void case_normalise(std::string& s) 00182 | { 00183 | for (std::size_t i = 0; i < s.size(); ++i) 00184 | { 00185 | s[i] = static_cast<std::string::value_type>(std::tolower(s[i])); 00186 | } 00187 | } 00188 | 00189 | inline bool imatch(const char_t c1, const char_t c2) 00190 | { 00191 | return std::tolower(c1) == std::tolower(c2); 00192 | } 00193 | 00194 | inline bool imatch(const std::string& s1, const std::string& s2) 00195 | { 00196 | if (s1.size() == s2.size()) 00197 | { 00198 | for (std::size_t i = 0; i < s1.size(); ++i) 00199 | { 00200 | if (std::tolower(s1[i]) != std::tolower(s2[i])) 00201 | { 00202 | return false; 00203 | } 00204 | } 00205 | 00206 | return true; 00207 | } 00208 | 00209 | return false; 00210 | } 00211 | 00212 | struct ilesscompare 00213 | { 00214 | inline bool operator() (const std::string& s1, const std::string& s2) const 00215 | { 00216 | const std::size_t length = std::min(s1.size(),s2.size()); 00217 | 00218 | for (std::size_t i = 0; i < length; ++i) 00219 | { 00220 | const char_t c1 = static_cast<char_t>(std::tolower(s1[i])); 00221 | const char_t c2 = static_cast<char_t>(std::tolower(s2[i])); 00222 | 00223 | if (c1 < c2) 00224 | return true; 00225 | else if (c2 < c1) 00226 | return false; 00227 | } 00228 | 00229 | return s1.size() < s2.size(); 00230 | } 00231 | }; 00232 | 00233 | #else 00234 | inline void case_normalise(std::string&) 00235 | {} 00236 | 00237 | inline bool imatch(const char_t c1, const char_t c2) 00238 | { 00239 | return c1 == c2; 00240 | } 00241 | 00242 | inline bool imatch(const std::string& s1, const std::string& s2) 00243 | { 00244 | return s1 == s2; 00245 | } 00246 | 00247 | struct ilesscompare 00248 | { 00249 | inline bool operator() (const std::string& s1, const std::string& s2) const 00250 | { 00251 | return s1 < s2; 00252 | } 00253 | }; 00254 | #endif 00255 | 00256 | inline bool is_valid_sf_symbol(const std::string& symbol) 00257 | { 00258 | // Special function: $f12 or $F34 00259 | return (4 == symbol.size()) && 00260 | ('$' == symbol[0]) && 00261 | imatch('f',symbol[1]) && 00262 | is_digit(symbol[2]) && 00263 | is_digit(symbol[3]); 00264 | } 00265 | 00266 | inline const char_t& front(const std::string& s) 00267 | { 00268 | return s[0]; 00269 | } 00270 | 00271 | inline const char_t& back(const std::string& s) 00272 | { 00273 | return s[s.size() - 1]; 00274 | } 00275 | 00276 | template <typename SignedType> 00277 | inline std::string to_str_impl(SignedType i) 00278 | { 00279 | if (0 == i) 00280 | return std::string("0"); 00281 | 00282 | std::string result; 00283 | 00284 | const int sign = (i < 0) ? -1 : 1; 00285 | 00286 | for ( ; i; i /= 10) 00287 | { 00288 | result += '0' + static_cast<char_t>(sign * (i % 10)); 00289 | } 00290 | 00291 | if (sign < 0) 00292 | { 00293 | result += '-'; 00294 | } 00295 | 00296 | std::reverse(result.begin(), result.end()); 00297 | 00298 | return result; 00299 | } 00300 | 00301 | inline std::string to_str(int i) 00302 | { 00303 | return to_str_impl(i); 00304 | } 00305 | 00306 | inline std::string to_str(std::size_t i) 00307 | { 00308 | return to_str_impl(static_cast<long long int>(i)); 00309 | } 00310 | 00311 | inline bool is_hex_digit(const uchar_t digit) 00312 | { 00313 | return (('0' <= digit) && (digit <= '9')) || 00314 | (('A' <= digit) && (digit <= 'F')) || 00315 | (('a' <= digit) && (digit <= 'f')) ; 00316 | } 00317 | 00318 | inline uchar_t hex_to_bin(uchar_t h) 00319 | { 00320 | if (('0' <= h) && (h <= '9')) 00321 | return (h - '0'); 00322 | else 00323 | return static_cast<uchar_t>(std::toupper(h) - 'A' + 10); 00324 | } 00325 | 00326 | template <typename Iterator> 00327 | inline bool parse_hex(Iterator& itr, Iterator end, 00328 | char_t& result) 00329 | { 00330 | if ( 00331 | (end == (itr )) || 00332 | (end == (itr + 1)) || 00333 | (end == (itr + 2)) || 00334 | (end == (itr + 3)) || 00335 | ('0' != *(itr )) || 00336 | ('X' != std::toupper(*(itr + 1))) || 00337 | (!is_hex_digit(*(itr + 2))) || 00338 | (!is_hex_digit(*(itr + 3))) 00339 | ) 00340 | { 00341 | return false; 00342 | } 00343 | 00344 | result = hex_to_bin(static_cast<uchar_t>(*(itr + 2))) << 4 | 00345 | hex_to_bin(static_cast<uchar_t>(*(itr + 3))) ; 00346 | 00347 | return true; 00348 | } 00349 | 00350 | inline bool cleanup_escapes(std::string& s) 00351 | { 00352 | typedef std::string::iterator str_itr_t; 00353 | 00354 | str_itr_t itr1 = s.begin(); 00355 | str_itr_t itr2 = s.begin(); 00356 | str_itr_t end = s.end (); 00357 | 00358 | std::size_t removal_count = 0; 00359 | 00360 | while (end != itr1) 00361 | { 00362 | if ('\\' == (*itr1)) 00363 | { 00364 | if (end == ++itr1) 00365 | { 00366 | return false; 00367 | } 00368 | else if (parse_hex(itr1, end, *itr2)) 00369 | { 00370 | itr1 += 4; 00371 | itr2 += 1; 00372 | removal_count += 4; 00373 | } 00374 | else if ('a' == (*itr1)) { (*itr2++) = '\a'; ++itr1; ++removal_count; } 00375 | else if ('b' == (*itr1)) { (*itr2++) = '\b'; ++itr1; ++removal_count; } 00376 | else if ('f' == (*itr1)) { (*itr2++) = '\f'; ++itr1; ++removal_count; } 00377 | else if ('n' == (*itr1)) { (*itr2++) = '\n'; ++itr1; ++removal_count; } 00378 | else if ('r' == (*itr1)) { (*itr2++) = '\r'; ++itr1; ++removal_count; } 00379 | else if ('t' == (*itr1)) { (*itr2++) = '\t'; ++itr1; ++removal_count; } 00380 | else if ('v' == (*itr1)) { (*itr2++) = '\v'; ++itr1; ++removal_count; } 00381 | else if ('0' == (*itr1)) { (*itr2++) = '\0'; ++itr1; ++removal_count; } 00382 | else 00383 | { 00384 | (*itr2++) = (*itr1++); 00385 | ++removal_count; 00386 | } 00387 | 00388 | continue; 00389 | } 00390 | else 00391 | (*itr2++) = (*itr1++); 00392 | } 00393 | 00394 | if ((removal_count > s.size()) || (0 == removal_count)) 00395 | return false; 00396 | 00397 | s.resize(s.size() - removal_count); 00398 | 00399 | return true; 00400 | } 00401 | 00402 | class build_string 00403 | { 00404 | public: 00405 | 00406 | explicit build_string(const std::size_t& initial_size = 64) 00407 | { 00408 | data_.reserve(initial_size); 00409 | } 00410 | 00411 | inline build_string& operator << (const std::string& s) 00412 | { 00413 | data_ += s; 00414 | return (*this); 00415 | } 00416 | 00417 | inline build_string& operator << (char_cptr s) 00418 | { 00419 | data_ += std::string(s); 00420 | return (*this); 00421 | } 00422 | 00423 | inline operator std::string () const 00424 | { 00425 | return data_; 00426 | } 00427 | 00428 | inline std::string as_string() const 00429 | { 00430 | return data_; 00431 | } 00432 | 00433 | private: 00434 | 00435 | std::string data_; 00436 | }; 00437 | 00438 | static const std::string reserved_words[] = 00439 | { 00440 | "assert", "break", "case", "continue", "const", "default", 00441 | "false", "for", "if", "else", "ilike", "in", "like", "and", 00442 | "nand", "nor", "not", "null", "or", "repeat", "return", 00443 | "shl", "shr", "swap", "switch", "true", "until", "var", 00444 | "while", "xnor", "xor", "&", "|" 00445 | }; 00446 | 00447 | static const std::size_t reserved_words_size = sizeof(reserved_words) / sizeof(std::string); 00448 | 00449 | static const std::string reserved_symbols[] = 00450 | { 00451 | "abs", "acos", "acosh", "and", "asin", "asinh", "assert", 00452 | "atan", "atanh", "atan2", "avg", "break", "case", "ceil", 00453 | "clamp", "continue", "const", "cos", "cosh", "cot", "csc", 00454 | "default", "deg2grad", "deg2rad", "equal", "erf", "erfc", 00455 | "exp", "expm1", "false", "floor", "for", "frac", "grad2deg", 00456 | "hypot", "iclamp", "if", "else", "ilike", "in", "inrange", 00457 | "like", "log", "log10", "log2", "logn", "log1p", "mand", 00458 | "max", "min", "mod", "mor", "mul", "ncdf", "nand", "nor", 00459 | "not", "not_equal", "null", "or", "pow", "rad2deg", 00460 | "repeat", "return", "root", "round", "roundn", "sec", "sgn", 00461 | "shl", "shr", "sin", "sinc", "sinh", "sqrt", "sum", "swap", 00462 | "switch", "tan", "tanh", "true", "trunc", "until", "var", 00463 | "while", "xnor", "xor", "&", "|" 00464 | }; 00465 | 00466 | static const std::size_t reserved_symbols_size = sizeof(reserved_symbols) / sizeof(std::string); 00467 | 00468 | static const std::string base_function_list[] = 00469 | { 00470 | "abs", "acos", "acosh", "asin", "asinh", "atan", "atanh", 00471 | "atan2", "avg", "ceil", "clamp", "cos", "cosh", "cot", 00472 | "csc", "equal", "erf", "erfc", "exp", "expm1", "floor", 00473 | "frac", "hypot", "iclamp", "like", "log", "log10", "log2", 00474 | "logn", "log1p", "mand", "max", "min", "mod", "mor", "mul", 00475 | "ncdf", "pow", "root", "round", "roundn", "sec", "sgn", 00476 | "sin", "sinc", "sinh", "sqrt", "sum", "swap", "tan", "tanh", 00477 | "trunc", "not_equal", "inrange", "deg2grad", "deg2rad", 00478 | "rad2deg", "grad2deg" 00479 | }; 00480 | 00481 | static const std::size_t base_function_list_size = sizeof(base_function_list) / sizeof(std::string); 00482 | 00483 | static const std::string logic_ops_list[] = 00484 | { 00485 | "and", "nand", "nor", "not", "or", "xnor", "xor", "&", "|" 00486 | }; 00487 | 00488 | static const std::size_t logic_ops_list_size = sizeof(logic_ops_list) / sizeof(std::string); 00489 | 00490 | static const std::string cntrl_struct_list[] = 00491 | { 00492 | "if", "switch", "for", "while", "repeat", "return" 00493 | }; 00494 | 00495 | static const std::size_t cntrl_struct_list_size = sizeof(cntrl_struct_list) / sizeof(std::string); 00496 | 00497 | static const std::string arithmetic_ops_list[] = 00498 | { 00499 | "+", "-", "*", "/", "%", "^" 00500 | }; 00501 | 00502 | static const std::size_t arithmetic_ops_list_size = sizeof(arithmetic_ops_list) / sizeof(std::string); 00503 | 00504 | static const std::string assignment_ops_list[] = 00505 | { 00506 | ":=", "+=", "-=", 00507 | "*=", "/=", "%=" 00508 | }; 00509 | 00510 | static const std::size_t assignment_ops_list_size = sizeof(assignment_ops_list) / sizeof(std::string); 00511 | 00512 | static const std::string inequality_ops_list[] = 00513 | { 00514 | "<", "<=", "==", 00515 | "=", "!=", "<>", 00516 | ">=", ">" 00517 | }; 00518 | 00519 | static const std::size_t inequality_ops_list_size = sizeof(inequality_ops_list) / sizeof(std::string); 00520 | 00521 | inline bool is_reserved_word(const std::string& symbol) 00522 | { 00523 | for (std::size_t i = 0; i < reserved_words_size; ++i) 00524 | { 00525 | if (imatch(symbol, reserved_words[i])) 00526 | { 00527 | return true; 00528 | } 00529 | } 00530 | 00531 | return false; 00532 | } 00533 | 00534 | inline bool is_reserved_symbol(const std::string& symbol) 00535 | { 00536 | for (std::size_t i = 0; i < reserved_symbols_size; ++i) 00537 | { 00538 | if (imatch(symbol, reserved_symbols[i])) 00539 | { 00540 | return true; 00541 | } 00542 | } 00543 | 00544 | return false; 00545 | } 00546 | 00547 | inline bool is_base_function(const std::string& function_name) 00548 | { 00549 | for (std::size_t i = 0; i < base_function_list_size; ++i) 00550 | { 00551 | if (imatch(function_name, base_function_list[i])) 00552 | { 00553 | return true; 00554 | } 00555 | } 00556 | 00557 | return false; 00558 | } 00559 | 00560 | inline bool is_control_struct(const std::string& cntrl_strct) 00561 | { 00562 | for (std::size_t i = 0; i < cntrl_struct_list_size; ++i) 00563 | { 00564 | if (imatch(cntrl_strct, cntrl_struct_list[i])) 00565 | { 00566 | return true; 00567 | } 00568 | } 00569 | 00570 | return false; 00571 | } 00572 | 00573 | inline bool is_logic_opr(const std::string& lgc_opr) 00574 | { 00575 | for (std::size_t i = 0; i < logic_ops_list_size; ++i) 00576 | { 00577 | if (imatch(lgc_opr, logic_ops_list[i])) 00578 | { 00579 | return true; 00580 | } 00581 | } 00582 | 00583 | return false; 00584 | } 00585 | 00586 | struct cs_match 00587 | { 00588 | static inline bool cmp(const char_t c0, const char_t c1) 00589 | { 00590 | return (c0 == c1); 00591 | } 00592 | }; 00593 | 00594 | struct cis_match 00595 | { 00596 | static inline bool cmp(const char_t c0, const char_t c1) 00597 | { 00598 | return (std::tolower(c0) == std::tolower(c1)); 00599 | } 00600 | }; 00601 | 00602 | template <typename Iterator, typename Compare> 00603 | inline bool match_impl(const Iterator pattern_begin, 00604 | const Iterator pattern_end , 00605 | const Iterator data_begin , 00606 | const Iterator data_end , 00607 | const typename std::iterator_traits<Iterator>::value_type& zero_or_more, 00608 | const typename std::iterator_traits<Iterator>::value_type& exactly_one ) 00609 | { 00610 | typedef typename std::iterator_traits<Iterator>::value_type type; 00611 | 00612 | const Iterator null_itr(0); 00613 | 00614 | Iterator p_itr = pattern_begin; 00615 | Iterator d_itr = data_begin; 00616 | Iterator np_itr = null_itr; 00617 | Iterator nd_itr = null_itr; 00618 | 00619 | for ( ; ; ) 00620 | { 00621 | if (p_itr != pattern_end) 00622 | { 00623 | const type c = *(p_itr); 00624 | 00625 | if ((data_end != d_itr) && (Compare::cmp(c,*(d_itr)) || (exactly_one == c))) 00626 | { 00627 | ++d_itr; 00628 | ++p_itr; 00629 | continue; 00630 | } 00631 | else if (zero_or_more == c) 00632 | { 00633 | while ((pattern_end != p_itr) && (zero_or_more == *(p_itr))) 00634 | { 00635 | ++p_itr; 00636 | } 00637 | 00638 | const type d = *(p_itr); 00639 | 00640 | while ((data_end != d_itr) && !(Compare::cmp(d,*(d_itr)) || (exactly_one == d))) 00641 | { 00642 | ++d_itr; 00643 | } 00644 | 00645 | // set backtrack iterators 00646 | np_itr = p_itr - 1; 00647 | nd_itr = d_itr + 1; 00648 | 00649 | continue; 00650 | } 00651 | } 00652 | else if (data_end == d_itr) 00653 | break; 00654 | 00655 | if ((data_end == d_itr) || (null_itr == nd_itr)) 00656 | return false; 00657 | 00658 | p_itr = np_itr; 00659 | d_itr = nd_itr; 00660 | } 00661 | 00662 | return true; 00663 | } 00664 | 00665 | inline bool wc_match(const std::string& wild_card, 00666 | const std::string& str) 00667 | { 00668 | return match_impl<char_cptr,cs_match> 00669 | ( 00670 | wild_card.data(), 00671 | wild_card.data() + wild_card.size(), 00672 | str.data(), 00673 | str.data() + str.size(), 00674 | '*', '?' 00675 | ); 00676 | } 00677 | 00678 | inline bool wc_imatch(const std::string& wild_card, 00679 | const std::string& str) 00680 | { 00681 | return match_impl<char_cptr,cis_match> 00682 | ( 00683 | wild_card.data(), 00684 | wild_card.data() + wild_card.size(), 00685 | str.data(), 00686 | str.data() + str.size(), 00687 | '*', '?' 00688 | ); 00689 | } 00690 | 00691 | inline bool sequence_match(const std::string& pattern, 00692 | const std::string& str, 00693 | std::size_t& diff_index, 00694 | char_t& diff_value) 00695 | { 00696 | if (str.empty()) 00697 | { 00698 | return ("Z" == pattern); 00699 | } 00700 | else if ('*' == pattern[0]) 00701 | return false; 00702 | 00703 | typedef std::string::const_iterator itr_t; 00704 | 00705 | itr_t p_itr = pattern.begin(); 00706 | itr_t s_itr = str .begin(); 00707 | 00708 | const itr_t p_end = pattern.end(); 00709 | const itr_t s_end = str .end(); 00710 | 00711 | while ((s_end != s_itr) && (p_end != p_itr)) 00712 | { 00713 | if ('*' == (*p_itr)) 00714 | { 00715 | const char_t target = static_cast<char_t>(std::toupper(*(p_itr - 1))); 00716 | 00717 | if ('*' == target) 00718 | { 00719 | diff_index = static_cast<std::size_t>(std::distance(str.begin(),s_itr)); 00720 | diff_value = static_cast<char_t>(std::toupper(*p_itr)); 00721 | 00722 | return false; 00723 | } 00724 | else 00725 | ++p_itr; 00726 | 00727 | while (s_itr != s_end) 00728 | { 00729 | if (target != std::toupper(*s_itr)) 00730 | break; 00731 | else 00732 | ++s_itr; 00733 | } 00734 | 00735 | continue; 00736 | } 00737 | else if ( 00738 | ('?' != *p_itr) && 00739 | std::toupper(*p_itr) != std::toupper(*s_itr) 00740 | ) 00741 | { 00742 | diff_index = static_cast<std::size_t>(std::distance(str.begin(),s_itr)); 00743 | diff_value = static_cast<char_t>(std::toupper(*p_itr)); 00744 | 00745 | return false; 00746 | } 00747 | 00748 | ++p_itr; 00749 | ++s_itr; 00750 | } 00751 | 00752 | return ( 00753 | (s_end == s_itr) && 00754 | ( 00755 | (p_end == p_itr) || 00756 | ('*' == *p_itr) 00757 | ) 00758 | ); 00759 | } 00760 | 00761 | template<typename T> 00762 | struct set_zero_value_impl 00763 | { 00764 | static inline void process(T* base_ptr, const std::size_t size) 00765 | { 00766 | const T zero = T(0); 00767 | for (std::size_t i = 0; i < size; ++i) 00768 | { 00769 | base_ptr[i] = zero; 00770 | } 00771 | } 00772 | }; 00773 | 00774 | #define pod_set_zero_value(T) \ 00775 | template <> \ 00776 | struct set_zero_value_impl<T> \ 00777 | { \ 00778 | static inline void process(T* base_ptr, const std::size_t size) \ 00779 | { std::memset(base_ptr, 0x00, size * sizeof(T)); } \ 00780 | }; \ 00781 | 00782 | pod_set_zero_value(float ) 00783 | pod_set_zero_value(double ) 00784 | pod_set_zero_value(long double) 00785 | 00786 | #ifdef pod_set_zero_value 00787 | #undef pod_set_zero_value 00788 | #endif 00789 | 00790 | template<typename T> 00791 | inline void set_zero_value(T* data, const std::size_t size) 00792 | { 00793 | set_zero_value_impl<T>::process(data,size); 00794 | } 00795 | 00796 | template<typename T> 00797 | inline void set_zero_value(std::vector<T>& v) 00798 | { 00799 | set_zero_value(v.data(),v.size()); 00800 | } 00801 | 00802 | static const double pow10[] = 00803 | { 00804 | 1.0, 00805 | 1.0E+001, 1.0E+002, 1.0E+003, 1.0E+004, 00806 | 1.0E+005, 1.0E+006, 1.0E+007, 1.0E+008, 00807 | 1.0E+009, 1.0E+010, 1.0E+011, 1.0E+012, 00808 | 1.0E+013, 1.0E+014, 1.0E+015, 1.0E+016 00809 | }; 00810 | 00811 | static const std::size_t pow10_size = sizeof(pow10) / sizeof(double); 00812 | 00813 | namespace numeric 00814 | { 00815 | namespace constant 00816 | { 00817 | static const double e = 2.71828182845904523536028747135266249775724709369996; 00818 | static const double pi = 3.14159265358979323846264338327950288419716939937510; 00819 | static const double pi_2 = 1.57079632679489661923132169163975144209858469968755; 00820 | static const double pi_4 = 0.78539816339744830961566084581987572104929234984378; 00821 | static const double pi_180 = 0.01745329251994329576923690768488612713442871888542; 00822 | static const double _1_pi = 0.31830988618379067153776752674502872406891929148091; 00823 | static const double _2_pi = 0.63661977236758134307553505349005744813783858296183; 00824 | static const double _180_pi = 57.29577951308232087679815481410517033240547246656443; 00825 | static const double log2 = 0.69314718055994530941723212145817656807550013436026; 00826 | static const double sqrt2 = 1.41421356237309504880168872420969807856967187537695; 00827 | } 00828 | 00829 | namespace details 00830 | { 00831 | struct unknown_type_tag { unknown_type_tag() {} }; 00832 | struct real_type_tag { real_type_tag () {} }; 00833 | struct int_type_tag { int_type_tag () {} }; 00834 | 00835 | template <typename T> 00836 | struct number_type 00837 | { 00838 | typedef unknown_type_tag type; 00839 | number_type() {} 00840 | }; 00841 | 00842 | #define exprtk_register_real_type_tag(T) \ 00843 | template <> struct number_type<T> \ 00844 | { typedef real_type_tag type; number_type() {} }; \ 00845 | 00846 | #define exprtk_register_int_type_tag(T) \ 00847 | template <> struct number_type<T> \ 00848 | { typedef int_type_tag type; number_type() {} }; \ 00849 | 00850 | exprtk_register_real_type_tag(float ) 00851 | exprtk_register_real_type_tag(double ) 00852 | exprtk_register_real_type_tag(long double) 00853 | 00854 | exprtk_register_int_type_tag(short ) 00855 | exprtk_register_int_type_tag(int ) 00856 | exprtk_register_int_type_tag(_int64_t ) 00857 | exprtk_register_int_type_tag(unsigned short) 00858 | exprtk_register_int_type_tag(unsigned int ) 00859 | exprtk_register_int_type_tag(_uint64_t ) 00860 | 00861 | #undef exprtk_register_real_type_tag 00862 | #undef exprtk_register_int_type_tag 00863 | 00864 | template <typename T> 00865 | struct epsilon_type {}; 00866 | 00867 | #define exprtk_define_epsilon_type(Type, Epsilon) \ 00868 | template <> struct epsilon_type<Type> \ 00869 | { \ 00870 | static inline Type value() \ 00871 | { \ 00872 | const Type epsilon = static_cast<Type>(Epsilon); \ 00873 | return epsilon; \ 00874 | } \ 00875 | }; \ 00876 | 00877 | exprtk_define_epsilon_type(float , 0.00000100000f) 00878 | exprtk_define_epsilon_type(double , 0.000000000100) 00879 | exprtk_define_epsilon_type(long double, 0.000000000001) 00880 | 00881 | #undef exprtk_define_epsilon_type 00882 | 00883 | template <typename T> 00884 | inline bool is_nan_impl(const T v, real_type_tag) 00885 | { 00886 | return std::not_equal_to<T>()(v,v); 00887 | } 00888 | 00889 | template <typename T> 00890 | inline int to_int32_impl(const T v, real_type_tag) 00891 | { 00892 | return static_cast<int>(v); 00893 | } 00894 | 00895 | template <typename T> 00896 | inline _int64_t to_int64_impl(const T v, real_type_tag) 00897 | { 00898 | return static_cast<_int64_t>(v); 00899 | } 00900 | 00901 | template <typename T> 00902 | inline _uint64_t to_uint64_impl(const T v, real_type_tag) 00903 | { 00904 | return static_cast<_uint64_t>(v); 00905 | } 00906 | 00907 | template <typename T> 00908 | inline bool is_true_impl(const T v) 00909 | { 00910 | return std::not_equal_to<T>()(T(0),v); 00911 | } 00912 | 00913 | template <typename T> 00914 | inline bool is_false_impl(const T v) 00915 | { 00916 | return std::equal_to<T>()(T(0),v); 00917 | } 00918 | 00919 | template <typename T> 00920 | inline T abs_impl(const T v, real_type_tag) 00921 | { 00922 | return ((v < T(0)) ? -v : v); 00923 | } 00924 | 00925 | template <typename T> 00926 | inline T min_impl(const T v0, const T v1, real_type_tag) 00927 | { 00928 | return std::min<T>(v0,v1); 00929 | } 00930 | 00931 | template <typename T> 00932 | inline T max_impl(const T v0, const T v1, real_type_tag) 00933 | { 00934 | return std::max<T>(v0,v1); 00935 | } 00936 | 00937 | template <typename T> 00938 | inline T equal_impl(const T v0, const T v1, real_type_tag) 00939 | { 00940 | const T epsilon = epsilon_type<T>::value(); 00941 | return (abs_impl(v0 - v1,real_type_tag()) <= (std::max(T(1),std::max(abs_impl(v0,real_type_tag()),abs_impl(v1,real_type_tag()))) * epsilon)) ? T(1) : T(0); 00942 | } 00943 | 00944 | inline float equal_impl(const float v0, const float v1, real_type_tag) 00945 | { 00946 | const float epsilon = epsilon_type<float>::value(); 00947 | return (abs_impl(v0 - v1,real_type_tag()) <= (std::max(1.0f,std::max(abs_impl(v0,real_type_tag()),abs_impl(v1,real_type_tag()))) * epsilon)) ? 1.0f : 0.0f; 00948 | } 00949 | 00950 | template <typename T> 00951 | inline T equal_impl(const T v0, const T v1, int_type_tag) 00952 | { 00953 | return (v0 == v1) ? 1 : 0; 00954 | } 00955 | 00956 | template <typename T> 00957 | inline T expm1_impl(const T v, real_type_tag) 00958 | { 00959 | // return std::expm1<T>(v); 00960 | if (abs_impl(v,real_type_tag()) < T(0.00001)) 00961 | return v + (T(0.5) * v * v); 00962 | else 00963 | return std::exp(v) - T(1); 00964 | } 00965 | 00966 | template <typename T> 00967 | inline T expm1_impl(const T v, int_type_tag) 00968 | { 00969 | return T(std::exp<double>(v)) - T(1); 00970 | } 00971 | 00972 | template <typename T> 00973 | inline T nequal_impl(const T v0, const T v1, real_type_tag) 00974 | { 00975 | typedef real_type_tag rtg; 00976 | const T epsilon = epsilon_type<T>::value(); 00977 | return (abs_impl(v0 - v1,rtg()) > (std::max(T(1),std::max(abs_impl(v0,rtg()),abs_impl(v1,rtg()))) * epsilon)) ? T(1) : T(0); 00978 | } 00979 | 00980 | inline float nequal_impl(const float v0, const float v1, real_type_tag) 00981 | { 00982 | typedef real_type_tag rtg; 00983 | const float epsilon = epsilon_type<float>::value(); 00984 | return (abs_impl(v0 - v1,rtg()) > (std::max(1.0f,std::max(abs_impl(v0,rtg()),abs_impl(v1,rtg()))) * epsilon)) ? 1.0f : 0.0f; 00985 | } 00986 | 00987 | template <typename T> 00988 | inline T nequal_impl(const T v0, const T v1, int_type_tag) 00989 | { 00990 | return (v0 != v1) ? 1 : 0; 00991 | } 00992 | 00993 | template <typename T> 00994 | inline T modulus_impl(const T v0, const T v1, real_type_tag) 00995 | { 00996 | return std::fmod(v0,v1); 00997 | } 00998 | 00999 | template <typename T> 01000 | inline T modulus_impl(const T v0, const T v1, int_type_tag) 01001 | { 01002 | return v0 % v1; 01003 | } 01004 | 01005 | template <typename T> 01006 | inline T pow_impl(const T v0, const T v1, real_type_tag) 01007 | { 01008 | return std::pow(v0,v1); 01009 | } 01010 | 01011 | template <typename T> 01012 | inline T pow_impl(const T v0, const T v1, int_type_tag) 01013 | { 01014 | return std::pow(static_cast<double>(v0),static_cast<double>(v1)); 01015 | } 01016 | 01017 | template <typename T> 01018 | inline T logn_impl(const T v0, const T v1, real_type_tag) 01019 | { 01020 | return std::log(v0) / std::log(v1); 01021 | } 01022 | 01023 | template <typename T> 01024 | inline T logn_impl(const T v0, const T v1, int_type_tag) 01025 | { 01026 | return static_cast<T>(logn_impl<double>(static_cast<double>(v0),static_cast<double>(v1),real_type_tag())); 01027 | } 01028 | 01029 | template <typename T> 01030 | inline T log1p_impl(const T v, real_type_tag) 01031 | { 01032 | if (v > T(-1)) 01033 | { 01034 | if (abs_impl(v,real_type_tag()) > T(0.0001)) 01035 | { 01036 | return std::log(T(1) + v); 01037 | } 01038 | else 01039 | return (T(-0.5) * v + T(1)) * v; 01040 | } 01041 | 01042 | return std::numeric_limits<T>::quiet_NaN(); 01043 | } 01044 | 01045 | template <typename T> 01046 | inline T log1p_impl(const T v, int_type_tag) 01047 | { 01048 | if (v > T(-1)) 01049 | { 01050 | return std::log(T(1) + v); 01051 | } 01052 | 01053 | return std::numeric_limits<T>::quiet_NaN(); 01054 | } 01055 | 01056 | template <typename T> 01057 | inline T root_impl(const T v0, const T v1, real_type_tag) 01058 | { 01059 | if (v0 < T(0)) 01060 | { 01061 | return (v1 == trunc_impl(v1, real_type_tag())) && 01062 | (modulus_impl(v1, T(2), real_type_tag()) != T(0)) ? 01063 | -std::pow(abs_impl(v0, real_type_tag()), T(1) / v1) : 01064 | std::numeric_limits<double>::quiet_NaN(); 01065 | } 01066 | 01067 | return std::pow(v0, T(1) / v1); 01068 | } 01069 | 01070 | template <typename T> 01071 | inline T root_impl(const T v0, const T v1, int_type_tag) 01072 | { 01073 | return root_impl<double>(static_cast<double>(v0),static_cast<double>(v1),real_type_tag()); 01074 | } 01075 | 01076 | template <typename T> 01077 | inline T round_impl(const T v, real_type_tag) 01078 | { 01079 | return ((v < T(0)) ? std::ceil(v - T(0.5)) : std::floor(v + T(0.5))); 01080 | } 01081 | 01082 | template <typename T> 01083 | inline T roundn_impl(const T v0, const T v1, real_type_tag) 01084 | { 01085 | const int index = std::max<int>(0, std::min<int>(pow10_size - 1, static_cast<int>(std::floor(v1)))); 01086 | const T p10 = T(pow10[index]); 01087 | 01088 | if (v0 < T(0)) 01089 | return T(std::ceil ((v0 * p10) - T(0.5)) / p10); 01090 | else 01091 | return T(std::floor((v0 * p10) + T(0.5)) / p10); 01092 | } 01093 | 01094 | template <typename T> 01095 | inline T roundn_impl(const T v0, const T, int_type_tag) 01096 | { 01097 | return v0; 01098 | } 01099 | 01100 | template <typename T> 01101 | inline T hypot_impl(const T v0, const T v1, real_type_tag) 01102 | { 01103 | return std::sqrt((v0 * v0) + (v1 * v1)); 01104 | } 01105 | 01106 | template <typename T> 01107 | inline T hypot_impl(const T v0, const T v1, int_type_tag) 01108 | { 01109 | return static_cast<T>(std::sqrt(static_cast<double>((v0 * v0) + (v1 * v1)))); 01110 | } 01111 | 01112 | template <typename T> 01113 | inline T atan2_impl(const T v0, const T v1, real_type_tag) 01114 | { 01115 | return std::atan2(v0,v1); 01116 | } 01117 | 01118 | template <typename T> 01119 | inline T atan2_impl(const T, const T, int_type_tag) 01120 | { 01121 | return 0; 01122 | } 01123 | 01124 | template <typename T> 01125 | inline T shr_impl(const T v0, const T v1, real_type_tag) 01126 | { 01127 | return v0 * (T(1) / std::pow(T(2),static_cast<T>(static_cast<int>(v1)))); 01128 | } 01129 | 01130 | template <typename T> 01131 | inline T shr_impl(const T v0, const T v1, int_type_tag) 01132 | { 01133 | return v0 >> v1; 01134 | } 01135 | 01136 | template <typename T> 01137 | inline T shl_impl(const T v0, const T v1, real_type_tag) 01138 | { 01139 | return v0 * std::pow(T(2),static_cast<T>(static_cast<int>(v1))); 01140 | } 01141 | 01142 | template <typename T> 01143 | inline T shl_impl(const T v0, const T v1, int_type_tag) 01144 | { 01145 | return v0 << v1; 01146 | } 01147 | 01148 | template <typename T> 01149 | inline T sgn_impl(const T v, real_type_tag) 01150 | { 01151 | if (v > T(0)) return T(+1); 01152 | else if (v < T(0)) return T(-1); 01153 | else return T( 0); 01154 | } 01155 | 01156 | template <typename T> 01157 | inline T sgn_impl(const T v, int_type_tag) 01158 | { 01159 | if (v > T(0)) return T(+1); 01160 | else if (v < T(0)) return T(-1); 01161 | else return T( 0); 01162 | } 01163 | 01164 | template <typename T> 01165 | inline T and_impl(const T v0, const T v1, real_type_tag) 01166 | { 01167 | return (is_true_impl(v0) && is_true_impl(v1)) ? T(1) : T(0); 01168 | } 01169 | 01170 | template <typename T> 01171 | inline T and_impl(const T v0, const T v1, int_type_tag) 01172 | { 01173 | return v0 && v1; 01174 | } 01175 | 01176 | template <typename T> 01177 | inline T nand_impl(const T v0, const T v1, real_type_tag) 01178 | { 01179 | return (is_false_impl(v0) || is_false_impl(v1)) ? T(1) : T(0); 01180 | } 01181 | 01182 | template <typename T> 01183 | inline T nand_impl(const T v0, const T v1, int_type_tag) 01184 | { 01185 | return !(v0 && v1); 01186 | } 01187 | 01188 | template <typename T> 01189 | inline T or_impl(const T v0, const T v1, real_type_tag) 01190 | { 01191 | return (is_true_impl(v0) || is_true_impl(v1)) ? T(1) : T(0); 01192 | } 01193 | 01194 | template <typename T> 01195 | inline T or_impl(const T v0, const T v1, int_type_tag) 01196 | { 01197 | return (v0 || v1); 01198 | } 01199 | 01200 | template <typename T> 01201 | inline T nor_impl(const T v0, const T v1, real_type_tag) 01202 | { 01203 | return (is_false_impl(v0) && is_false_impl(v1)) ? T(1) : T(0); 01204 | } 01205 | 01206 | template <typename T> 01207 | inline T nor_impl(const T v0, const T v1, int_type_tag) 01208 | { 01209 | return !(v0 || v1); 01210 | } 01211 | 01212 | template <typename T> 01213 | inline T xor_impl(const T v0, const T v1, real_type_tag) 01214 | { 01215 | return (is_false_impl(v0) != is_false_impl(v1)) ? T(1) : T(0); 01216 | } 01217 | 01218 | template <typename T> 01219 | inline T xor_impl(const T v0, const T v1, int_type_tag) 01220 | { 01221 | return v0 ^ v1; 01222 | } 01223 | 01224 | template <typename T> 01225 | inline T xnor_impl(const T v0, const T v1, real_type_tag) 01226 | { 01227 | const bool v0_true = is_true_impl(v0); 01228 | const bool v1_true = is_true_impl(v1); 01229 | 01230 | if ((v0_true && v1_true) || (!v0_true && !v1_true)) 01231 | return T(1); 01232 | else 01233 | return T(0); 01234 | } 01235 | 01236 | template <typename T> 01237 | inline T xnor_impl(const T v0, const T v1, int_type_tag) 01238 | { 01239 | const bool v0_true = is_true_impl(v0); 01240 | const bool v1_true = is_true_impl(v1); 01241 | 01242 | if ((v0_true && v1_true) || (!v0_true && !v1_true)) 01243 | return T(1); 01244 | else 01245 | return T(0); 01246 | } 01247 | 01248 | #if (defined(_MSC_VER) && (_MSC_VER >= 1900)) || !defined(_MSC_VER) 01249 | #define exprtk_define_erf(TT, impl) \ 01250 | inline TT erf_impl(const TT v) { return impl(v); } \ 01251 | 01252 | exprtk_define_erf(float , ::erff) 01253 | exprtk_define_erf(double , ::erf ) 01254 | exprtk_define_erf(long double, ::erfl) 01255 | #undef exprtk_define_erf 01256 | #endif 01257 | 01258 | template <typename T> 01259 | inline T erf_impl(const T v, real_type_tag) 01260 | { 01261 | #if defined(_MSC_VER) && (_MSC_VER < 1900) 01262 | // Credits: Abramowitz & Stegun Equations 7.1.25-28 01263 | static const T c[] = 01264 | { 01265 | T( 1.26551223), T(1.00002368), 01266 | T( 0.37409196), T(0.09678418), 01267 | T(-0.18628806), T(0.27886807), 01268 | T(-1.13520398), T(1.48851587), 01269 | T(-0.82215223), T(0.17087277) 01270 | }; 01271 | 01272 | const T t = T(1) / (T(1) + T(0.5) * abs_impl(v,real_type_tag())); 01273 | 01274 | const T result = T(1) - t * std::exp((-v * v) - 01275 | c[0] + t * (c[1] + t * 01276 | (c[2] + t * (c[3] + t * 01277 | (c[4] + t * (c[5] + t * 01278 | (c[6] + t * (c[7] + t * 01279 | (c[8] + t * (c[9])))))))))); 01280 | 01281 | return (v >= T(0)) ? result : -result; 01282 | #else 01283 | return erf_impl(v); 01284 | #endif 01285 | } 01286 | 01287 | template <typename T> 01288 | inline T erf_impl(const T v, int_type_tag) 01289 | { 01290 | return erf_impl(static_cast<double>(v),real_type_tag()); 01291 | } 01292 | 01293 | #if (defined(_MSC_VER) && (_MSC_VER >= 1900)) || !defined(_MSC_VER) 01294 | #define exprtk_define_erfc(TT, impl) \ 01295 | inline TT erfc_impl(const TT v) { return impl(v); } \ 01296 | 01297 | exprtk_define_erfc(float ,::erfcf) 01298 | exprtk_define_erfc(double ,::erfc ) 01299 | exprtk_define_erfc(long double,::erfcl) 01300 | #undef exprtk_define_erfc 01301 | #endif 01302 | 01303 | template <typename T> 01304 | inline T erfc_impl(const T v, real_type_tag) 01305 | { 01306 | #if defined(_MSC_VER) && (_MSC_VER < 1900) 01307 | return T(1) - erf_impl(v,real_type_tag()); 01308 | #else 01309 | return erfc_impl(v); 01310 | #endif 01311 | } 01312 | 01313 | template <typename T> 01314 | inline T erfc_impl(const T v, int_type_tag) 01315 | { 01316 | return erfc_impl(static_cast<double>(v),real_type_tag()); 01317 | } 01318 | 01319 | template <typename T> 01320 | inline T ncdf_impl(const T v, real_type_tag) 01321 | { 01322 | return T(0.5) * erfc_impl(-(v / T(numeric::constant::sqrt2)),real_type_tag()); 01323 | } 01324 | 01325 | template <typename T> 01326 | inline T ncdf_impl(const T v, int_type_tag) 01327 | { 01328 | return ncdf_impl(static_cast<double>(v),real_type_tag()); 01329 | } 01330 | 01331 | template <typename T> 01332 | inline T sinc_impl(const T v, real_type_tag) 01333 | { 01334 | if (std::abs(v) >= std::numeric_limits<T>::epsilon()) 01335 | return(std::sin(v) / v); 01336 | else 01337 | return T(1); 01338 | } 01339 | 01340 | template <typename T> 01341 | inline T sinc_impl(const T v, int_type_tag) 01342 | { 01343 | return sinc_impl(static_cast<double>(v),real_type_tag()); 01344 | } 01345 | 01346 | #if __cplusplus >= 201103L 01347 | template <typename T> 01348 | inline T acosh_impl(const T v, real_type_tag) 01349 | { 01350 | return std::acosh(v); 01351 | } 01352 | 01353 | template <typename T> 01354 | inline T asinh_impl(const T v, real_type_tag) 01355 | { 01356 | return std::asinh(v); 01357 | } 01358 | 01359 | template <typename T> 01360 | inline T atanh_impl(const T v, real_type_tag) 01361 | { 01362 | return std::atanh(v); 01363 | } 01364 | 01365 | template <typename T> 01366 | inline T trunc_impl(const T v, real_type_tag) 01367 | { 01368 | return std::trunc(v); 01369 | } 01370 | #else 01371 | template <typename T> 01372 | inline T acosh_impl(const T v, real_type_tag) 01373 | { 01374 | return std::log(v + std::sqrt((v * v) - T(1))); 01375 | } 01376 | 01377 | template <typename T> 01378 | inline T asinh_impl(const T v, real_type_tag) 01379 | { 01380 | return std::log(v + std::sqrt((v * v) + T(1))); 01381 | } 01382 | 01383 | template <typename T> 01384 | inline T atanh_impl(const T v, real_type_tag) 01385 | { 01386 | return (std::log(T(1) + v) - std::log(T(1) - v)) / T(2); 01387 | } 01388 | 01389 | template <typename T> 01390 | inline T trunc_impl(const T v, real_type_tag) 01391 | { 01392 | return T(static_cast<long long>(v)); 01393 | } 01394 | #endif 01395 | 01396 | template <typename T> inline T acos_impl(const T v, real_type_tag) { return std::acos (v); } 01397 | template <typename T> inline T asin_impl(const T v, real_type_tag) { return std::asin (v); } 01398 | template <typename T> inline T atan_impl(const T v, real_type_tag) { return std::atan (v); } 01399 | template <typename T> inline T ceil_impl(const T v, real_type_tag) { return std::ceil (v); } 01400 | template <typename T> inline T cos_impl(const T v, real_type_tag) { return std::cos (v); } 01401 | template <typename T> inline T cosh_impl(const T v, real_type_tag) { return std::cosh (v); } 01402 | template <typename T> inline T exp_impl(const T v, real_type_tag) { return std::exp (v); } 01403 | template <typename T> inline T floor_impl(const T v, real_type_tag) { return std::floor(v); } 01404 | template <typename T> inline T log_impl(const T v, real_type_tag) { return std::log (v); } 01405 | template <typename T> inline T log10_impl(const T v, real_type_tag) { return std::log10(v); } 01406 | template <typename T> inline T log2_impl(const T v, real_type_tag) { return std::log(v)/T(numeric::constant::log2); } 01407 | template <typename T> inline T neg_impl(const T v, real_type_tag) { return -v; } 01408 | template <typename T> inline T pos_impl(const T v, real_type_tag) { return +v; } 01409 | template <typename T> inline T sin_impl(const T v, real_type_tag) { return std::sin (v); } 01410 | template <typename T> inline T sinh_impl(const T v, real_type_tag) { return std::sinh (v); } 01411 | template <typename T> inline T sqrt_impl(const T v, real_type_tag) { return std::sqrt (v); } 01412 | template <typename T> inline T tan_impl(const T v, real_type_tag) { return std::tan (v); } 01413 | template <typename T> inline T tanh_impl(const T v, real_type_tag) { return std::tanh (v); } 01414 | template <typename T> inline T cot_impl(const T v, real_type_tag) { return T(1) / std::tan(v); } 01415 | template <typename T> inline T sec_impl(const T v, real_type_tag) { return T(1) / std::cos(v); } 01416 | template <typename T> inline T csc_impl(const T v, real_type_tag) { return T(1) / std::sin(v); } 01417 | template <typename T> inline T r2d_impl(const T v, real_type_tag) { return (v * T(numeric::constant::_180_pi)); } 01418 | template <typename T> inline T d2r_impl(const T v, real_type_tag) { return (v * T(numeric::constant::pi_180)); } 01419 | template <typename T> inline T d2g_impl(const T v, real_type_tag) { return (v * T(10.0/9.0)); } 01420 | template <typename T> inline T g2d_impl(const T v, real_type_tag) { return (v * T(9.0/10.0)); } 01421 | template <typename T> inline T notl_impl(const T v, real_type_tag) { return (std::not_equal_to<T>()(T(0),v) ? T(0) : T(1)); } 01422 | template <typename T> inline T frac_impl(const T v, real_type_tag) { return (v - trunc_impl(v,real_type_tag())); } 01423 | 01424 | template <typename T> inline T const_pi_impl(real_type_tag) { return T(numeric::constant::pi); } 01425 | template <typename T> inline T const_e_impl(real_type_tag) { return T(numeric::constant::e); } 01426 | template <typename T> inline T const_qnan_impl(real_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01427 | 01428 | template <typename T> inline T abs_impl(const T v, int_type_tag) { return ((v >= T(0)) ? v : -v); } 01429 | template <typename T> inline T exp_impl(const T v, int_type_tag) { return std::exp (v); } 01430 | template <typename T> inline T log_impl(const T v, int_type_tag) { return std::log (v); } 01431 | template <typename T> inline T log10_impl(const T v, int_type_tag) { return std::log10(v); } 01432 | template <typename T> inline T log2_impl(const T v, int_type_tag) { return std::log(v)/T(numeric::constant::log2); } 01433 | template <typename T> inline T neg_impl(const T v, int_type_tag) { return -v; } 01434 | template <typename T> inline T pos_impl(const T v, int_type_tag) { return +v; } 01435 | template <typename T> inline T ceil_impl(const T v, int_type_tag) { return v; } 01436 | template <typename T> inline T floor_impl(const T v, int_type_tag) { return v; } 01437 | template <typename T> inline T round_impl(const T v, int_type_tag) { return v; } 01438 | template <typename T> inline T notl_impl(const T v, int_type_tag) { return !v; } 01439 | template <typename T> inline T sqrt_impl(const T v, int_type_tag) { return std::sqrt (v); } 01440 | template <typename T> inline T frac_impl(const T , int_type_tag) { return T(0); } 01441 | template <typename T> inline T trunc_impl(const T v, int_type_tag) { return v; } 01442 | template <typename T> inline T acos_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01443 | template <typename T> inline T acosh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01444 | template <typename T> inline T asin_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01445 | template <typename T> inline T asinh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01446 | template <typename T> inline T atan_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01447 | template <typename T> inline T atanh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01448 | template <typename T> inline T cos_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01449 | template <typename T> inline T cosh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01450 | template <typename T> inline T sin_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01451 | template <typename T> inline T sinh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01452 | template <typename T> inline T tan_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01453 | template <typename T> inline T tanh_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01454 | template <typename T> inline T cot_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01455 | template <typename T> inline T sec_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01456 | template <typename T> inline T csc_impl(const T , int_type_tag) { return std::numeric_limits<T>::quiet_NaN(); } 01457 | 01458 | template <typename T> 01459 | inline bool is_integer_impl(const T& v, real_type_tag) 01460 | { 01461 | return std::equal_to<T>()(T(0),std::fmod(v,T(1))); 01462 | } 01463 | 01464 | template <typename T> 01465 | inline bool is_integer_impl(const T&, int_type_tag) 01466 | { 01467 | return true; 01468 | } 01469 | } 01470 | 01471 | template <typename Type> 01472 | struct numeric_info { enum { length = 0, size = 32, bound_length = 0, min_exp = 0, max_exp = 0 }; }; 01473 | 01474 | template <> struct numeric_info<int > { enum { length = 10, size = 16, bound_length = 9 }; }; 01475 | template <> struct numeric_info<float > { enum { min_exp = -38, max_exp = +38 }; }; 01476 | template <> struct numeric_info<double > { enum { min_exp = -308, max_exp = +308 }; }; 01477 | template <> struct numeric_info<long double> { enum { min_exp = -308, max_exp = +308 }; }; 01478 | 01479 | template <typename T> 01480 | inline int to_int32(const T v) 01481 | { 01482 | const typename details::number_type<T>::type num_type; 01483 | return to_int32_impl(v, num_type); 01484 | } 01485 | 01486 | template <typename T> 01487 | inline _int64_t to_int64(const T v) 01488 | { 01489 | const typename details::number_type<T>::type num_type; 01490 | return to_int64_impl(v, num_type); 01491 | } 01492 | 01493 | template <typename T> 01494 | inline _uint64_t to_uint64(const T v) 01495 | { 01496 | const typename details::number_type<T>::type num_type; 01497 | return to_uint64_impl(v, num_type); 01498 | } 01499 | 01500 | template <typename T> 01501 | inline bool is_nan(const T v) 01502 | { 01503 | const typename details::number_type<T>::type num_type; 01504 | return is_nan_impl(v, num_type); 01505 | } 01506 | 01507 | template <typename T> 01508 | inline T min(const T v0, const T v1) 01509 | { 01510 | const typename details::number_type<T>::type num_type; 01511 | return min_impl(v0, v1, num_type); 01512 | } 01513 | 01514 | template <typename T> 01515 | inline T max(const T v0, const T v1) 01516 | { 01517 | const typename details::number_type<T>::type num_type; 01518 | return max_impl(v0, v1, num_type); 01519 | } 01520 | 01521 | template <typename T> 01522 | inline T equal(const T v0, const T v1) 01523 | { 01524 | const typename details::number_type<T>::type num_type; 01525 | return equal_impl(v0, v1, num_type); 01526 | } 01527 | 01528 | template <typename T> 01529 | inline T nequal(const T v0, const T v1) 01530 | { 01531 | const typename details::number_type<T>::type num_type; 01532 | return nequal_impl(v0, v1, num_type); 01533 | } 01534 | 01535 | template <typename T> 01536 | inline T modulus(const T v0, const T v1) 01537 | { 01538 | const typename details::number_type<T>::type num_type; 01539 | return modulus_impl(v0, v1, num_type); 01540 | } 01541 | 01542 | template <typename T> 01543 | inline T pow(const T v0, const T v1) 01544 | { 01545 | const typename details::number_type<T>::type num_type; 01546 | return pow_impl(v0, v1, num_type); 01547 | } 01548 | 01549 | template <typename T> 01550 | inline T logn(const T v0, const T v1) 01551 | { 01552 | const typename details::number_type<T>::type num_type; 01553 | return logn_impl(v0, v1, num_type); 01554 | } 01555 | 01556 | template <typename T> 01557 | inline T root(const T v0, const T v1) 01558 | { 01559 | const typename details::number_type<T>::type num_type; 01560 | return root_impl(v0, v1, num_type); 01561 | } 01562 | 01563 | template <typename T> 01564 | inline T roundn(const T v0, const T v1) 01565 | { 01566 | const typename details::number_type<T>::type num_type; 01567 | return roundn_impl(v0, v1, num_type); 01568 | } 01569 | 01570 | template <typename T> 01571 | inline T hypot(const T v0, const T v1) 01572 | { 01573 | const typename details::number_type<T>::type num_type; 01574 | return hypot_impl(v0, v1, num_type); 01575 | } 01576 | 01577 | template <typename T> 01578 | inline T atan2(const T v0, const T v1) 01579 | { 01580 | const typename details::number_type<T>::type num_type; 01581 | return atan2_impl(v0, v1, num_type); 01582 | } 01583 | 01584 | template <typename T> 01585 | inline T shr(const T v0, const T v1) 01586 | { 01587 | const typename details::number_type<T>::type num_type; 01588 | return shr_impl(v0, v1, num_type); 01589 | } 01590 | 01591 | template <typename T> 01592 | inline T shl(const T v0, const T v1) 01593 | { 01594 | const typename details::number_type<T>::type num_type; 01595 | return shl_impl(v0, v1, num_type); 01596 | } 01597 | 01598 | template <typename T> 01599 | inline T and_opr(const T v0, const T v1) 01600 | { 01601 | const typename details::number_type<T>::type num_type; 01602 | return and_impl(v0, v1, num_type); 01603 | } 01604 | 01605 | template <typename T> 01606 | inline T nand_opr(const T v0, const T v1) 01607 | { 01608 | const typename details::number_type<T>::type num_type; 01609 | return nand_impl(v0, v1, num_type); 01610 | } 01611 | 01612 | template <typename T> 01613 | inline T or_opr(const T v0, const T v1) 01614 | { 01615 | const typename details::number_type<T>::type num_type; 01616 | return or_impl(v0, v1, num_type); 01617 | } 01618 | 01619 | template <typename T> 01620 | inline T nor_opr(const T v0, const T v1) 01621 | { 01622 | const typename details::number_type<T>::type num_type; 01623 | return nor_impl(v0, v1, num_type); 01624 | } 01625 | 01626 | template <typename T> 01627 | inline T xor_opr(const T v0, const T v1) 01628 | { 01629 | const typename details::number_type<T>::type num_type; 01630 | return xor_impl(v0, v1, num_type); 01631 | } 01632 | 01633 | template <typename T> 01634 | inline T xnor_opr(const T v0, const T v1) 01635 | { 01636 | const typename details::number_type<T>::type num_type; 01637 | return xnor_impl(v0, v1, num_type); 01638 | } 01639 | 01640 | template <typename T> 01641 | inline bool is_integer(const T v) 01642 | { 01643 | const typename details::number_type<T>::type num_type; 01644 | return is_integer_impl(v, num_type); 01645 | } 01646 | 01647 | template <typename T, unsigned int N> 01648 | struct fast_exp 01649 | { 01650 | static inline T result(T v) 01651 | { 01652 | unsigned int k = N; 01653 | T l = T(1); 01654 | 01655 | while (k) 01656 | { 01657 | if (1 == (k % 2)) 01658 | { 01659 | l *= v; 01660 | --k; 01661 | } 01662 | 01663 | v *= v; 01664 | k /= 2; 01665 | } 01666 | 01667 | return l; 01668 | } 01669 | }; 01670 | 01671 | template <typename T> struct fast_exp<T,10> { static inline T result(const T v) { T v_5 = fast_exp<T,5>::result(v); return v_5 * v_5; } }; 01672 | template <typename T> struct fast_exp<T, 9> { static inline T result(const T v) { return fast_exp<T,8>::result(v) * v; } }; 01673 | template <typename T> struct fast_exp<T, 8> { static inline T result(const T v) { T v_4 = fast_exp<T,4>::result(v); return v_4 * v_4; } }; 01674 | template <typename T> struct fast_exp<T, 7> { static inline T result(const T v) { return fast_exp<T,6>::result(v) * v; } }; 01675 | template <typename T> struct fast_exp<T, 6> { static inline T result(const T v) { T v_3 = fast_exp<T,3>::result(v); return v_3 * v_3; } }; 01676 | template <typename T> struct fast_exp<T, 5> { static inline T result(const T v) { return fast_exp<T,4>::result(v) * v; } }; 01677 | template <typename T> struct fast_exp<T, 4> { static inline T result(const T v) { T v_2 = v * v; return v_2 * v_2; } }; 01678 | template <typename T> struct fast_exp<T, 3> { static inline T result(const T v) { return v * v * v; } }; 01679 | template <typename T> struct fast_exp<T, 2> { static inline T result(const T v) { return v * v; } }; 01680 | template <typename T> struct fast_exp<T, 1> { static inline T result(const T v) { return v; } }; 01681 | template <typename T> struct fast_exp<T, 0> { static inline T result(const T ) { return T(1); } }; 01682 | 01683 | #define exprtk_define_unary_function(FunctionName) \ 01684 | template <typename T> \ 01685 | inline T FunctionName (const T v) \ 01686 | { \ 01687 | const typename details::number_type<T>::type num_type; \ 01688 | return FunctionName##_impl(v,num_type); \ 01689 | } \ 01690 | 01691 | exprtk_define_unary_function(abs ) 01692 | exprtk_define_unary_function(acos ) 01693 | exprtk_define_unary_function(acosh) 01694 | exprtk_define_unary_function(asin ) 01695 | exprtk_define_unary_function(asinh) 01696 | exprtk_define_unary_function(atan ) 01697 | exprtk_define_unary_function(atanh) 01698 | exprtk_define_unary_function(ceil ) 01699 | exprtk_define_unary_function(cos ) 01700 | exprtk_define_unary_function(cosh ) 01701 | exprtk_define_unary_function(exp ) 01702 | exprtk_define_unary_function(expm1) 01703 | exprtk_define_unary_function(floor) 01704 | exprtk_define_unary_function(log ) 01705 | exprtk_define_unary_function(log10) 01706 | exprtk_define_unary_function(log2 ) 01707 | exprtk_define_unary_function(log1p) 01708 | exprtk_define_unary_function(neg ) 01709 | exprtk_define_unary_function(pos ) 01710 | exprtk_define_unary_function(round) 01711 | exprtk_define_unary_function(sin ) 01712 | exprtk_define_unary_function(sinc ) 01713 | exprtk_define_unary_function(sinh ) 01714 | exprtk_define_unary_function(sqrt ) 01715 | exprtk_define_unary_function(tan ) 01716 | exprtk_define_unary_function(tanh ) 01717 | exprtk_define_unary_function(cot ) 01718 | exprtk_define_unary_function(sec ) 01719 | exprtk_define_unary_function(csc ) 01720 | exprtk_define_unary_function(r2d ) 01721 | exprtk_define_unary_function(d2r ) 01722 | exprtk_define_unary_function(d2g ) 01723 | exprtk_define_unary_function(g2d ) 01724 | exprtk_define_unary_function(notl ) 01725 | exprtk_define_unary_function(sgn ) 01726 | exprtk_define_unary_function(erf ) 01727 | exprtk_define_unary_function(erfc ) 01728 | exprtk_define_unary_function(ncdf ) 01729 | exprtk_define_unary_function(frac ) 01730 | exprtk_define_unary_function(trunc) 01731 | #undef exprtk_define_unary_function 01732 | } 01733 | 01734 | template <typename T> 01735 | inline T compute_pow10(T d, const int exponent) 01736 | { 01737 | static const double fract10[] = 01738 | { 01739 | 0.0, 01740 | 1.0E+001, 1.0E+002, 1.0E+003, 1.0E+004, 1.0E+005, 1.0E+006, 1.0E+007, 1.0E+008, 1.0E+009, 1.0E+010, 01741 | 1.0E+011, 1.0E+012, 1.0E+013, 1.0E+014, 1.0E+015, 1.0E+016, 1.0E+017, 1.0E+018, 1.0E+019, 1.0E+020, 01742 | 1.0E+021, 1.0E+022, 1.0E+023, 1.0E+024, 1.0E+025, 1.0E+026, 1.0E+027, 1.0E+028, 1.0E+029, 1.0E+030, 01743 | 1.0E+031, 1.0E+032, 1.0E+033, 1.0E+034, 1.0E+035, 1.0E+036, 1.0E+037, 1.0E+038, 1.0E+039, 1.0E+040, 01744 | 1.0E+041, 1.0E+042, 1.0E+043, 1.0E+044, 1.0E+045, 1.0E+046, 1.0E+047, 1.0E+048, 1.0E+049, 1.0E+050, 01745 | 1.0E+051, 1.0E+052, 1.0E+053, 1.0E+054, 1.0E+055, 1.0E+056, 1.0E+057, 1.0E+058, 1.0E+059, 1.0E+060, 01746 | 1.0E+061, 1.0E+062, 1.0E+063, 1.0E+064, 1.0E+065, 1.0E+066, 1.0E+067, 1.0E+068, 1.0E+069, 1.0E+070, 01747 | 1.0E+071, 1.0E+072, 1.0E+073, 1.0E+074, 1.0E+075, 1.0E+076, 1.0E+077, 1.0E+078, 1.0E+079, 1.0E+080, 01748 | 1.0E+081, 1.0E+082, 1.0E+083, 1.0E+084, 1.0E+085, 1.0E+086, 1.0E+087, 1.0E+088, 1.0E+089, 1.0E+090, 01749 | 1.0E+091, 1.0E+092, 1.0E+093, 1.0E+094, 1.0E+095, 1.0E+096, 1.0E+097, 1.0E+098, 1.0E+099, 1.0E+100, 01750 | 1.0E+101, 1.0E+102, 1.0E+103, 1.0E+104, 1.0E+105, 1.0E+106, 1.0E+107, 1.0E+108, 1.0E+109, 1.0E+110, 01751 | 1.0E+111, 1.0E+112, 1.0E+113, 1.0E+114, 1.0E+115, 1.0E+116, 1.0E+117, 1.0E+118, 1.0E+119, 1.0E+120, 01752 | 1.0E+121, 1.0E+122, 1.0E+123, 1.0E+124, 1.0E+125, 1.0E+126, 1.0E+127, 1.0E+128, 1.0E+129, 1.0E+130, 01753 | 1.0E+131, 1.0E+132, 1.0E+133, 1.0E+134, 1.0E+135, 1.0E+136, 1.0E+137, 1.0E+138, 1.0E+139, 1.0E+140, 01754 | 1.0E+141, 1.0E+142, 1.0E+143, 1.0E+144, 1.0E+145, 1.0E+146, 1.0E+147, 1.0E+148, 1.0E+149, 1.0E+150, 01755 | 1.0E+151, 1.0E+152, 1.0E+153, 1.0E+154, 1.0E+155, 1.0E+156, 1.0E+157, 1.0E+158, 1.0E+159, 1.0E+160, 01756 | 1.0E+161, 1.0E+162, 1.0E+163, 1.0E+164, 1.0E+165, 1.0E+166, 1.0E+167, 1.0E+168, 1.0E+169, 1.0E+170, 01757 | 1.0E+171, 1.0E+172, 1.0E+173, 1.0E+174, 1.0E+175, 1.0E+176, 1.0E+177, 1.0E+178, 1.0E+179, 1.0E+180, 01758 | 1.0E+181, 1.0E+182, 1.0E+183, 1.0E+184, 1.0E+185, 1.0E+186, 1.0E+187, 1.0E+188, 1.0E+189, 1.0E+190, 01759 | 1.0E+191, 1.0E+192, 1.0E+193, 1.0E+194, 1.0E+195, 1.0E+196, 1.0E+197, 1.0E+198, 1.0E+199, 1.0E+200, 01760 | 1.0E+201, 1.0E+202, 1.0E+203, 1.0E+204, 1.0E+205, 1.0E+206, 1.0E+207, 1.0E+208, 1.0E+209, 1.0E+210, 01761 | 1.0E+211, 1.0E+212, 1.0E+213, 1.0E+214, 1.0E+215, 1.0E+216, 1.0E+217, 1.0E+218, 1.0E+219, 1.0E+220, 01762 | 1.0E+221, 1.0E+222, 1.0E+223, 1.0E+224, 1.0E+225, 1.0E+226, 1.0E+227, 1.0E+228, 1.0E+229, 1.0E+230, 01763 | 1.0E+231, 1.0E+232, 1.0E+233, 1.0E+234, 1.0E+235, 1.0E+236, 1.0E+237, 1.0E+238, 1.0E+239, 1.0E+240, 01764 | 1.0E+241, 1.0E+242, 1.0E+243, 1.0E+244, 1.0E+245, 1.0E+246, 1.0E+247, 1.0E+248, 1.0E+249, 1.0E+250, 01765 | 1.0E+251, 1.0E+252, 1.0E+253, 1.0E+254, 1.0E+255, 1.0E+256, 1.0E+257, 1.0E+258, 1.0E+259, 1.0E+260, 01766 | 1.0E+261, 1.0E+262, 1.0E+263, 1.0E+264, 1.0E+265, 1.0E+266, 1.0E+267, 1.0E+268, 1.0E+269, 1.0E+270, 01767 | 1.0E+271, 1.0E+272, 1.0E+273, 1.0E+274, 1.0E+275, 1.0E+276, 1.0E+277, 1.0E+278, 1.0E+279, 1.0E+280, 01768 | 1.0E+281, 1.0E+282, 1.0E+283, 1.0E+284, 1.0E+285, 1.0E+286, 1.0E+287, 1.0E+288, 1.0E+289, 1.0E+290, 01769 | 1.0E+291, 1.0E+292, 1.0E+293, 1.0E+294, 1.0E+295, 1.0E+296, 1.0E+297, 1.0E+298, 1.0E+299, 1.0E+300, 01770 | 1.0E+301, 1.0E+302, 1.0E+303, 1.0E+304, 1.0E+305, 1.0E+306, 1.0E+307, 1.0E+308 01771 | }; 01772 | 01773 | static const int fract10_size = static_cast<int>(sizeof(fract10) / sizeof(double)); 01774 | 01775 | const int e = std::abs(exponent); 01776 | 01777 | if (exponent >= std::numeric_limits<T>::min_exponent10) 01778 | { 01779 | if (e < fract10_size) 01780 | { 01781 | if (exponent > 0) 01782 | return T(d * fract10[e]); 01783 | else 01784 | return T(d / fract10[e]); 01785 | } 01786 | else 01787 | return T(d * std::pow(10.0, 10.0 * exponent)); 01788 | } 01789 | else 01790 | { 01791 | d /= T(fract10[ -std::numeric_limits<T>::min_exponent10]); 01792 | return T(d / fract10[-exponent + std::numeric_limits<T>::min_exponent10]); 01793 | } 01794 | } 01795 | 01796 | template <typename Iterator, typename T> 01797 | inline bool string_to_type_converter_impl_ref(Iterator& itr, const Iterator end, T& result) 01798 | { 01799 | if (itr == end) 01800 | return false; 01801 | 01802 | const bool negative = ('-' == (*itr)); 01803 | 01804 | if (negative || ('+' == (*itr))) 01805 | { 01806 | if (end == ++itr) 01807 | return false; 01808 | } 01809 | 01810 | static const uchar_t zero = static_cast<uchar_t>('0'); 01811 | 01812 | while ((end != itr) && (zero == (*itr))) ++itr; 01813 | 01814 | bool return_result = true; 01815 | unsigned int digit = 0; 01816 | const std::size_t length = static_cast<std::size_t>(std::distance(itr,end)); 01817 | 01818 | if (length <= 4) 01819 | { 01820 | switch (length) 01821 | { 01822 | #ifdef exprtk_use_lut 01823 | 01824 | #define exprtk_process_digit \ 01825 | if ((digit = details::digit_table[(int)*itr++]) < 10) \ 01826 | result = result * 10 + (digit); \ 01827 | else \ 01828 | { \ 01829 | return_result = false; \ 01830 | break; \ 01831 | } \ 01832 | exprtk_fallthrough \ 01833 | 01834 | #else 01835 | 01836 | #define exprtk_process_digit \ 01837 | if ((digit = (*itr++ - zero)) < 10) \ 01838 | result = result * T(10) + digit; \ 01839 | else \ 01840 | { \ 01841 | return_result = false; \ 01842 | break; \ 01843 | } \ 01844 | exprtk_fallthrough \ 01845 | 01846 | #endif 01847 | 01848 | case 4 : exprtk_process_digit 01849 | case 3 : exprtk_process_digit 01850 | case 2 : exprtk_process_digit 01851 | case 1 : if ((digit = (*itr - zero))>= 10) 01852 | { 01853 | digit = 0; 01854 | return_result = false; 01855 | } 01856 | 01857 | #undef exprtk_process_digit 01858 | } 01859 | } 01860 | else 01861 | return_result = false; 01862 | 01863 | if (length && return_result) 01864 | { 01865 | result = result * 10 + static_cast<T>(digit); 01866 | ++itr; 01867 | } 01868 | 01869 | result = negative ? -result : result; 01870 | return return_result; 01871 | } 01872 | 01873 | template <typename Iterator, typename T> 01874 | static inline bool parse_nan(Iterator& itr, const Iterator end, T& t) 01875 | { 01876 | typedef typename std::iterator_traits<Iterator>::value_type type; 01877 | 01878 | static const std::size_t nan_length = 3; 01879 | 01880 | if (std::distance(itr,end) != static_cast<int>(nan_length)) 01881 | return false; 01882 | 01883 | if (static_cast<type>('n') == (*itr)) 01884 | { 01885 | if ( 01886 | (static_cast<type>('a') != *(itr + 1)) || 01887 | (static_cast<type>('n') != *(itr + 2)) 01888 | ) 01889 | { 01890 | return false; 01891 | } 01892 | } 01893 | else if ( 01894 | (static_cast<type>('A') != *(itr + 1)) || 01895 | (static_cast<type>('N') != *(itr + 2)) 01896 | ) 01897 | { 01898 | return false; 01899 | } 01900 | 01901 | t = std::numeric_limits<T>::quiet_NaN(); 01902 | 01903 | return true; 01904 | } 01905 | 01906 | template <typename Iterator, typename T> 01907 | static inline bool parse_inf(Iterator& itr, const Iterator end, T& t, const bool negative) 01908 | { 01909 | static const char_t inf_uc[] = "INFINITY" 01910 | static const char_t inf_lc[] = "infinity" 01911 | static const std::size_t inf_length = 8; 01912 | 01913 | const std::size_t length = static_cast<std::size_t>(std::distance(itr,end)); 01914 | 01915 | if ((3 != length) && (inf_length != length)) 01916 | return false; 01917 | 01918 | char_cptr inf_itr = ('i' == (*itr)) ? inf_lc : inf_uc; 01919 | 01920 | while (end != itr) 01921 | { 01922 | if (*inf_itr == static_cast<char_t>(*itr)) 01923 | { 01924 | ++itr; 01925 | ++inf_itr; 01926 | continue; 01927 | } 01928 | else 01929 | return false; 01930 | } 01931 | 01932 | if (negative) 01933 | t = -std::numeric_limits<T>::infinity(); 01934 | else 01935 | t = std::numeric_limits<T>::infinity(); 01936 | 01937 | return true; 01938 | } 01939 | 01940 | template <typename T> 01941 | inline bool valid_exponent(const int exponent, numeric::details::real_type_tag) 01942 | { 01943 | using namespace details::numeric; 01944 | return (numeric_info<T>::min_exp <= exponent) && (exponent <= numeric_info<T>::max_exp); 01945 | } 01946 | 01947 | template <typename Iterator, typename T> 01948 | inline bool string_to_real(Iterator& itr_external, const Iterator end, T& t, numeric::details::real_type_tag) 01949 | { 01950 | if (end == itr_external) return false; 01951 | 01952 | Iterator itr = itr_external; 01953 | 01954 | T d = T(0); 01955 | 01956 | const bool negative = ('-' == (*itr)); 01957 | 01958 | if (negative || '+' == (*itr)) 01959 | { 01960 | if (end == ++itr) 01961 | return false; 01962 | } 01963 | 01964 | bool instate = false; 01965 | 01966 | static const char_t zero = static_cast<uchar_t>('0'); 01967 | 01968 | #define parse_digit_1(d) \ 01969 | if ((digit = (*itr - zero)) < 10) \ 01970 | { d = d * T(10) + digit; } \ 01971 | else \ 01972 | { break; } \ 01973 | if (end == ++itr) break; \ 01974 | 01975 | #define parse_digit_2(d) \ 01976 | if ((digit = (*itr - zero)) < 10) \ 01977 | { d = d * T(10) + digit; } \ 01978 | else \ 01979 | { break; } \ 01980 | ++itr; \ 01981 | 01982 | if ('.' != (*itr)) 01983 | { 01984 | const Iterator curr = itr; 01985 | 01986 | while ((end != itr) && (zero == (*itr))) ++itr; 01987 | 01988 | while (end != itr) 01989 | { 01990 | unsigned int digit; 01991 | parse_digit_1(d) 01992 | parse_digit_1(d) 01993 | parse_digit_2(d) 01994 | } 01995 | 01996 | if (curr != itr) instate = true; 01997 | } 01998 | 01999 | int exponent = 0; 02000 | 02001 | if (end != itr) 02002 | { 02003 | if ('.' == (*itr)) 02004 | { 02005 | const Iterator curr = ++itr; 02006 | T tmp_d = T(0); 02007 | 02008 | while (end != itr) 02009 | { 02010 | unsigned int digit; 02011 | parse_digit_1(tmp_d) 02012 | parse_digit_1(tmp_d) 02013 | parse_digit_2(tmp_d) 02014 | } 02015 | 02016 | if (curr != itr) 02017 | { 02018 | instate = true; 02019 | 02020 | const int frac_exponent = static_cast<int>(-std::distance(curr, itr)); 02021 | 02022 | if (!valid_exponent<T>(frac_exponent, numeric::details::real_type_tag())) 02023 | return false; 02024 | 02025 | d += compute_pow10(tmp_d, frac_exponent); 02026 | } 02027 | 02028 | #undef parse_digit_1 02029 | #undef parse_digit_2 02030 | } 02031 | 02032 | if (end != itr) 02033 | { 02034 | typename std::iterator_traits<Iterator>::value_type c = (*itr); 02035 | 02036 | if (('e' == c) || ('E' == c)) 02037 | { 02038 | int exp = 0; 02039 | 02040 | if (!details::string_to_type_converter_impl_ref(++itr, end, exp)) 02041 | { 02042 | if (end == itr) 02043 | return false; 02044 | else 02045 | c = (*itr); 02046 | } 02047 | 02048 | exponent += exp; 02049 | } 02050 | 02051 | if (end != itr) 02052 | { 02053 | if (('f' == c) || ('F' == c) || ('l' == c) || ('L' == c)) 02054 | ++itr; 02055 | else if ('#' == c) 02056 | { 02057 | if (end == ++itr) 02058 | return false; 02059 | else if (('I' <= (*itr)) && ((*itr) <= 'n')) 02060 | { 02061 | if (('i' == (*itr)) || ('I' == (*itr))) 02062 | { 02063 | return parse_inf(itr, end, t, negative); 02064 | } 02065 | else if (('n' == (*itr)) || ('N' == (*itr))) 02066 | { 02067 | return parse_nan(itr, end, t); 02068 | } 02069 | else 02070 | return false; 02071 | } 02072 | else 02073 | return false; 02074 | } 02075 | else if (('I' <= (*itr)) && ((*itr) <= 'n')) 02076 | { 02077 | if (('i' == (*itr)) || ('I' == (*itr))) 02078 | { 02079 | return parse_inf(itr, end, t, negative); 02080 | } 02081 | else if (('n' == (*itr)) || ('N' == (*itr))) 02082 | { 02083 | return parse_nan(itr, end, t); 02084 | } 02085 | else 02086 | return false; 02087 | } 02088 | else 02089 | return false; 02090 | } 02091 | } 02092 | } 02093 | 02094 | if ((end != itr) || (!instate)) 02095 | return false; 02096 | else if (!valid_exponent<T>(exponent, numeric::details::real_type_tag())) 02097 | return false; 02098 | else if (exponent) 02099 | d = compute_pow10(d,exponent); 02100 | 02101 | t = static_cast<T>((negative) ? -d : d); 02102 | return true; 02103 | } 02104 | 02105 | template <typename T> 02106 | inline bool string_to_real(const std::string& s, T& t) 02107 | { 02108 | const typename numeric::details::number_type<T>::type num_type; 02109 | 02110 | char_cptr begin = s.data(); 02111 | char_cptr end = s.data() + s.size(); 02112 | 02113 | return string_to_real(begin, end, t, num_type); 02114 | } 02115 | 02116 | template <typename T> 02117 | struct functor_t 02118 | { 02119 | /* 02120 | Note: The following definitions for Type, may require tweaking 02121 | based on the compiler and target architecture. The benchmark 02122 | should provide enough information to make the right choice. 02123 | */ 02124 | //typedef T Type; 02125 | //typedef const T Type; 02126 | typedef const T& Type; 02127 | typedef T& RefType; 02128 | typedef T (*qfunc_t)(Type t0, Type t1, Type t2, Type t3); 02129 | typedef T (*tfunc_t)(Type t0, Type t1, Type t2); 02130 | typedef T (*bfunc_t)(Type t0, Type t1); 02131 | typedef T (*ufunc_t)(Type t0); 02132 | }; 02133 | 02134 | } // namespace details 02135 | 02136 | struct loop_runtime_check 02137 | { 02138 | enum loop_types 02139 | { 02140 | e_invalid = 0, 02141 | e_for_loop = 1, 02142 | e_while_loop = 2, 02143 | e_repeat_until_loop = 4, 02144 | e_all_loops = 7 02145 | }; 02146 | 02147 | enum violation_type 02148 | { 02149 | e_unknown = 0, 02150 | e_iteration_count = 1, 02151 | e_timeout = 2 02152 | }; 02153 | 02154 | loop_types loop_set; 02155 | 02156 | loop_runtime_check() 02157 | : loop_set(e_invalid) 02158 | , max_loop_iterations(0) 02159 | {} 02160 | 02161 | details::_uint64_t max_loop_iterations; 02162 | 02163 | struct violation_context 02164 | { 02165 | loop_types loop; 02166 | violation_type violation; 02167 | details::_uint64_t iteration_count; 02168 | }; 02169 | 02170 | virtual bool check() 02171 | { 02172 | return true; 02173 | } 02174 | 02175 | virtual void handle_runtime_violation(const violation_context&) 02176 | { 02177 | throw std::runtime_error("ExprTk Loop runtime violation."); 02178 | } 02179 | 02180 | virtual ~loop_runtime_check() 02181 | {} 02182 | }; 02183 | 02184 | typedef loop_runtime_check* loop_runtime_check_ptr; 02185 | 02186 | struct vector_access_runtime_check 02187 | { 02188 | struct violation_context 02189 | { 02190 | void* base_ptr; 02191 | void* end_ptr; 02192 | void* access_ptr; 02193 | std::size_t type_size; 02194 | }; 02195 | 02196 | virtual ~vector_access_runtime_check() 02197 | {} 02198 | 02199 | virtual bool handle_runtime_violation(violation_context& /*context*/) 02200 | { 02201 | throw std::runtime_error("ExprTk runtime vector access violation."); 02202 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 02203 | return false; 02204 | #endif 02205 | } 02206 | }; 02207 | 02208 | typedef vector_access_runtime_check* vector_access_runtime_check_ptr; 02209 | 02210 | struct assert_check 02211 | { 02212 | struct assert_context 02213 | { 02214 | std::string condition; 02215 | std::string message; 02216 | std::string id; 02217 | std::size_t offet; 02218 | }; 02219 | 02220 | virtual ~assert_check() 02221 | {} 02222 | 02223 | virtual void handle_assert(const assert_context& /*context*/) 02224 | { 02225 | } 02226 | }; 02227 | 02228 | typedef assert_check* assert_check_ptr; 02229 | 02230 | struct compilation_check 02231 | { 02232 | struct compilation_context 02233 | { 02234 | std::string error_message; 02235 | }; 02236 | 02237 | virtual bool continue_compilation(compilation_context& /*context*/) = 0; 02238 | 02239 | virtual ~compilation_check() 02240 | {} 02241 | }; 02242 | 02243 | typedef compilation_check* compilation_check_ptr; 02244 | 02245 | namespace lexer 02246 | { 02247 | struct token 02248 | { 02249 | enum token_type 02250 | { 02251 | e_none = 0, e_error = 1, e_err_symbol = 2, 02252 | e_err_number = 3, e_err_string = 4, e_err_sfunc = 5, 02253 | e_eof = 6, e_number = 7, e_symbol = 8, 02254 | e_string = 9, e_assign = 10, e_addass = 11, 02255 | e_subass = 12, e_mulass = 13, e_divass = 14, 02256 | e_modass = 15, e_shr = 16, e_shl = 17, 02257 | e_lte = 18, e_ne = 19, e_gte = 20, 02258 | e_swap = 21, e_lt = '<', e_gt = '>', 02259 | e_eq = '=', e_rbracket = ')', e_lbracket = '(', 02260 | e_rsqrbracket = ']', e_lsqrbracket = '[', e_rcrlbracket = '}', 02261 | e_lcrlbracket = '{', e_comma = ',', e_add = '+', 02262 | e_sub = '-', e_div = '/', e_mul = '*', 02263 | e_mod = '%', e_pow = '^', e_colon = ':', 02264 | e_ternary = '?' 02265 | }; 02266 | 02267 | token() 02268 | : type(e_none) 02269 | , value("") 02270 | , position(std::numeric_limits<std::size_t>::max()) 02271 | {} 02272 | 02273 | void clear() 02274 | { 02275 | type = e_none; 02276 | value = "" 02277 | position = std::numeric_limits<std::size_t>::max(); 02278 | } 02279 | 02280 | template <typename Iterator> 02281 | inline token& set_operator(const token_type tt, 02282 | const Iterator begin, const Iterator end, 02283 | const Iterator base_begin = Iterator(0)) 02284 | { 02285 | type = tt; 02286 | value.assign(begin,end); 02287 | if (base_begin) 02288 | position = static_cast<std::size_t>(std::distance(base_begin,begin)); 02289 | return (*this); 02290 | } 02291 | 02292 | template <typename Iterator> 02293 | inline token& set_symbol(const Iterator begin, const Iterator end, const Iterator base_begin = Iterator(0)) 02294 | { 02295 | type = e_symbol; 02296 | value.assign(begin,end); 02297 | if (base_begin) 02298 | position = static_cast<std::size_t>(std::distance(base_begin,begin)); 02299 | return (*this); 02300 | } 02301 | 02302 | template <typename Iterator> 02303 | inline token& set_numeric(const Iterator begin, const Iterator end, const Iterator base_begin = Iterator(0)) 02304 | { 02305 | type = e_number; 02306 | value.assign(begin,end); 02307 | if (base_begin) 02308 | position = static_cast<std::size_t>(std::distance(base_begin,begin)); 02309 | return (*this); 02310 | } 02311 | 02312 | template <typename Iterator> 02313 | inline token& set_string(const Iterator begin, const Iterator end, const Iterator base_begin = Iterator(0)) 02314 | { 02315 | type = e_string; 02316 | value.assign(begin,end); 02317 | if (base_begin) 02318 | position = static_cast<std::size_t>(std::distance(base_begin,begin)); 02319 | return (*this); 02320 | } 02321 | 02322 | inline token& set_string(const std::string& s, const std::size_t p) 02323 | { 02324 | type = e_string; 02325 | value = s; 02326 | position = p; 02327 | return (*this); 02328 | } 02329 | 02330 | template <typename Iterator> 02331 | inline token& set_error(const token_type et, 02332 | const Iterator begin, const Iterator end, 02333 | const Iterator base_begin = Iterator(0)) 02334 | { 02335 | if ( 02336 | (e_error == et) || 02337 | (e_err_symbol == et) || 02338 | (e_err_number == et) || 02339 | (e_err_string == et) || 02340 | (e_err_sfunc == et) 02341 | ) 02342 | { 02343 | type = et; 02344 | } 02345 | else 02346 | type = e_error; 02347 | 02348 | value.assign(begin,end); 02349 | 02350 | if (base_begin) 02351 | position = static_cast<std::size_t>(std::distance(base_begin,begin)); 02352 | 02353 | return (*this); 02354 | } 02355 | 02356 | static inline std::string to_str(token_type t) 02357 | { 02358 | switch (t) 02359 | { 02360 | case e_none : return "NONE" 02361 | case e_error : return "ERROR" 02362 | case e_err_symbol : return "ERROR_SYMBOL" 02363 | case e_err_number : return "ERROR_NUMBER" 02364 | case e_err_string : return "ERROR_STRING" 02365 | case e_eof : return "EOF" 02366 | case e_number : return "NUMBER" 02367 | case e_symbol : return "SYMBOL" 02368 | case e_string : return "STRING" 02369 | case e_assign : return ":=" 02370 | case e_addass : return "+=" 02371 | case e_subass : return "-=" 02372 | case e_mulass : return "*=" 02373 | case e_divass : return "/=" 02374 | case e_modass : return "%=" 02375 | case e_shr : return ">>" 02376 | case e_shl : return "<<" 02377 | case e_lte : return "<=" 02378 | case e_ne : return "!=" 02379 | case e_gte : return ">=" 02380 | case e_lt : return "<" 02381 | case e_gt : return ">" 02382 | case e_eq : return "=" 02383 | case e_rbracket : return ")" 02384 | case e_lbracket : return "(" 02385 | case e_rsqrbracket : return "]" 02386 | case e_lsqrbracket : return "[" 02387 | case e_rcrlbracket : return "}" 02388 | case e_lcrlbracket : return "{" 02389 | case e_comma : return "," 02390 | case e_add : return "+" 02391 | case e_sub : return "-" 02392 | case e_div : return "/" 02393 | case e_mul : return "*" 02394 | case e_mod : return "%" 02395 | case e_pow : return "^" 02396 | case e_colon : return ":" 02397 | case e_ternary : return "?" 02398 | case e_swap : return "<=>" 02399 | default : return "UNKNOWN" 02400 | } 02401 | } 02402 | 02403 | static inline std::string seperator_to_str(const token_type t) 02404 | { 02405 | switch (t) 02406 | { 02407 | case e_comma : return "," 02408 | case e_colon : return ":" 02409 | case e_eof : return "" 02410 | default : return "UNKNOWN" 02411 | } 02412 | 02413 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 02414 | return "UNKNOWN" 02415 | #endif 02416 | } 02417 | 02418 | inline bool is_error() const 02419 | { 02420 | return ( 02421 | (e_error == type) || 02422 | (e_err_symbol == type) || 02423 | (e_err_number == type) || 02424 | (e_err_string == type) || 02425 | (e_err_sfunc == type) 02426 | ); 02427 | } 02428 | 02429 | token_type type; 02430 | std::string value; 02431 | std::size_t position; 02432 | }; 02433 | 02434 | class generator 02435 | { 02436 | public: 02437 | 02438 | typedef token token_t; 02439 | typedef std::vector<token_t> token_list_t; 02440 | typedef token_list_t::iterator token_list_itr_t; 02441 | typedef details::char_t char_t; 02442 | 02443 | generator() 02444 | : base_itr_(0) 02445 | , s_itr_ (0) 02446 | , s_end_ (0) 02447 | { 02448 | clear(); 02449 | } 02450 | 02451 | inline void clear() 02452 | { 02453 | base_itr_ = 0; 02454 | s_itr_ = 0; 02455 | s_end_ = 0; 02456 | token_list_.clear(); 02457 | token_itr_ = token_list_.end(); 02458 | store_token_itr_ = token_list_.end(); 02459 | } 02460 | 02461 | inline bool process(const std::string& str) 02462 | { 02463 | base_itr_ = str.data(); 02464 | s_itr_ = str.data(); 02465 | s_end_ = str.data() + str.size(); 02466 | 02467 | eof_token_.set_operator(token_t::e_eof, s_end_, s_end_, base_itr_); 02468 | token_list_.clear(); 02469 | 02470 | while (!is_end(s_itr_)) 02471 | { 02472 | scan_token(); 02473 | 02474 | if (!token_list_.empty() && token_list_.back().is_error()) 02475 | return false; 02476 | } 02477 | 02478 | return true; 02479 | } 02480 | 02481 | inline bool empty() const 02482 | { 02483 | return token_list_.empty(); 02484 | } 02485 | 02486 | inline std::size_t size() const 02487 | { 02488 | return token_list_.size(); 02489 | } 02490 | 02491 | inline void begin() 02492 | { 02493 | token_itr_ = token_list_.begin(); 02494 | store_token_itr_ = token_list_.begin(); 02495 | } 02496 | 02497 | inline void store() 02498 | { 02499 | store_token_itr_ = token_itr_; 02500 | } 02501 | 02502 | inline void restore() 02503 | { 02504 | token_itr_ = store_token_itr_; 02505 | } 02506 | 02507 | inline token_t& next_token() 02508 | { 02509 | if (token_list_.end() != token_itr_) 02510 | { 02511 | return *token_itr_++; 02512 | } 02513 | else 02514 | return eof_token_; 02515 | } 02516 | 02517 | inline token_t& peek_next_token() 02518 | { 02519 | if (token_list_.end() != token_itr_) 02520 | { 02521 | return *token_itr_; 02522 | } 02523 | else 02524 | return eof_token_; 02525 | } 02526 | 02527 | inline token_t& operator[](const std::size_t& index) 02528 | { 02529 | if (index < token_list_.size()) 02530 | { 02531 | return token_list_[index]; 02532 | } 02533 | else 02534 | return eof_token_; 02535 | } 02536 | 02537 | inline token_t operator[](const std::size_t& index) const 02538 | { 02539 | if (index < token_list_.size()) 02540 | { 02541 | return token_list_[index]; 02542 | } 02543 | else 02544 | return eof_token_; 02545 | } 02546 | 02547 | inline bool finished() const 02548 | { 02549 | return (token_list_.end() == token_itr_); 02550 | } 02551 | 02552 | inline void insert_front(token_t::token_type tk_type) 02553 | { 02554 | if ( 02555 | !token_list_.empty() && 02556 | (token_list_.end() != token_itr_) 02557 | ) 02558 | { 02559 | token_t t = *token_itr_; 02560 | 02561 | t.type = tk_type; 02562 | token_itr_ = token_list_.insert(token_itr_,t); 02563 | } 02564 | } 02565 | 02566 | inline std::string substr(const std::size_t& begin, const std::size_t& end) const 02567 | { 02568 | const details::char_cptr begin_itr = ((base_itr_ + begin) < s_end_) ? (base_itr_ + begin) : s_end_; 02569 | const details::char_cptr end_itr = ((base_itr_ + end ) < s_end_) ? (base_itr_ + end ) : s_end_; 02570 | 02571 | return std::string(begin_itr,end_itr); 02572 | } 02573 | 02574 | inline std::string remaining() const 02575 | { 02576 | if (finished()) 02577 | return "" 02578 | else if (token_list_.begin() != token_itr_) 02579 | return std::string(base_itr_ + (token_itr_ - 1)->position, s_end_); 02580 | else 02581 | return std::string(base_itr_ + token_itr_->position, s_end_); 02582 | } 02583 | 02584 | private: 02585 | 02586 | inline bool is_end(details::char_cptr itr) const 02587 | { 02588 | return (s_end_ == itr); 02589 | } 02590 | 02591 | #ifndef exprtk_disable_comments 02592 | inline bool is_comment_start(details::char_cptr itr) const 02593 | { 02594 | const char_t c0 = *(itr + 0); 02595 | const char_t c1 = *(itr + 1); 02596 | 02597 | if ('#' == c0) 02598 | return true; 02599 | else if (!is_end(itr + 1)) 02600 | { 02601 | if (('/' == c0) && ('/' == c1)) return true; 02602 | if (('/' == c0) && ('*' == c1)) return true; 02603 | } 02604 | return false; 02605 | } 02606 | #else 02607 | inline bool is_comment_start(details::char_cptr) const 02608 | { 02609 | return false; 02610 | } 02611 | #endif 02612 | 02613 | inline void skip_whitespace() 02614 | { 02615 | while (!is_end(s_itr_) && details::is_whitespace(*s_itr_)) 02616 | { 02617 | ++s_itr_; 02618 | } 02619 | } 02620 | 02621 | inline void skip_comments() 02622 | { 02623 | #ifndef exprtk_disable_comments 02624 | // The following comment styles are supported: 02625 | // 1. // .... \n 02626 | // 2. # .... \n 02627 | // 3. /* .... */ 02628 | struct test 02629 | { 02630 | static inline bool comment_start(const char_t c0, const char_t c1, int& mode, int& incr) 02631 | { 02632 | mode = 0; 02633 | if ('#' == c0) { mode = 1; incr = 1; } 02634 | else if ('/' == c0) 02635 | { 02636 | if ('/' == c1) { mode = 1; incr = 2; } 02637 | else if ('*' == c1) { mode = 2; incr = 2; } 02638 | } 02639 | return (0 != mode); 02640 | } 02641 | 02642 | static inline bool comment_end(const char_t c0, const char_t c1, int& mode) 02643 | { 02644 | if ( 02645 | ((1 == mode) && ('\n' == c0)) || 02646 | ((2 == mode) && ( '*' == c0) && ('/' == c1)) 02647 | ) 02648 | { 02649 | mode = 0; 02650 | return true; 02651 | } 02652 | else 02653 | return false; 02654 | } 02655 | }; 02656 | 02657 | int mode = 0; 02658 | int increment = 0; 02659 | 02660 | if (is_end(s_itr_)) 02661 | return; 02662 | else if (!test::comment_start(*s_itr_, *(s_itr_ + 1), mode, increment)) 02663 | return; 02664 | 02665 | details::char_cptr cmt_start = s_itr_; 02666 | 02667 | s_itr_ += increment; 02668 | 02669 | while (!is_end(s_itr_)) 02670 | { 02671 | if ((1 == mode) && test::comment_end(*s_itr_, 0, mode)) 02672 | { 02673 | ++s_itr_; 02674 | return; 02675 | } 02676 | 02677 | if ((2 == mode)) 02678 | { 02679 | if (!is_end((s_itr_ + 1)) && test::comment_end(*s_itr_, *(s_itr_ + 1), mode)) 02680 | { 02681 | s_itr_ += 2; 02682 | return; 02683 | } 02684 | } 02685 | 02686 | ++s_itr_; 02687 | } 02688 | 02689 | if (2 == mode) 02690 | { 02691 | token_t t; 02692 | t.set_error(token::e_error, cmt_start, cmt_start + mode, base_itr_); 02693 | token_list_.push_back(t); 02694 | } 02695 | #endif 02696 | } 02697 | 02698 | inline bool next_is_digit(const details::char_cptr itr) const 02699 | { 02700 | return ((itr + 1) != s_end_) && 02701 | details::is_digit(*(itr + 1)); 02702 | } 02703 | 02704 | inline void scan_token() 02705 | { 02706 | const char_t c = *s_itr_; 02707 | 02708 | if (details::is_whitespace(c)) 02709 | { 02710 | skip_whitespace(); 02711 | return; 02712 | } 02713 | else if (is_comment_start(s_itr_)) 02714 | { 02715 | skip_comments(); 02716 | return; 02717 | } 02718 | else if (details::is_operator_char(c)) 02719 | { 02720 | scan_operator(); 02721 | return; 02722 | } 02723 | else if (details::is_letter(c)) 02724 | { 02725 | scan_symbol(); 02726 | return; 02727 | } 02728 | else if (('.' == c) && !next_is_digit(s_itr_)) 02729 | { 02730 | scan_operator(); 02731 | return; 02732 | } 02733 | else if (details::is_digit(c) || ('.' == c)) 02734 | { 02735 | scan_number(); 02736 | return; 02737 | } 02738 | else if ('$' == c) 02739 | { 02740 | scan_special_function(); 02741 | return; 02742 | } 02743 | #ifndef exprtk_disable_string_capabilities 02744 | else if ('\'' == c) 02745 | { 02746 | scan_string(); 02747 | return; 02748 | } 02749 | #endif 02750 | else if ('~' == c) 02751 | { 02752 | token_t t; 02753 | t.set_symbol(s_itr_, s_itr_ + 1, base_itr_); 02754 | token_list_.push_back(t); 02755 | ++s_itr_; 02756 | return; 02757 | } 02758 | else 02759 | { 02760 | token_t t; 02761 | t.set_error(token::e_error, s_itr_, s_itr_ + 2, base_itr_); 02762 | token_list_.push_back(t); 02763 | ++s_itr_; 02764 | } 02765 | } 02766 | 02767 | inline void scan_operator() 02768 | { 02769 | token_t t; 02770 | 02771 | const char_t c0 = s_itr_[0]; 02772 | 02773 | if (!is_end(s_itr_ + 1)) 02774 | { 02775 | const char_t c1 = s_itr_[1]; 02776 | 02777 | if (!is_end(s_itr_ + 2)) 02778 | { 02779 | const char_t c2 = s_itr_[2]; 02780 | 02781 | if ((c0 == '<') && (c1 == '=') && (c2 == '>')) 02782 | { 02783 | t.set_operator(token_t::e_swap, s_itr_, s_itr_ + 3, base_itr_); 02784 | token_list_.push_back(t); 02785 | s_itr_ += 3; 02786 | return; 02787 | } 02788 | } 02789 | 02790 | token_t::token_type ttype = token_t::e_none; 02791 | 02792 | if ((c0 == '<') && (c1 == '=')) ttype = token_t::e_lte; 02793 | else if ((c0 == '>') && (c1 == '=')) ttype = token_t::e_gte; 02794 | else if ((c0 == '<') && (c1 == '>')) ttype = token_t::e_ne; 02795 | else if ((c0 == '!') && (c1 == '=')) ttype = token_t::e_ne; 02796 | else if ((c0 == '=') && (c1 == '=')) ttype = token_t::e_eq; 02797 | else if ((c0 == ':') && (c1 == '=')) ttype = token_t::e_assign; 02798 | else if ((c0 == '<') && (c1 == '<')) ttype = token_t::e_shl; 02799 | else if ((c0 == '>') && (c1 == '>')) ttype = token_t::e_shr; 02800 | else if ((c0 == '+') && (c1 == '=')) ttype = token_t::e_addass; 02801 | else if ((c0 == '-') && (c1 == '=')) ttype = token_t::e_subass; 02802 | else if ((c0 == '*') && (c1 == '=')) ttype = token_t::e_mulass; 02803 | else if ((c0 == '/') && (c1 == '=')) ttype = token_t::e_divass; 02804 | else if ((c0 == '%') && (c1 == '=')) ttype = token_t::e_modass; 02805 | 02806 | if (token_t::e_none != ttype) 02807 | { 02808 | t.set_operator(ttype, s_itr_, s_itr_ + 2, base_itr_); 02809 | token_list_.push_back(t); 02810 | s_itr_ += 2; 02811 | return; 02812 | } 02813 | } 02814 | 02815 | if ('<' == c0) 02816 | t.set_operator(token_t::e_lt , s_itr_, s_itr_ + 1, base_itr_); 02817 | else if ('>' == c0) 02818 | t.set_operator(token_t::e_gt , s_itr_, s_itr_ + 1, base_itr_); 02819 | else if (';' == c0) 02820 | t.set_operator(token_t::e_eof, s_itr_, s_itr_ + 1, base_itr_); 02821 | else if ('&' == c0) 02822 | t.set_symbol(s_itr_, s_itr_ + 1, base_itr_); 02823 | else if ('|' == c0) 02824 | t.set_symbol(s_itr_, s_itr_ + 1, base_itr_); 02825 | else 02826 | t.set_operator(token_t::token_type(c0), s_itr_, s_itr_ + 1, base_itr_); 02827 | 02828 | token_list_.push_back(t); 02829 | ++s_itr_; 02830 | } 02831 | 02832 | inline void scan_symbol() 02833 | { 02834 | details::char_cptr initial_itr = s_itr_; 02835 | 02836 | while (!is_end(s_itr_)) 02837 | { 02838 | if (!details::is_letter_or_digit(*s_itr_) && ('_' != (*s_itr_))) 02839 | { 02840 | if ('.' != (*s_itr_)) 02841 | break; 02842 | /* 02843 | Permit symbols that contain a 'dot' 02844 | Allowed : abc.xyz, a123.xyz, abc.123, abc_.xyz a123_.xyz abc._123 02845 | Disallowed: .abc, abc.<white-space>, abc.<eof>, abc.<operator +,-,*,/...> 02846 | */ 02847 | if ( 02848 | (s_itr_ != initial_itr) && 02849 | !is_end(s_itr_ + 1) && 02850 | !details::is_letter_or_digit(*(s_itr_ + 1)) && 02851 | ('_' != (*(s_itr_ + 1))) 02852 | ) 02853 | break; 02854 | } 02855 | 02856 | ++s_itr_; 02857 | } 02858 | 02859 | token_t t; 02860 | t.set_symbol(initial_itr, s_itr_, base_itr_); 02861 | token_list_.push_back(t); 02862 | } 02863 | 02864 | inline void scan_number() 02865 | { 02866 | /* 02867 | Attempt to match a valid numeric value in one of the following formats: 02868 | (01) 123456 02869 | (02) 123456. 02870 | (03) 123.456 02871 | (04) 123.456e3 02872 | (05) 123.456E3 02873 | (06) 123.456e+3 02874 | (07) 123.456E+3 02875 | (08) 123.456e-3 02876 | (09) 123.456E-3 02877 | (00) .1234 02878 | (11) .1234e3 02879 | (12) .1234E+3 02880 | (13) .1234e+3 02881 | (14) .1234E-3 02882 | (15) .1234e-3 02883 | */ 02884 | 02885 | details::char_cptr initial_itr = s_itr_; 02886 | bool dot_found = false; 02887 | bool e_found = false; 02888 | bool post_e_sign_found = false; 02889 | bool post_e_digit_found = false; 02890 | token_t t; 02891 | 02892 | while (!is_end(s_itr_)) 02893 | { 02894 | if ('.' == (*s_itr_)) 02895 | { 02896 | if (dot_found) 02897 | { 02898 | t.set_error(token::e_err_number, initial_itr, s_itr_, base_itr_); 02899 | token_list_.push_back(t); 02900 | 02901 | return; 02902 | } 02903 | 02904 | dot_found = true; 02905 | ++s_itr_; 02906 | 02907 | continue; 02908 | } 02909 | else if ('e' == std::tolower(*s_itr_)) 02910 | { 02911 | const char_t& c = *(s_itr_ + 1); 02912 | 02913 | if (is_end(s_itr_ + 1)) 02914 | { 02915 | t.set_error(token::e_err_number, initial_itr, s_itr_, base_itr_); 02916 | token_list_.push_back(t); 02917 | 02918 | return; 02919 | } 02920 | else if ( 02921 | ('+' != c) && 02922 | ('-' != c) && 02923 | !details::is_digit(c) 02924 | ) 02925 | { 02926 | t.set_error(token::e_err_number, initial_itr, s_itr_, base_itr_); 02927 | token_list_.push_back(t); 02928 | 02929 | return; 02930 | } 02931 | 02932 | e_found = true; 02933 | ++s_itr_; 02934 | 02935 | continue; 02936 | } 02937 | else if (e_found && details::is_sign(*s_itr_) && !post_e_digit_found) 02938 | { 02939 | if (post_e_sign_found) 02940 | { 02941 | t.set_error(token::e_err_number, initial_itr, s_itr_, base_itr_); 02942 | token_list_.push_back(t); 02943 | 02944 | return; 02945 | } 02946 | 02947 | post_e_sign_found = true; 02948 | ++s_itr_; 02949 | 02950 | continue; 02951 | } 02952 | else if (e_found && details::is_digit(*s_itr_)) 02953 | { 02954 | post_e_digit_found = true; 02955 | ++s_itr_; 02956 | 02957 | continue; 02958 | } 02959 | else if (('.' != (*s_itr_)) && !details::is_digit(*s_itr_)) 02960 | break; 02961 | else 02962 | ++s_itr_; 02963 | } 02964 | 02965 | t.set_numeric(initial_itr, s_itr_, base_itr_); 02966 | token_list_.push_back(t); 02967 | 02968 | return; 02969 | } 02970 | 02971 | inline void scan_special_function() 02972 | { 02973 | details::char_cptr initial_itr = s_itr_; 02974 | token_t t; 02975 | 02976 | // $fdd(x,x,x) = at least 11 chars 02977 | if (std::distance(s_itr_,s_end_) < 11) 02978 | { 02979 | t.set_error( 02980 | token::e_err_sfunc, 02981 | initial_itr, std::min(initial_itr + 11, s_end_), 02982 | base_itr_); 02983 | token_list_.push_back(t); 02984 | 02985 | return; 02986 | } 02987 | 02988 | if ( 02989 | !(('$' == *s_itr_) && 02990 | (details::imatch ('f',*(s_itr_ + 1))) && 02991 | (details::is_digit(*(s_itr_ + 2))) && 02992 | (details::is_digit(*(s_itr_ + 3)))) 02993 | ) 02994 | { 02995 | t.set_error( 02996 | token::e_err_sfunc, 02997 | initial_itr, std::min(initial_itr + 4, s_end_), 02998 | base_itr_); 02999 | token_list_.push_back(t); 03000 | 03001 | return; 03002 | } 03003 | 03004 | s_itr_ += 4; // $fdd = 4chars 03005 | 03006 | t.set_symbol(initial_itr, s_itr_, base_itr_); 03007 | token_list_.push_back(t); 03008 | 03009 | return; 03010 | } 03011 | 03012 | #ifndef exprtk_disable_string_capabilities 03013 | inline void scan_string() 03014 | { 03015 | details::char_cptr initial_itr = s_itr_ + 1; 03016 | token_t t; 03017 | 03018 | if (std::distance(s_itr_,s_end_) < 2) 03019 | { 03020 | t.set_error(token::e_err_string, s_itr_, s_end_, base_itr_); 03021 | token_list_.push_back(t); 03022 | 03023 | return; 03024 | } 03025 | 03026 | ++s_itr_; 03027 | 03028 | bool escaped_found = false; 03029 | bool escaped = false; 03030 | 03031 | while (!is_end(s_itr_)) 03032 | { 03033 | if (!details::is_valid_string_char(*s_itr_)) 03034 | { 03035 | t.set_error(token::e_err_string, initial_itr, s_itr_, base_itr_); 03036 | token_list_.push_back(t); 03037 | 03038 | return; 03039 | } 03040 | else if (!escaped && ('\\' == *s_itr_)) 03041 | { 03042 | escaped_found = true; 03043 | escaped = true; 03044 | ++s_itr_; 03045 | 03046 | continue; 03047 | } 03048 | else if (!escaped) 03049 | { 03050 | if ('\'' == *s_itr_) 03051 | break; 03052 | } 03053 | else if (escaped) 03054 | { 03055 | if ( 03056 | !is_end(s_itr_) && ('0' == *(s_itr_)) && 03057 | ((s_itr_ + 4) <= s_end_) 03058 | ) 03059 | { 03060 | const bool x_separator = ('X' == std::toupper(*(s_itr_ + 1))); 03061 | 03062 | const bool both_digits = details::is_hex_digit(*(s_itr_ + 2)) && 03063 | details::is_hex_digit(*(s_itr_ + 3)) ; 03064 | 03065 | if (!(x_separator && both_digits)) 03066 | { 03067 | t.set_error(token::e_err_string, initial_itr, s_itr_, base_itr_); 03068 | token_list_.push_back(t); 03069 | 03070 | return; 03071 | } 03072 | else 03073 | s_itr_ += 3; 03074 | } 03075 | 03076 | escaped = false; 03077 | } 03078 | 03079 | ++s_itr_; 03080 | } 03081 | 03082 | if (is_end(s_itr_)) 03083 | { 03084 | t.set_error(token::e_err_string, initial_itr, s_itr_, base_itr_); 03085 | token_list_.push_back(t); 03086 | 03087 | return; 03088 | } 03089 | 03090 | if (!escaped_found) 03091 | t.set_string(initial_itr, s_itr_, base_itr_); 03092 | else 03093 | { 03094 | std::string parsed_string(initial_itr,s_itr_); 03095 | 03096 | if (!details::cleanup_escapes(parsed_string)) 03097 | { 03098 | t.set_error(token::e_err_string, initial_itr, s_itr_, base_itr_); 03099 | token_list_.push_back(t); 03100 | 03101 | return; 03102 | } 03103 | 03104 | t.set_string( 03105 | parsed_string, 03106 | static_cast<std::size_t>(std::distance(base_itr_,initial_itr))); 03107 | } 03108 | 03109 | token_list_.push_back(t); 03110 | ++s_itr_; 03111 | 03112 | return; 03113 | } 03114 | #endif 03115 | 03116 | private: 03117 | 03118 | token_list_t token_list_; 03119 | token_list_itr_t token_itr_; 03120 | token_list_itr_t store_token_itr_; 03121 | token_t eof_token_; 03122 | details::char_cptr base_itr_; 03123 | details::char_cptr s_itr_; 03124 | details::char_cptr s_end_; 03125 | 03126 | friend class token_scanner; 03127 | friend class token_modifier; 03128 | friend class token_inserter; 03129 | friend class token_joiner; 03130 | }; // class generator 03131 | 03132 | class helper_interface 03133 | { 03134 | public: 03135 | 03136 | virtual void init() { } 03137 | virtual void reset() { } 03138 | virtual bool result() { return true; } 03139 | virtual std::size_t process(generator&) { return 0; } 03140 | virtual ~helper_interface() { } 03141 | }; 03142 | 03143 | class token_scanner : public helper_interface 03144 | { 03145 | public: 03146 | 03147 | virtual ~token_scanner() exprtk_override 03148 | {} 03149 | 03150 | explicit token_scanner(const std::size_t& stride) 03151 | : stride_(stride) 03152 | { 03153 | if (stride > 4) 03154 | { 03155 | throw std::invalid_argument("token_scanner() - Invalid stride value"); 03156 | } 03157 | } 03158 | 03159 | inline std::size_t process(generator& g) exprtk_override 03160 | { 03161 | if (g.token_list_.size() >= stride_) 03162 | { 03163 | for (std::size_t i = 0; i < (g.token_list_.size() - stride_ + 1); ++i) 03164 | { 03165 | token t; 03166 | 03167 | switch (stride_) 03168 | { 03169 | case 1 : 03170 | { 03171 | const token& t0 = g.token_list_[i]; 03172 | 03173 | if (!operator()(t0)) 03174 | { 03175 | return 0; 03176 | } 03177 | } 03178 | break; 03179 | 03180 | case 2 : 03181 | { 03182 | const token& t0 = g.token_list_[i ]; 03183 | const token& t1 = g.token_list_[i + 1]; 03184 | 03185 | if (!operator()(t0, t1)) 03186 | { 03187 | return 0; 03188 | } 03189 | } 03190 | break; 03191 | 03192 | case 3 : 03193 | { 03194 | const token& t0 = g.token_list_[i ]; 03195 | const token& t1 = g.token_list_[i + 1]; 03196 | const token& t2 = g.token_list_[i + 2]; 03197 | 03198 | if (!operator()(t0, t1, t2)) 03199 | { 03200 | return 0; 03201 | } 03202 | } 03203 | break; 03204 | 03205 | case 4 : 03206 | { 03207 | const token& t0 = g.token_list_[i ]; 03208 | const token& t1 = g.token_list_[i + 1]; 03209 | const token& t2 = g.token_list_[i + 2]; 03210 | const token& t3 = g.token_list_[i + 3]; 03211 | 03212 | if (!operator()(t0, t1, t2, t3)) 03213 | { 03214 | return 0; 03215 | } 03216 | } 03217 | break; 03218 | } 03219 | } 03220 | } 03221 | 03222 | return 0; 03223 | } 03224 | 03225 | virtual bool operator() (const token&) 03226 | { 03227 | return false; 03228 | } 03229 | 03230 | virtual bool operator() (const token&, const token&) 03231 | { 03232 | return false; 03233 | } 03234 | 03235 | virtual bool operator() (const token&, const token&, const token&) 03236 | { 03237 | return false; 03238 | } 03239 | 03240 | virtual bool operator() (const token&, const token&, const token&, const token&) 03241 | { 03242 | return false; 03243 | } 03244 | 03245 | private: 03246 | 03247 | const std::size_t stride_; 03248 | }; // class token_scanner 03249 | 03250 | class token_modifier : public helper_interface 03251 | { 03252 | public: 03253 | 03254 | inline std::size_t process(generator& g) exprtk_override 03255 | { 03256 | std::size_t changes = 0; 03257 | 03258 | for (std::size_t i = 0; i < g.token_list_.size(); ++i) 03259 | { 03260 | if (modify(g.token_list_[i])) changes++; 03261 | } 03262 | 03263 | return changes; 03264 | } 03265 | 03266 | virtual bool modify(token& t) = 0; 03267 | }; 03268 | 03269 | class token_inserter : public helper_interface 03270 | { 03271 | public: 03272 | 03273 | explicit token_inserter(const std::size_t& stride) 03274 | : stride_(stride) 03275 | { 03276 | if (stride > 5) 03277 | { 03278 | throw std::invalid_argument("token_inserter() - Invalid stride value"); 03279 | } 03280 | } 03281 | 03282 | inline std::size_t process(generator& g) exprtk_override 03283 | { 03284 | if (g.token_list_.empty()) 03285 | return 0; 03286 | else if (g.token_list_.size() < stride_) 03287 | return 0; 03288 | 03289 | std::size_t changes = 0; 03290 | 03291 | typedef std::pair<std::size_t, token> insert_t; 03292 | std::vector<insert_t> insert_list; 03293 | insert_list.reserve(10000); 03294 | 03295 | for (std::size_t i = 0; i < (g.token_list_.size() - stride_ + 1); ++i) 03296 | { 03297 | int insert_index = -1; 03298 | token t; 03299 | 03300 | switch (stride_) 03301 | { 03302 | case 1 : insert_index = insert(g.token_list_[i],t); 03303 | break; 03304 | 03305 | case 2 : insert_index = insert(g.token_list_[i], g.token_list_[i + 1], t); 03306 | break; 03307 | 03308 | case 3 : insert_index = insert(g.token_list_[i], g.token_list_[i + 1], g.token_list_[i + 2], t); 03309 | break; 03310 | 03311 | case 4 : insert_index = insert(g.token_list_[i], g.token_list_[i + 1], g.token_list_[i + 2], g.token_list_[i + 3], t); 03312 | break; 03313 | 03314 | case 5 : insert_index = insert(g.token_list_[i], g.token_list_[i + 1], g.token_list_[i + 2], g.token_list_[i + 3], g.token_list_[i + 4], t); 03315 | break; 03316 | } 03317 | 03318 | if ((insert_index >= 0) && (insert_index <= (static_cast<int>(stride_) + 1))) 03319 | { 03320 | insert_list.push_back(insert_t(i, t)); 03321 | changes++; 03322 | } 03323 | } 03324 | 03325 | if (!insert_list.empty()) 03326 | { 03327 | generator::token_list_t token_list; 03328 | 03329 | std::size_t insert_index = 0; 03330 | 03331 | for (std::size_t i = 0; i < g.token_list_.size(); ++i) 03332 | { 03333 | token_list.push_back(g.token_list_[i]); 03334 | 03335 | if ( 03336 | (insert_index < insert_list.size()) && 03337 | (insert_list[insert_index].first == i) 03338 | ) 03339 | { 03340 | token_list.push_back(insert_list[insert_index].second); 03341 | insert_index++; 03342 | } 03343 | } 03344 | 03345 | std::swap(g.token_list_,token_list); 03346 | } 03347 | 03348 | return changes; 03349 | } 03350 | 03351 | #define token_inserter_empty_body \ 03352 | { \ 03353 | return -1; \ 03354 | } \ 03355 | 03356 | inline virtual int insert(const token&, token&) 03357 | token_inserter_empty_body 03358 | 03359 | inline virtual int insert(const token&, const token&, token&) 03360 | token_inserter_empty_body 03361 | 03362 | inline virtual int insert(const token&, const token&, const token&, token&) 03363 | token_inserter_empty_body 03364 | 03365 | inline virtual int insert(const token&, const token&, const token&, const token&, token&) 03366 | token_inserter_empty_body 03367 | 03368 | inline virtual int insert(const token&, const token&, const token&, const token&, const token&, token&) 03369 | token_inserter_empty_body 03370 | 03371 | #undef token_inserter_empty_body 03372 | 03373 | private: 03374 | 03375 | const std::size_t stride_; 03376 | }; 03377 | 03378 | class token_joiner : public helper_interface 03379 | { 03380 | public: 03381 | 03382 | explicit token_joiner(const std::size_t& stride) 03383 | : stride_(stride) 03384 | {} 03385 | 03386 | inline std::size_t process(generator& g) exprtk_override 03387 | { 03388 | if (g.token_list_.empty()) 03389 | return 0; 03390 | 03391 | switch (stride_) 03392 | { 03393 | case 2 : return process_stride_2(g); 03394 | case 3 : return process_stride_3(g); 03395 | default : return 0; 03396 | } 03397 | } 03398 | 03399 | virtual bool join(const token&, const token&, token&) { return false; } 03400 | virtual bool join(const token&, const token&, const token&, token&) { return false; } 03401 | 03402 | private: 03403 | 03404 | inline std::size_t process_stride_2(generator& g) 03405 | { 03406 | if (g.token_list_.size() < 2) 03407 | return 0; 03408 | 03409 | std::size_t changes = 0; 03410 | 03411 | generator::token_list_t token_list; 03412 | token_list.reserve(10000); 03413 | 03414 | for (int i = 0; i < static_cast<int>(g.token_list_.size() - 1); ++i) 03415 | { 03416 | token t; 03417 | 03418 | for ( ; ; ) 03419 | { 03420 | if (!join(g[i], g[i + 1], t)) 03421 | { 03422 | token_list.push_back(g[i]); 03423 | break; 03424 | } 03425 | 03426 | token_list.push_back(t); 03427 | 03428 | ++changes; 03429 | 03430 | i += 2; 03431 | 03432 | if (static_cast<std::size_t>(i) >= (g.token_list_.size() - 1)) 03433 | break; 03434 | } 03435 | } 03436 | 03437 | token_list.push_back(g.token_list_.back()); 03438 | 03439 | assert(token_list.size() <= g.token_list_.size()); 03440 | 03441 | std::swap(token_list, g.token_list_); 03442 | 03443 | return changes; 03444 | } 03445 | 03446 | inline std::size_t process_stride_3(generator& g) 03447 | { 03448 | if (g.token_list_.size() < 3) 03449 | return 0; 03450 | 03451 | std::size_t changes = 0; 03452 | 03453 | generator::token_list_t token_list; 03454 | token_list.reserve(10000); 03455 | 03456 | for (int i = 0; i < static_cast<int>(g.token_list_.size() - 2); ++i) 03457 | { 03458 | token t; 03459 | 03460 | for ( ; ; ) 03461 | { 03462 | if (!join(g[i], g[i + 1], g[i + 2], t)) 03463 | { 03464 | token_list.push_back(g[i]); 03465 | break; 03466 | } 03467 | 03468 | token_list.push_back(t); 03469 | 03470 | ++changes; 03471 | 03472 | i += 3; 03473 | 03474 | if (static_cast<std::size_t>(i) >= (g.token_list_.size() - 2)) 03475 | break; 03476 | } 03477 | } 03478 | 03479 | token_list.push_back(*(g.token_list_.begin() + g.token_list_.size() - 2)); 03480 | token_list.push_back(*(g.token_list_.begin() + g.token_list_.size() - 1)); 03481 | 03482 | assert(token_list.size() <= g.token_list_.size()); 03483 | 03484 | std::swap(token_list, g.token_list_); 03485 | 03486 | return changes; 03487 | } 03488 | 03489 | const std::size_t stride_; 03490 | }; 03491 | 03492 | namespace helper 03493 | { 03494 | 03495 | inline void dump(const lexer::generator& generator) 03496 | { 03497 | for (std::size_t i = 0; i < generator.size(); ++i) 03498 | { 03499 | const lexer::token& t = generator[i]; 03500 | printf("Token[%02d] @ %03d %6s --> '%s'\n", 03501 | static_cast<int>(i), 03502 | static_cast<int>(t.position), 03503 | t.to_str(t.type).c_str(), 03504 | t.value.c_str()); 03505 | } 03506 | } 03507 | 03508 | class commutative_inserter : public lexer::token_inserter 03509 | { 03510 | public: 03511 | 03512 | using lexer::token_inserter::insert; 03513 | 03514 | commutative_inserter() 03515 | : lexer::token_inserter(2) 03516 | {} 03517 | 03518 | inline void ignore_symbol(const std::string& symbol) 03519 | { 03520 | ignore_set_.insert(symbol); 03521 | } 03522 | 03523 | inline int insert(const lexer::token& t0, const lexer::token& t1, lexer::token& new_token) exprtk_override 03524 | { 03525 | bool match = false; 03526 | new_token.type = lexer::token::e_mul; 03527 | new_token.value = "*" 03528 | new_token.position = t1.position; 03529 | 03530 | if (t0.type == lexer::token::e_symbol) 03531 | { 03532 | if (ignore_set_.end() != ignore_set_.find(t0.value)) 03533 | { 03534 | return -1; 03535 | } 03536 | else if (!t0.value.empty() && ('$' == t0.value[0])) 03537 | { 03538 | return -1; 03539 | } 03540 | } 03541 | 03542 | if (t1.type == lexer::token::e_symbol) 03543 | { 03544 | if (ignore_set_.end() != ignore_set_.find(t1.value)) 03545 | { 03546 | return -1; 03547 | } 03548 | } 03549 | if ((t0.type == lexer::token::e_number ) && (t1.type == lexer::token::e_symbol )) match = true; 03550 | else if ((t0.type == lexer::token::e_number ) && (t1.type == lexer::token::e_lbracket )) match = true; 03551 | else if ((t0.type == lexer::token::e_number ) && (t1.type == lexer::token::e_lcrlbracket)) match = true; 03552 | else if ((t0.type == lexer::token::e_number ) && (t1.type == lexer::token::e_lsqrbracket)) match = true; 03553 | else if ((t0.type == lexer::token::e_symbol ) && (t1.type == lexer::token::e_number )) match = true; 03554 | else if ((t0.type == lexer::token::e_rbracket ) && (t1.type == lexer::token::e_number )) match = true; 03555 | else if ((t0.type == lexer::token::e_rcrlbracket) && (t1.type == lexer::token::e_number )) match = true; 03556 | else if ((t0.type == lexer::token::e_rsqrbracket) && (t1.type == lexer::token::e_number )) match = true; 03557 | else if ((t0.type == lexer::token::e_rbracket ) && (t1.type == lexer::token::e_symbol )) match = true; 03558 | else if ((t0.type == lexer::token::e_rcrlbracket) && (t1.type == lexer::token::e_symbol )) match = true; 03559 | else if ((t0.type == lexer::token::e_rsqrbracket) && (t1.type == lexer::token::e_symbol )) match = true; 03560 | else if ((t0.type == lexer::token::e_symbol ) && (t1.type == lexer::token::e_symbol )) match = true; 03561 | 03562 | return (match) ? 1 : -1; 03563 | } 03564 | 03565 | private: 03566 | 03567 | std::set<std::string,details::ilesscompare> ignore_set_; 03568 | }; 03569 | 03570 | class operator_joiner exprtk_final : public token_joiner 03571 | { 03572 | public: 03573 | 03574 | explicit operator_joiner(const std::size_t& stride) 03575 | : token_joiner(stride) 03576 | {} 03577 | 03578 | inline bool join(const lexer::token& t0, const lexer::token& t1, lexer::token& t) exprtk_override 03579 | { 03580 | // ': =' --> ':=' 03581 | if ((t0.type == lexer::token::e_colon) && (t1.type == lexer::token::e_eq)) 03582 | { 03583 | t.type = lexer::token::e_assign; 03584 | t.value = ":=" 03585 | t.position = t0.position; 03586 | 03587 | return true; 03588 | } 03589 | // '+ =' --> '+=' 03590 | else if ((t0.type == lexer::token::e_add) && (t1.type == lexer::token::e_eq)) 03591 | { 03592 | t.type = lexer::token::e_addass; 03593 | t.value = "+=" 03594 | t.position = t0.position; 03595 | 03596 | return true; 03597 | } 03598 | // '- =' --> '-=' 03599 | else if ((t0.type == lexer::token::e_sub) && (t1.type == lexer::token::e_eq)) 03600 | { 03601 | t.type = lexer::token::e_subass; 03602 | t.value = "-=" 03603 | t.position = t0.position; 03604 | 03605 | return true; 03606 | } 03607 | // '* =' --> '*=' 03608 | else if ((t0.type == lexer::token::e_mul) && (t1.type == lexer::token::e_eq)) 03609 | { 03610 | t.type = lexer::token::e_mulass; 03611 | t.value = "*=" 03612 | t.position = t0.position; 03613 | 03614 | return true; 03615 | } 03616 | // '/ =' --> '/=' 03617 | else if ((t0.type == lexer::token::e_div) && (t1.type == lexer::token::e_eq)) 03618 | { 03619 | t.type = lexer::token::e_divass; 03620 | t.value = "/=" 03621 | t.position = t0.position; 03622 | 03623 | return true; 03624 | } 03625 | // '% =' --> '%=' 03626 | else if ((t0.type == lexer::token::e_mod) && (t1.type == lexer::token::e_eq)) 03627 | { 03628 | t.type = lexer::token::e_modass; 03629 | t.value = "%=" 03630 | t.position = t0.position; 03631 | 03632 | return true; 03633 | } 03634 | // '> =' --> '>=' 03635 | else if ((t0.type == lexer::token::e_gt) && (t1.type == lexer::token::e_eq)) 03636 | { 03637 | t.type = lexer::token::e_gte; 03638 | t.value = ">=" 03639 | t.position = t0.position; 03640 | 03641 | return true; 03642 | } 03643 | // '< =' --> '<=' 03644 | else if ((t0.type == lexer::token::e_lt) && (t1.type == lexer::token::e_eq)) 03645 | { 03646 | t.type = lexer::token::e_lte; 03647 | t.value = "<=" 03648 | t.position = t0.position; 03649 | 03650 | return true; 03651 | } 03652 | // '= =' --> '==' 03653 | else if ((t0.type == lexer::token::e_eq) && (t1.type == lexer::token::e_eq)) 03654 | { 03655 | t.type = lexer::token::e_eq; 03656 | t.value = "==" 03657 | t.position = t0.position; 03658 | 03659 | return true; 03660 | } 03661 | // '! =' --> '!=' 03662 | else if ((static_cast<details::char_t>(t0.type) == '!') && (t1.type == lexer::token::e_eq)) 03663 | { 03664 | t.type = lexer::token::e_ne; 03665 | t.value = "!=" 03666 | t.position = t0.position; 03667 | 03668 | return true; 03669 | } 03670 | // '< >' --> '<>' 03671 | else if ((t0.type == lexer::token::e_lt) && (t1.type == lexer::token::e_gt)) 03672 | { 03673 | t.type = lexer::token::e_ne; 03674 | t.value = "<>" 03675 | t.position = t0.position; 03676 | 03677 | return true; 03678 | } 03679 | // '<= >' --> '<=>' 03680 | else if ((t0.type == lexer::token::e_lte) && (t1.type == lexer::token::e_gt)) 03681 | { 03682 | t.type = lexer::token::e_swap; 03683 | t.value = "<=>" 03684 | t.position = t0.position; 03685 | 03686 | return true; 03687 | } 03688 | // '+ -' --> '-' 03689 | else if ((t0.type == lexer::token::e_add) && (t1.type == lexer::token::e_sub)) 03690 | { 03691 | t.type = lexer::token::e_sub; 03692 | t.value = "-" 03693 | t.position = t0.position; 03694 | 03695 | return true; 03696 | } 03697 | // '- +' --> '-' 03698 | else if ((t0.type == lexer::token::e_sub) && (t1.type == lexer::token::e_add)) 03699 | { 03700 | t.type = lexer::token::e_sub; 03701 | t.value = "-" 03702 | t.position = t0.position; 03703 | 03704 | return true; 03705 | } 03706 | // '- -' --> '+' 03707 | else if ((t0.type == lexer::token::e_sub) && (t1.type == lexer::token::e_sub)) 03708 | { 03709 | /* 03710 | Note: May need to reconsider this when wanting to implement 03711 | pre/postfix decrement operator 03712 | */ 03713 | t.type = lexer::token::e_add; 03714 | t.value = "+" 03715 | t.position = t0.position; 03716 | 03717 | return true; 03718 | } 03719 | else 03720 | return false; 03721 | } 03722 | 03723 | inline bool join(const lexer::token& t0, 03724 | const lexer::token& t1, 03725 | const lexer::token& t2, 03726 | lexer::token& t) exprtk_override 03727 | { 03728 | // '[ * ]' --> '[*]' 03729 | if ( 03730 | (t0.type == lexer::token::e_lsqrbracket) && 03731 | (t1.type == lexer::token::e_mul ) && 03732 | (t2.type == lexer::token::e_rsqrbracket) 03733 | ) 03734 | { 03735 | t.type = lexer::token::e_symbol; 03736 | t.value = "[*]" 03737 | t.position = t0.position; 03738 | 03739 | return true; 03740 | } 03741 | else 03742 | return false; 03743 | } 03744 | }; 03745 | 03746 | class bracket_checker exprtk_final : public lexer::token_scanner 03747 | { 03748 | public: 03749 | 03750 | using lexer::token_scanner::operator(); 03751 | 03752 | bracket_checker() 03753 | : token_scanner(1) 03754 | , state_(true) 03755 | {} 03756 | 03757 | bool result() exprtk_override 03758 | { 03759 | if (!stack_.empty()) 03760 | { 03761 | lexer::token t; 03762 | t.value = stack_.top().first; 03763 | t.position = stack_.top().second; 03764 | error_token_ = t; 03765 | state_ = false; 03766 | 03767 | return false; 03768 | } 03769 | else 03770 | return state_; 03771 | } 03772 | 03773 | lexer::token error_token() const 03774 | { 03775 | return error_token_; 03776 | } 03777 | 03778 | void reset() exprtk_override 03779 | { 03780 | // Why? because msvc doesn't support swap properly. 03781 | stack_ = std::stack<std::pair<char,std::size_t> >(); 03782 | state_ = true; 03783 | error_token_.clear(); 03784 | } 03785 | 03786 | bool operator() (const lexer::token& t) exprtk_override 03787 | { 03788 | if ( 03789 | !t.value.empty() && 03790 | (lexer::token::e_string != t.type) && 03791 | (lexer::token::e_symbol != t.type) && 03792 | exprtk::details::is_bracket(t.value[0]) 03793 | ) 03794 | { 03795 | details::char_t c = t.value[0]; 03796 | 03797 | if (t.type == lexer::token::e_lbracket ) stack_.push(std::make_pair(')',t.position)); 03798 | else if (t.type == lexer::token::e_lcrlbracket) stack_.push(std::make_pair('}',t.position)); 03799 | else if (t.type == lexer::token::e_lsqrbracket) stack_.push(std::make_pair(']',t.position)); 03800 | else if (exprtk::details::is_right_bracket(c)) 03801 | { 03802 | if (stack_.empty()) 03803 | { 03804 | state_ = false; 03805 | error_token_ = t; 03806 | 03807 | return false; 03808 | } 03809 | else if (c != stack_.top().first) 03810 | { 03811 | state_ = false; 03812 | error_token_ = t; 03813 | 03814 | return false; 03815 | } 03816 | else 03817 | stack_.pop(); 03818 | } 03819 | } 03820 | 03821 | return true; 03822 | } 03823 | 03824 | private: 03825 | 03826 | bool state_; 03827 | std::stack<std::pair<char,std::size_t> > stack_; 03828 | lexer::token error_token_; 03829 | }; 03830 | 03831 | template <typename T> 03832 | class numeric_checker exprtk_final : public lexer::token_scanner 03833 | { 03834 | public: 03835 | 03836 | using lexer::token_scanner::operator(); 03837 | 03838 | numeric_checker() 03839 | : token_scanner (1) 03840 | , current_index_(0) 03841 | {} 03842 | 03843 | bool result() exprtk_override 03844 | { 03845 | return error_list_.empty(); 03846 | } 03847 | 03848 | void reset() exprtk_override 03849 | { 03850 | error_list_.clear(); 03851 | current_index_ = 0; 03852 | } 03853 | 03854 | bool operator() (const lexer::token& t) exprtk_override 03855 | { 03856 | if (token::e_number == t.type) 03857 | { 03858 | T v; 03859 | 03860 | if (!exprtk::details::string_to_real(t.value,v)) 03861 | { 03862 | error_list_.push_back(current_index_); 03863 | } 03864 | } 03865 | 03866 | ++current_index_; 03867 | 03868 | return true; 03869 | } 03870 | 03871 | std::size_t error_count() const 03872 | { 03873 | return error_list_.size(); 03874 | } 03875 | 03876 | std::size_t error_index(const std::size_t& i) const 03877 | { 03878 | if (i < error_list_.size()) 03879 | return error_list_[i]; 03880 | else 03881 | return std::numeric_limits<std::size_t>::max(); 03882 | } 03883 | 03884 | void clear_errors() 03885 | { 03886 | error_list_.clear(); 03887 | } 03888 | 03889 | private: 03890 | 03891 | std::size_t current_index_; 03892 | std::vector<std::size_t> error_list_; 03893 | }; 03894 | 03895 | class symbol_replacer exprtk_final : public lexer::token_modifier 03896 | { 03897 | private: 03898 | 03899 | typedef std::map<std::string,std::pair<std::string,token::token_type>,details::ilesscompare> replace_map_t; 03900 | 03901 | public: 03902 | 03903 | bool remove(const std::string& target_symbol) 03904 | { 03905 | const replace_map_t::iterator itr = replace_map_.find(target_symbol); 03906 | 03907 | if (replace_map_.end() == itr) 03908 | return false; 03909 | 03910 | replace_map_.erase(itr); 03911 | 03912 | return true; 03913 | } 03914 | 03915 | bool add_replace(const std::string& target_symbol, 03916 | const std::string& replace_symbol, 03917 | const lexer::token::token_type token_type = lexer::token::e_symbol) 03918 | { 03919 | const replace_map_t::iterator itr = replace_map_.find(target_symbol); 03920 | 03921 | if (replace_map_.end() != itr) 03922 | { 03923 | return false; 03924 | } 03925 | 03926 | replace_map_[target_symbol] = std::make_pair(replace_symbol,token_type); 03927 | 03928 | return true; 03929 | } 03930 | 03931 | void clear() 03932 | { 03933 | replace_map_.clear(); 03934 | } 03935 | 03936 | private: 03937 | 03938 | bool modify(lexer::token& t) exprtk_override 03939 | { 03940 | if (lexer::token::e_symbol == t.type) 03941 | { 03942 | if (replace_map_.empty()) 03943 | return false; 03944 | 03945 | const replace_map_t::iterator itr = replace_map_.find(t.value); 03946 | 03947 | if (replace_map_.end() != itr) 03948 | { 03949 | t.value = itr->second.first; 03950 | t.type = itr->second.second; 03951 | 03952 | return true; 03953 | } 03954 | } 03955 | 03956 | return false; 03957 | } 03958 | 03959 | replace_map_t replace_map_; 03960 | }; 03961 | 03962 | class sequence_validator exprtk_final : public lexer::token_scanner 03963 | { 03964 | private: 03965 | 03966 | typedef std::pair<lexer::token::token_type,lexer::token::token_type> token_pair_t; 03967 | typedef std::set<token_pair_t> set_t; 03968 | 03969 | public: 03970 | 03971 | using lexer::token_scanner::operator(); 03972 | 03973 | sequence_validator() 03974 | : lexer::token_scanner(2) 03975 | { 03976 | add_invalid(lexer::token::e_number, lexer::token::e_number); 03977 | add_invalid(lexer::token::e_string, lexer::token::e_string); 03978 | add_invalid(lexer::token::e_number, lexer::token::e_string); 03979 | add_invalid(lexer::token::e_string, lexer::token::e_number); 03980 | 03981 | add_invalid_set1(lexer::token::e_assign ); 03982 | add_invalid_set1(lexer::token::e_shr ); 03983 | add_invalid_set1(lexer::token::e_shl ); 03984 | add_invalid_set1(lexer::token::e_lte ); 03985 | add_invalid_set1(lexer::token::e_ne ); 03986 | add_invalid_set1(lexer::token::e_gte ); 03987 | add_invalid_set1(lexer::token::e_lt ); 03988 | add_invalid_set1(lexer::token::e_gt ); 03989 | add_invalid_set1(lexer::token::e_eq ); 03990 | add_invalid_set1(lexer::token::e_comma ); 03991 | add_invalid_set1(lexer::token::e_add ); 03992 | add_invalid_set1(lexer::token::e_sub ); 03993 | add_invalid_set1(lexer::token::e_div ); 03994 | add_invalid_set1(lexer::token::e_mul ); 03995 | add_invalid_set1(lexer::token::e_mod ); 03996 | add_invalid_set1(lexer::token::e_pow ); 03997 | add_invalid_set1(lexer::token::e_colon ); 03998 | add_invalid_set1(lexer::token::e_ternary); 03999 | } 04000 | 04001 | bool result() exprtk_override 04002 | { 04003 | return error_list_.empty(); 04004 | } 04005 | 04006 | bool operator() (const lexer::token& t0, const lexer::token& t1) exprtk_override 04007 | { 04008 | const set_t::value_type p = std::make_pair(t0.type,t1.type); 04009 | 04010 | if (invalid_bracket_check(t0.type,t1.type)) 04011 | { 04012 | error_list_.push_back(std::make_pair(t0,t1)); 04013 | } 04014 | else if (invalid_comb_.find(p) != invalid_comb_.end()) 04015 | { 04016 | error_list_.push_back(std::make_pair(t0,t1)); 04017 | } 04018 | 04019 | return true; 04020 | } 04021 | 04022 | std::size_t error_count() const 04023 | { 04024 | return error_list_.size(); 04025 | } 04026 | 04027 | std::pair<lexer::token,lexer::token> error(const std::size_t index) 04028 | { 04029 | if (index < error_list_.size()) 04030 | { 04031 | return error_list_[index]; 04032 | } 04033 | else 04034 | { 04035 | static const lexer::token error_token; 04036 | return std::make_pair(error_token,error_token); 04037 | } 04038 | } 04039 | 04040 | void clear_errors() 04041 | { 04042 | error_list_.clear(); 04043 | } 04044 | 04045 | private: 04046 | 04047 | void add_invalid(const lexer::token::token_type base, const lexer::token::token_type t) 04048 | { 04049 | invalid_comb_.insert(std::make_pair(base,t)); 04050 | } 04051 | 04052 | void add_invalid_set1(const lexer::token::token_type t) 04053 | { 04054 | add_invalid(t, lexer::token::e_assign); 04055 | add_invalid(t, lexer::token::e_shr ); 04056 | add_invalid(t, lexer::token::e_shl ); 04057 | add_invalid(t, lexer::token::e_lte ); 04058 | add_invalid(t, lexer::token::e_ne ); 04059 | add_invalid(t, lexer::token::e_gte ); 04060 | add_invalid(t, lexer::token::e_lt ); 04061 | add_invalid(t, lexer::token::e_gt ); 04062 | add_invalid(t, lexer::token::e_eq ); 04063 | add_invalid(t, lexer::token::e_comma ); 04064 | add_invalid(t, lexer::token::e_div ); 04065 | add_invalid(t, lexer::token::e_mul ); 04066 | add_invalid(t, lexer::token::e_mod ); 04067 | add_invalid(t, lexer::token::e_pow ); 04068 | add_invalid(t, lexer::token::e_colon ); 04069 | } 04070 | 04071 | bool invalid_bracket_check(const lexer::token::token_type base, const lexer::token::token_type t) 04072 | { 04073 | if (details::is_right_bracket(static_cast<details::char_t>(base))) 04074 | { 04075 | switch (t) 04076 | { 04077 | case lexer::token::e_assign : return (']' != base); 04078 | case lexer::token::e_string : return (')' != base); 04079 | default : return false; 04080 | } 04081 | } 04082 | else if (details::is_left_bracket(static_cast<details::char_t>(base))) 04083 | { 04084 | if (details::is_right_bracket(static_cast<details::char_t>(t))) 04085 | return false; 04086 | else if (details::is_left_bracket(static_cast<details::char_t>(t))) 04087 | return false; 04088 | else 04089 | { 04090 | switch (t) 04091 | { 04092 | case lexer::token::e_number : return false; 04093 | case lexer::token::e_symbol : return false; 04094 | case lexer::token::e_string : return false; 04095 | case lexer::token::e_add : return false; 04096 | case lexer::token::e_sub : return false; 04097 | case lexer::token::e_colon : return false; 04098 | case lexer::token::e_ternary : return false; 04099 | default : return true ; 04100 | } 04101 | } 04102 | } 04103 | else if (details::is_right_bracket(static_cast<details::char_t>(t))) 04104 | { 04105 | switch (base) 04106 | { 04107 | case lexer::token::e_number : return false; 04108 | case lexer::token::e_symbol : return false; 04109 | case lexer::token::e_string : return false; 04110 | case lexer::token::e_eof : return false; 04111 | case lexer::token::e_colon : return false; 04112 | case lexer::token::e_ternary : return false; 04113 | default : return true ; 04114 | } 04115 | } 04116 | else if (details::is_left_bracket(static_cast<details::char_t>(t))) 04117 | { 04118 | switch (base) 04119 | { 04120 | case lexer::token::e_rbracket : return true; 04121 | case lexer::token::e_rsqrbracket : return true; 04122 | case lexer::token::e_rcrlbracket : return true; 04123 | default : return false; 04124 | } 04125 | } 04126 | 04127 | return false; 04128 | } 04129 | 04130 | set_t invalid_comb_; 04131 | std::vector<std::pair<lexer::token,lexer::token> > error_list_; 04132 | }; 04133 | 04134 | class sequence_validator_3tokens exprtk_final : public lexer::token_scanner 04135 | { 04136 | private: 04137 | 04138 | typedef lexer::token::token_type token_t; 04139 | typedef std::pair<token_t,std::pair<token_t,token_t> > token_triplet_t; 04140 | typedef std::set<token_triplet_t> set_t; 04141 | 04142 | public: 04143 | 04144 | using lexer::token_scanner::operator(); 04145 | 04146 | sequence_validator_3tokens() 04147 | : lexer::token_scanner(3) 04148 | { 04149 | add_invalid(lexer::token::e_number , lexer::token::e_number , lexer::token::e_number); 04150 | add_invalid(lexer::token::e_string , lexer::token::e_string , lexer::token::e_string); 04151 | add_invalid(lexer::token::e_comma , lexer::token::e_comma , lexer::token::e_comma ); 04152 | 04153 | add_invalid(lexer::token::e_add , lexer::token::e_add , lexer::token::e_add ); 04154 | add_invalid(lexer::token::e_sub , lexer::token::e_sub , lexer::token::e_sub ); 04155 | add_invalid(lexer::token::e_div , lexer::token::e_div , lexer::token::e_div ); 04156 | add_invalid(lexer::token::e_mul , lexer::token::e_mul , lexer::token::e_mul ); 04157 | add_invalid(lexer::token::e_mod , lexer::token::e_mod , lexer::token::e_mod ); 04158 | add_invalid(lexer::token::e_pow , lexer::token::e_pow , lexer::token::e_pow ); 04159 | 04160 | add_invalid(lexer::token::e_add , lexer::token::e_sub , lexer::token::e_add ); 04161 | add_invalid(lexer::token::e_sub , lexer::token::e_add , lexer::token::e_sub ); 04162 | add_invalid(lexer::token::e_div , lexer::token::e_mul , lexer::token::e_div ); 04163 | add_invalid(lexer::token::e_mul , lexer::token::e_div , lexer::token::e_mul ); 04164 | add_invalid(lexer::token::e_mod , lexer::token::e_pow , lexer::token::e_mod ); 04165 | add_invalid(lexer::token::e_pow , lexer::token::e_mod , lexer::token::e_pow ); 04166 | } 04167 | 04168 | bool result() exprtk_override 04169 | { 04170 | return error_list_.empty(); 04171 | } 04172 | 04173 | bool operator() (const lexer::token& t0, const lexer::token& t1, const lexer::token& t2) exprtk_override 04174 | { 04175 | const set_t::value_type p = std::make_pair(t0.type,std::make_pair(t1.type,t2.type)); 04176 | 04177 | if (invalid_comb_.find(p) != invalid_comb_.end()) 04178 | { 04179 | error_list_.push_back(std::make_pair(t0,t1)); 04180 | } 04181 | 04182 | return true; 04183 | } 04184 | 04185 | std::size_t error_count() const 04186 | { 04187 | return error_list_.size(); 04188 | } 04189 | 04190 | std::pair<lexer::token,lexer::token> error(const std::size_t index) 04191 | { 04192 | if (index < error_list_.size()) 04193 | { 04194 | return error_list_[index]; 04195 | } 04196 | else 04197 | { 04198 | static const lexer::token error_token; 04199 | return std::make_pair(error_token,error_token); 04200 | } 04201 | } 04202 | 04203 | void clear_errors() 04204 | { 04205 | error_list_.clear(); 04206 | } 04207 | 04208 | private: 04209 | 04210 | void add_invalid(const token_t t0, const token_t t1, const token_t t2) 04211 | { 04212 | invalid_comb_.insert(std::make_pair(t0,std::make_pair(t1,t2))); 04213 | } 04214 | 04215 | set_t invalid_comb_; 04216 | std::vector<std::pair<lexer::token,lexer::token> > error_list_; 04217 | }; 04218 | 04219 | struct helper_assembly 04220 | { 04221 | inline bool register_scanner(lexer::token_scanner* scanner) 04222 | { 04223 | if (token_scanner_list.end() != std::find(token_scanner_list.begin(), 04224 | token_scanner_list.end (), 04225 | scanner)) 04226 | { 04227 | return false; 04228 | } 04229 | 04230 | token_scanner_list.push_back(scanner); 04231 | 04232 | return true; 04233 | } 04234 | 04235 | inline bool register_modifier(lexer::token_modifier* modifier) 04236 | { 04237 | if (token_modifier_list.end() != std::find(token_modifier_list.begin(), 04238 | token_modifier_list.end (), 04239 | modifier)) 04240 | { 04241 | return false; 04242 | } 04243 | 04244 | token_modifier_list.push_back(modifier); 04245 | 04246 | return true; 04247 | } 04248 | 04249 | inline bool register_joiner(lexer::token_joiner* joiner) 04250 | { 04251 | if (token_joiner_list.end() != std::find(token_joiner_list.begin(), 04252 | token_joiner_list.end (), 04253 | joiner)) 04254 | { 04255 | return false; 04256 | } 04257 | 04258 | token_joiner_list.push_back(joiner); 04259 | 04260 | return true; 04261 | } 04262 | 04263 | inline bool register_inserter(lexer::token_inserter* inserter) 04264 | { 04265 | if (token_inserter_list.end() != std::find(token_inserter_list.begin(), 04266 | token_inserter_list.end (), 04267 | inserter)) 04268 | { 04269 | return false; 04270 | } 04271 | 04272 | token_inserter_list.push_back(inserter); 04273 | 04274 | return true; 04275 | } 04276 | 04277 | inline bool run_modifiers(lexer::generator& g) 04278 | { 04279 | error_token_modifier = reinterpret_cast<lexer::token_modifier*>(0); 04280 | 04281 | for (std::size_t i = 0; i < token_modifier_list.size(); ++i) 04282 | { 04283 | lexer::token_modifier& modifier = (*token_modifier_list[i]); 04284 | 04285 | modifier.reset(); 04286 | modifier.process(g); 04287 | 04288 | if (!modifier.result()) 04289 | { 04290 | error_token_modifier = token_modifier_list[i]; 04291 | 04292 | return false; 04293 | } 04294 | } 04295 | 04296 | return true; 04297 | } 04298 | 04299 | inline bool run_joiners(lexer::generator& g) 04300 | { 04301 | error_token_joiner = reinterpret_cast<lexer::token_joiner*>(0); 04302 | 04303 | for (std::size_t i = 0; i < token_joiner_list.size(); ++i) 04304 | { 04305 | lexer::token_joiner& joiner = (*token_joiner_list[i]); 04306 | 04307 | joiner.reset(); 04308 | joiner.process(g); 04309 | 04310 | if (!joiner.result()) 04311 | { 04312 | error_token_joiner = token_joiner_list[i]; 04313 | 04314 | return false; 04315 | } 04316 | } 04317 | 04318 | return true; 04319 | } 04320 | 04321 | inline bool run_inserters(lexer::generator& g) 04322 | { 04323 | error_token_inserter = reinterpret_cast<lexer::token_inserter*>(0); 04324 | 04325 | for (std::size_t i = 0; i < token_inserter_list.size(); ++i) 04326 | { 04327 | lexer::token_inserter& inserter = (*token_inserter_list[i]); 04328 | 04329 | inserter.reset(); 04330 | inserter.process(g); 04331 | 04332 | if (!inserter.result()) 04333 | { 04334 | error_token_inserter = token_inserter_list[i]; 04335 | 04336 | return false; 04337 | } 04338 | } 04339 | 04340 | return true; 04341 | } 04342 | 04343 | inline bool run_scanners(lexer::generator& g) 04344 | { 04345 | error_token_scanner = reinterpret_cast<lexer::token_scanner*>(0); 04346 | 04347 | for (std::size_t i = 0; i < token_scanner_list.size(); ++i) 04348 | { 04349 | lexer::token_scanner& scanner = (*token_scanner_list[i]); 04350 | 04351 | scanner.reset(); 04352 | scanner.process(g); 04353 | 04354 | if (!scanner.result()) 04355 | { 04356 | error_token_scanner = token_scanner_list[i]; 04357 | 04358 | return false; 04359 | } 04360 | } 04361 | 04362 | return true; 04363 | } 04364 | 04365 | std::vector<lexer::token_scanner*> token_scanner_list; 04366 | std::vector<lexer::token_modifier*> token_modifier_list; 04367 | std::vector<lexer::token_joiner*> token_joiner_list; 04368 | std::vector<lexer::token_inserter*> token_inserter_list; 04369 | 04370 | lexer::token_scanner* error_token_scanner; 04371 | lexer::token_modifier* error_token_modifier; 04372 | lexer::token_joiner* error_token_joiner; 04373 | lexer::token_inserter* error_token_inserter; 04374 | }; 04375 | } 04376 | 04377 | class parser_helper 04378 | { 04379 | public: 04380 | 04381 | typedef token token_t; 04382 | typedef generator generator_t; 04383 | 04384 | inline bool init(const std::string& str) 04385 | { 04386 | if (!lexer_.process(str)) 04387 | { 04388 | return false; 04389 | } 04390 | 04391 | lexer_.begin(); 04392 | 04393 | next_token(); 04394 | 04395 | return true; 04396 | } 04397 | 04398 | inline generator_t& lexer() 04399 | { 04400 | return lexer_; 04401 | } 04402 | 04403 | inline const generator_t& lexer() const 04404 | { 04405 | return lexer_; 04406 | } 04407 | 04408 | inline void store_token() 04409 | { 04410 | lexer_.store(); 04411 | store_current_token_ = current_token_; 04412 | } 04413 | 04414 | inline void restore_token() 04415 | { 04416 | lexer_.restore(); 04417 | current_token_ = store_current_token_; 04418 | } 04419 | 04420 | inline void next_token() 04421 | { 04422 | current_token_ = lexer_.next_token(); 04423 | } 04424 | 04425 | inline const token_t& current_token() const 04426 | { 04427 | return current_token_; 04428 | } 04429 | 04430 | inline const token_t& peek_next_token() 04431 | { 04432 | return lexer_.peek_next_token(); 04433 | } 04434 | 04435 | enum token_advance_mode 04436 | { 04437 | e_hold = 0, 04438 | e_advance = 1 04439 | }; 04440 | 04441 | inline void advance_token(const token_advance_mode mode) 04442 | { 04443 | if (e_advance == mode) 04444 | { 04445 | next_token(); 04446 | } 04447 | } 04448 | 04449 | inline bool token_is(const token_t::token_type& ttype, const token_advance_mode mode = e_advance) 04450 | { 04451 | if (current_token().type != ttype) 04452 | { 04453 | return false; 04454 | } 04455 | 04456 | advance_token(mode); 04457 | 04458 | return true; 04459 | } 04460 | 04461 | inline bool token_is(const token_t::token_type& ttype, 04462 | const std::string& value, 04463 | const token_advance_mode mode = e_advance) 04464 | { 04465 | if ( 04466 | (current_token().type != ttype) || 04467 | !exprtk::details::imatch(value,current_token().value) 04468 | ) 04469 | { 04470 | return false; 04471 | } 04472 | 04473 | advance_token(mode); 04474 | 04475 | return true; 04476 | } 04477 | 04478 | inline bool token_is(const std::string& value, 04479 | const token_advance_mode mode = e_advance) 04480 | { 04481 | if (!exprtk::details::imatch(value,current_token().value)) 04482 | { 04483 | return false; 04484 | } 04485 | 04486 | advance_token(mode); 04487 | 04488 | return true; 04489 | } 04490 | 04491 | inline bool token_is_arithmetic_opr(const token_advance_mode mode = e_advance) 04492 | { 04493 | switch (current_token().type) 04494 | { 04495 | case token_t::e_add : 04496 | case token_t::e_sub : 04497 | case token_t::e_div : 04498 | case token_t::e_mul : 04499 | case token_t::e_mod : 04500 | case token_t::e_pow : break; 04501 | default : return false; 04502 | } 04503 | 04504 | advance_token(mode); 04505 | 04506 | return true; 04507 | } 04508 | 04509 | inline bool token_is_ineq_opr(const token_advance_mode mode = e_advance) 04510 | { 04511 | switch (current_token().type) 04512 | { 04513 | case token_t::e_eq : 04514 | case token_t::e_lte : 04515 | case token_t::e_ne : 04516 | case token_t::e_gte : 04517 | case token_t::e_lt : 04518 | case token_t::e_gt : break; 04519 | default : return false; 04520 | } 04521 | 04522 | advance_token(mode); 04523 | 04524 | return true; 04525 | } 04526 | 04527 | inline bool token_is_left_bracket(const token_advance_mode mode = e_advance) 04528 | { 04529 | switch (current_token().type) 04530 | { 04531 | case token_t::e_lbracket : 04532 | case token_t::e_lcrlbracket : 04533 | case token_t::e_lsqrbracket : break; 04534 | default : return false; 04535 | } 04536 | 04537 | advance_token(mode); 04538 | 04539 | return true; 04540 | } 04541 | 04542 | inline bool token_is_right_bracket(const token_advance_mode mode = e_advance) 04543 | { 04544 | switch (current_token().type) 04545 | { 04546 | case token_t::e_rbracket : 04547 | case token_t::e_rcrlbracket : 04548 | case token_t::e_rsqrbracket : break; 04549 | default : return false; 04550 | } 04551 | 04552 | advance_token(mode); 04553 | 04554 | return true; 04555 | } 04556 | 04557 | inline bool token_is_bracket(const token_advance_mode mode = e_advance) 04558 | { 04559 | switch (current_token().type) 04560 | { 04561 | case token_t::e_rbracket : 04562 | case token_t::e_rcrlbracket : 04563 | case token_t::e_rsqrbracket : 04564 | case token_t::e_lbracket : 04565 | case token_t::e_lcrlbracket : 04566 | case token_t::e_lsqrbracket : break; 04567 | default : return false; 04568 | } 04569 | 04570 | advance_token(mode); 04571 | 04572 | return true; 04573 | } 04574 | 04575 | inline bool token_is_loop(const token_advance_mode mode = e_advance) 04576 | { 04577 | return token_is("for" , mode) || 04578 | token_is("while" , mode) || 04579 | token_is("repeat", mode) ; 04580 | } 04581 | 04582 | inline bool peek_token_is(const token_t::token_type& ttype) 04583 | { 04584 | return (lexer_.peek_next_token().type == ttype); 04585 | } 04586 | 04587 | inline bool peek_token_is(const std::string& s) 04588 | { 04589 | return (exprtk::details::imatch(lexer_.peek_next_token().value,s)); 04590 | } 04591 | 04592 | private: 04593 | 04594 | generator_t lexer_; 04595 | token_t current_token_; 04596 | token_t store_current_token_; 04597 | }; 04598 | } 04599 | 04600 | template <typename T> 04601 | class vector_view 04602 | { 04603 | public: 04604 | 04605 | typedef T* data_ptr_t; 04606 | 04607 | vector_view(data_ptr_t data, const std::size_t& size) 04608 | : base_size_(size) 04609 | , size_(size) 04610 | , data_(data) 04611 | , data_ref_(0) 04612 | { 04613 | assert(size_ > 0); 04614 | } 04615 | 04616 | vector_view(const vector_view<T>& vv) 04617 | : base_size_(vv.base_size_) 04618 | , size_(vv.size_) 04619 | , data_(vv.data_) 04620 | , data_ref_(0) 04621 | { 04622 | assert(size_ > 0); 04623 | } 04624 | 04625 | inline void rebase(data_ptr_t data) 04626 | { 04627 | data_ = data; 04628 | 04629 | if (!data_ref_.empty()) 04630 | { 04631 | for (std::size_t i = 0; i < data_ref_.size(); ++i) 04632 | { 04633 | (*data_ref_[i]) = data; 04634 | } 04635 | } 04636 | } 04637 | 04638 | inline data_ptr_t data() const 04639 | { 04640 | return data_; 04641 | } 04642 | 04643 | inline std::size_t base_size() const 04644 | { 04645 | return base_size_; 04646 | } 04647 | 04648 | inline std::size_t size() const 04649 | { 04650 | return size_; 04651 | } 04652 | 04653 | inline const T& operator[](const std::size_t index) const 04654 | { 04655 | assert(index < size_); 04656 | return data_[index]; 04657 | } 04658 | 04659 | inline T& operator[](const std::size_t index) 04660 | { 04661 | assert(index < size_); 04662 | return data_[index]; 04663 | } 04664 | 04665 | void set_ref(data_ptr_t* data_ref) 04666 | { 04667 | data_ref_.push_back(data_ref); 04668 | exprtk_debug(("vector_view::set_ref() - data_ref: %p data_ref_.size(): %d\n", 04669 | reinterpret_cast<void*>(data_ref), 04670 | static_cast<int>(data_ref_.size()))); 04671 | } 04672 | 04673 | void remove_ref(data_ptr_t* data_ref) 04674 | { 04675 | data_ref_.erase( 04676 | std::remove(data_ref_.begin(), data_ref_.end(), data_ref), 04677 | data_ref_.end()); 04678 | exprtk_debug(("vector_view::remove_ref() - data_ref: %p data_ref_.size(): %d\n", 04679 | reinterpret_cast<void*>(data_ref), 04680 | static_cast<int>(data_ref_.size()))); 04681 | } 04682 | 04683 | bool set_size(const std::size_t new_size) 04684 | { 04685 | if ((new_size > 0) && (new_size <= base_size_)) 04686 | { 04687 | size_ = new_size; 04688 | exprtk_debug(("vector_view::set_size() - data_: %p size: %lu\n", 04689 | reinterpret_cast<void*>(data_), 04690 | size_)); 04691 | return true; 04692 | } 04693 | 04694 | exprtk_debug(("vector_view::set_size() - error invalid new_size: %lu base_size: %lu\n", 04695 | new_size, 04696 | base_size_)); 04697 | return false; 04698 | } 04699 | 04700 | private: 04701 | 04702 | const std::size_t base_size_; 04703 | std::size_t size_; 04704 | data_ptr_t data_; 04705 | std::vector<data_ptr_t*> data_ref_; 04706 | }; 04707 | 04708 | template <typename T> 04709 | inline vector_view<T> make_vector_view(T* data, 04710 | const std::size_t size, const std::size_t offset = 0) 04711 | { 04712 | return vector_view<T>(data + offset, size); 04713 | } 04714 | 04715 | template <typename T> 04716 | inline vector_view<T> make_vector_view(std::vector<T>& v, 04717 | const std::size_t size, const std::size_t offset = 0) 04718 | { 04719 | return vector_view<T>(v.data() + offset, size); 04720 | } 04721 | 04722 | template <typename T> class results_context; 04723 | 04724 | template <typename T> 04725 | struct type_store 04726 | { 04727 | enum store_type 04728 | { 04729 | e_unknown, 04730 | e_scalar , 04731 | e_vector , 04732 | e_string 04733 | }; 04734 | 04735 | type_store() 04736 | : data(0) 04737 | , size(0) 04738 | , type(e_unknown) 04739 | {} 04740 | 04741 | union 04742 | { 04743 | void* data; 04744 | T* vec_data; 04745 | }; 04746 | 04747 | std::size_t size; 04748 | store_type type; 04749 | 04750 | class parameter_list 04751 | { 04752 | public: 04753 | 04754 | explicit parameter_list(std::vector<type_store>& pl) 04755 | : parameter_list_(pl) 04756 | {} 04757 | 04758 | inline bool empty() const 04759 | { 04760 | return parameter_list_.empty(); 04761 | } 04762 | 04763 | inline std::size_t size() const 04764 | { 04765 | return parameter_list_.size(); 04766 | } 04767 | 04768 | inline type_store& operator[](const std::size_t& index) 04769 | { 04770 | return parameter_list_[index]; 04771 | } 04772 | 04773 | inline const type_store& operator[](const std::size_t& index) const 04774 | { 04775 | return parameter_list_[index]; 04776 | } 04777 | 04778 | inline type_store& front() 04779 | { 04780 | return parameter_list_[0]; 04781 | } 04782 | 04783 | inline const type_store& front() const 04784 | { 04785 | return parameter_list_[0]; 04786 | } 04787 | 04788 | inline type_store& back() 04789 | { 04790 | return parameter_list_.back(); 04791 | } 04792 | 04793 | inline const type_store& back() const 04794 | { 04795 | return parameter_list_.back(); 04796 | } 04797 | 04798 | private: 04799 | 04800 | std::vector<type_store>& parameter_list_; 04801 | 04802 | friend class results_context<T>; 04803 | }; 04804 | 04805 | template <typename ViewType> 04806 | struct type_view 04807 | { 04808 | typedef type_store<T> type_store_t; 04809 | typedef ViewType value_t; 04810 | 04811 | explicit type_view(type_store_t& ts) 04812 | : ts_(ts) 04813 | , data_(reinterpret_cast<value_t*>(ts_.data)) 04814 | {} 04815 | 04816 | explicit type_view(const type_store_t& ts) 04817 | : ts_(const_cast<type_store_t&>(ts)) 04818 | , data_(reinterpret_cast<value_t*>(ts_.data)) 04819 | {} 04820 | 04821 | inline std::size_t size() const 04822 | { 04823 | return ts_.size; 04824 | } 04825 | 04826 | inline value_t& operator[](const std::size_t& i) 04827 | { 04828 | return data_[i]; 04829 | } 04830 | 04831 | inline const value_t& operator[](const std::size_t& i) const 04832 | { 04833 | return data_[i]; 04834 | } 04835 | 04836 | inline const value_t* begin() const { return data_; } 04837 | inline value_t* begin() { return data_; } 04838 | 04839 | inline const value_t* end() const 04840 | { 04841 | return static_cast<value_t*>(data_ + ts_.size); 04842 | } 04843 | 04844 | inline value_t* end() 04845 | { 04846 | return static_cast<value_t*>(data_ + ts_.size); 04847 | } 04848 | 04849 | type_store_t& ts_; 04850 | value_t* data_; 04851 | }; 04852 | 04853 | typedef type_view<T> vector_view; 04854 | typedef type_view<char> string_view; 04855 | 04856 | struct scalar_view 04857 | { 04858 | typedef type_store<T> type_store_t; 04859 | typedef T value_t; 04860 | 04861 | explicit scalar_view(type_store_t& ts) 04862 | : v_(*reinterpret_cast<value_t*>(ts.data)) 04863 | {} 04864 | 04865 | explicit scalar_view(const type_store_t& ts) 04866 | : v_(*reinterpret_cast<value_t*>(const_cast<type_store_t&>(ts).data)) 04867 | {} 04868 | 04869 | inline value_t& operator() () 04870 | { 04871 | return v_; 04872 | } 04873 | 04874 | inline const value_t& operator() () const 04875 | { 04876 | return v_; 04877 | } 04878 | 04879 | inline operator value_t() const 04880 | { 04881 | return v_; 04882 | } 04883 | 04884 | inline operator value_t() 04885 | { 04886 | return v_; 04887 | } 04888 | 04889 | template <typename IntType> 04890 | inline bool to_int(IntType& i) const 04891 | { 04892 | if (!exprtk::details::numeric::is_integer(v_)) 04893 | return false; 04894 | 04895 | i = static_cast<IntType>(v_); 04896 | 04897 | return true; 04898 | } 04899 | 04900 | template <typename UIntType> 04901 | inline bool to_uint(UIntType& u) const 04902 | { 04903 | if (v_ < T(0)) 04904 | return false; 04905 | else if (!exprtk::details::numeric::is_integer(v_)) 04906 | return false; 04907 | 04908 | u = static_cast<UIntType>(v_); 04909 | 04910 | return true; 04911 | } 04912 | 04913 | T& v_; 04914 | }; 04915 | }; 04916 | 04917 | template <typename StringView> 04918 | inline std::string to_str(const StringView& view) 04919 | { 04920 | return std::string(view.begin(),view.size()); 04921 | } 04922 | 04923 | #ifndef exprtk_disable_return_statement 04924 | namespace details 04925 | { 04926 | template <typename T> class return_node; 04927 | template <typename T> class return_envelope_node; 04928 | } 04929 | #endif 04930 | 04931 | template <typename T> 04932 | class results_context 04933 | { 04934 | public: 04935 | 04936 | typedef type_store<T> type_store_t; 04937 | typedef typename type_store_t::scalar_view scalar_t; 04938 | typedef typename type_store_t::vector_view vector_t; 04939 | typedef typename type_store_t::string_view string_t; 04940 | 04941 | results_context() 04942 | : results_available_(false) 04943 | {} 04944 | 04945 | inline std::size_t count() const 04946 | { 04947 | if (results_available_) 04948 | return parameter_list_.size(); 04949 | else 04950 | return 0; 04951 | } 04952 | 04953 | inline type_store_t& operator[](const std::size_t& index) 04954 | { 04955 | return parameter_list_[index]; 04956 | } 04957 | 04958 | inline const type_store_t& operator[](const std::size_t& index) const 04959 | { 04960 | return parameter_list_[index]; 04961 | } 04962 | 04963 | inline bool get_scalar(const std::size_t& index, T& out) const 04964 | { 04965 | if ( 04966 | (index < parameter_list_.size()) && 04967 | (parameter_list_[index].type == type_store_t::e_scalar) 04968 | ) 04969 | { 04970 | const scalar_t scalar(parameter_list_[index]); 04971 | out = scalar(); 04972 | return true; 04973 | } 04974 | 04975 | return false; 04976 | } 04977 | 04978 | template <typename OutputIterator> 04979 | inline bool get_vector(const std::size_t& index, OutputIterator out_itr) const 04980 | { 04981 | if ( 04982 | (index < parameter_list_.size()) && 04983 | (parameter_list_[index].type == type_store_t::e_vector) 04984 | ) 04985 | { 04986 | const vector_t vector(parameter_list_[index]); 04987 | for (std::size_t i = 0; i < vector.size(); ++i) 04988 | { 04989 | *(out_itr++) = vector[i]; 04990 | } 04991 | 04992 | return true; 04993 | } 04994 | 04995 | return false; 04996 | } 04997 | 04998 | inline bool get_vector(const std::size_t& index, std::vector<T>& out) const 04999 | { 05000 | return get_vector(index,std::back_inserter(out)); 05001 | } 05002 | 05003 | inline bool get_string(const std::size_t& index, std::string& out) const 05004 | { 05005 | if ( 05006 | (index < parameter_list_.size()) && 05007 | (parameter_list_[index].type == type_store_t::e_string) 05008 | ) 05009 | { 05010 | const string_t str(parameter_list_[index]); 05011 | out.assign(str.begin(),str.size()); 05012 | return true; 05013 | } 05014 | 05015 | return false; 05016 | } 05017 | 05018 | private: 05019 | 05020 | inline void clear() 05021 | { 05022 | results_available_ = false; 05023 | } 05024 | 05025 | typedef std::vector<type_store_t> ts_list_t; 05026 | typedef typename type_store_t::parameter_list parameter_list_t; 05027 | 05028 | inline void assign(const parameter_list_t& pl) 05029 | { 05030 | parameter_list_ = pl.parameter_list_; 05031 | results_available_ = true; 05032 | } 05033 | 05034 | bool results_available_; 05035 | ts_list_t parameter_list_; 05036 | 05037 | #ifndef exprtk_disable_return_statement 05038 | friend class details::return_node<T>; 05039 | friend class details::return_envelope_node<T>; 05040 | #endif 05041 | }; 05042 | 05043 | namespace details 05044 | { 05045 | enum operator_type 05046 | { 05047 | e_default , e_null , e_add , e_sub , 05048 | e_mul , e_div , e_mod , e_pow , 05049 | e_atan2 , e_min , e_max , e_avg , 05050 | e_sum , e_prod , e_lt , e_lte , 05051 | e_eq , e_equal , e_ne , e_nequal , 05052 | e_gte , e_gt , e_and , e_nand , 05053 | e_or , e_nor , e_xor , e_xnor , 05054 | e_mand , e_mor , e_scand , e_scor , 05055 | e_shr , e_shl , e_abs , e_acos , 05056 | e_acosh , e_asin , e_asinh , e_atan , 05057 | e_atanh , e_ceil , e_cos , e_cosh , 05058 | e_exp , e_expm1 , e_floor , e_log , 05059 | e_log10 , e_log2 , e_log1p , e_logn , 05060 | e_neg , e_pos , e_round , e_roundn , 05061 | e_root , e_sqrt , e_sin , e_sinc , 05062 | e_sinh , e_sec , e_csc , e_tan , 05063 | e_tanh , e_cot , e_clamp , e_iclamp , 05064 | e_inrange , e_sgn , e_r2d , e_d2r , 05065 | e_d2g , e_g2d , e_hypot , e_notl , 05066 | e_erf , e_erfc , e_ncdf , e_frac , 05067 | e_trunc , e_assign , e_addass , e_subass , 05068 | e_mulass , e_divass , e_modass , e_in , 05069 | e_like , e_ilike , e_multi , e_smulti , 05070 | e_swap , 05071 | 05072 | // Do not add new functions/operators after this point. 05073 | e_sf00 = 1000, e_sf01 = 1001, e_sf02 = 1002, e_sf03 = 1003, 05074 | e_sf04 = 1004, e_sf05 = 1005, e_sf06 = 1006, e_sf07 = 1007, 05075 | e_sf08 = 1008, e_sf09 = 1009, e_sf10 = 1010, e_sf11 = 1011, 05076 | e_sf12 = 1012, e_sf13 = 1013, e_sf14 = 1014, e_sf15 = 1015, 05077 | e_sf16 = 1016, e_sf17 = 1017, e_sf18 = 1018, e_sf19 = 1019, 05078 | e_sf20 = 1020, e_sf21 = 1021, e_sf22 = 1022, e_sf23 = 1023, 05079 | e_sf24 = 1024, e_sf25 = 1025, e_sf26 = 1026, e_sf27 = 1027, 05080 | e_sf28 = 1028, e_sf29 = 1029, e_sf30 = 1030, e_sf31 = 1031, 05081 | e_sf32 = 1032, e_sf33 = 1033, e_sf34 = 1034, e_sf35 = 1035, 05082 | e_sf36 = 1036, e_sf37 = 1037, e_sf38 = 1038, e_sf39 = 1039, 05083 | e_sf40 = 1040, e_sf41 = 1041, e_sf42 = 1042, e_sf43 = 1043, 05084 | e_sf44 = 1044, e_sf45 = 1045, e_sf46 = 1046, e_sf47 = 1047, 05085 | e_sf48 = 1048, e_sf49 = 1049, e_sf50 = 1050, e_sf51 = 1051, 05086 | e_sf52 = 1052, e_sf53 = 1053, e_sf54 = 1054, e_sf55 = 1055, 05087 | e_sf56 = 1056, e_sf57 = 1057, e_sf58 = 1058, e_sf59 = 1059, 05088 | e_sf60 = 1060, e_sf61 = 1061, e_sf62 = 1062, e_sf63 = 1063, 05089 | e_sf64 = 1064, e_sf65 = 1065, e_sf66 = 1066, e_sf67 = 1067, 05090 | e_sf68 = 1068, e_sf69 = 1069, e_sf70 = 1070, e_sf71 = 1071, 05091 | e_sf72 = 1072, e_sf73 = 1073, e_sf74 = 1074, e_sf75 = 1075, 05092 | e_sf76 = 1076, e_sf77 = 1077, e_sf78 = 1078, e_sf79 = 1079, 05093 | e_sf80 = 1080, e_sf81 = 1081, e_sf82 = 1082, e_sf83 = 1083, 05094 | e_sf84 = 1084, e_sf85 = 1085, e_sf86 = 1086, e_sf87 = 1087, 05095 | e_sf88 = 1088, e_sf89 = 1089, e_sf90 = 1090, e_sf91 = 1091, 05096 | e_sf92 = 1092, e_sf93 = 1093, e_sf94 = 1094, e_sf95 = 1095, 05097 | e_sf96 = 1096, e_sf97 = 1097, e_sf98 = 1098, e_sf99 = 1099, 05098 | e_sffinal = 1100, 05099 | e_sf4ext00 = 2000, e_sf4ext01 = 2001, e_sf4ext02 = 2002, e_sf4ext03 = 2003, 05100 | e_sf4ext04 = 2004, e_sf4ext05 = 2005, e_sf4ext06 = 2006, e_sf4ext07 = 2007, 05101 | e_sf4ext08 = 2008, e_sf4ext09 = 2009, e_sf4ext10 = 2010, e_sf4ext11 = 2011, 05102 | e_sf4ext12 = 2012, e_sf4ext13 = 2013, e_sf4ext14 = 2014, e_sf4ext15 = 2015, 05103 | e_sf4ext16 = 2016, e_sf4ext17 = 2017, e_sf4ext18 = 2018, e_sf4ext19 = 2019, 05104 | e_sf4ext20 = 2020, e_sf4ext21 = 2021, e_sf4ext22 = 2022, e_sf4ext23 = 2023, 05105 | e_sf4ext24 = 2024, e_sf4ext25 = 2025, e_sf4ext26 = 2026, e_sf4ext27 = 2027, 05106 | e_sf4ext28 = 2028, e_sf4ext29 = 2029, e_sf4ext30 = 2030, e_sf4ext31 = 2031, 05107 | e_sf4ext32 = 2032, e_sf4ext33 = 2033, e_sf4ext34 = 2034, e_sf4ext35 = 2035, 05108 | e_sf4ext36 = 2036, e_sf4ext37 = 2037, e_sf4ext38 = 2038, e_sf4ext39 = 2039, 05109 | e_sf4ext40 = 2040, e_sf4ext41 = 2041, e_sf4ext42 = 2042, e_sf4ext43 = 2043, 05110 | e_sf4ext44 = 2044, e_sf4ext45 = 2045, e_sf4ext46 = 2046, e_sf4ext47 = 2047, 05111 | e_sf4ext48 = 2048, e_sf4ext49 = 2049, e_sf4ext50 = 2050, e_sf4ext51 = 2051, 05112 | e_sf4ext52 = 2052, e_sf4ext53 = 2053, e_sf4ext54 = 2054, e_sf4ext55 = 2055, 05113 | e_sf4ext56 = 2056, e_sf4ext57 = 2057, e_sf4ext58 = 2058, e_sf4ext59 = 2059, 05114 | e_sf4ext60 = 2060, e_sf4ext61 = 2061 05115 | }; 05116 | 05117 | inline std::string to_str(const operator_type opr) 05118 | { 05119 | switch (opr) 05120 | { 05121 | case e_add : return "+" ; 05122 | case e_sub : return "-" ; 05123 | case e_mul : return "*" ; 05124 | case e_div : return "/" ; 05125 | case e_mod : return "%" ; 05126 | case e_pow : return "^" ; 05127 | case e_assign : return ":=" ; 05128 | case e_addass : return "+=" ; 05129 | case e_subass : return "-=" ; 05130 | case e_mulass : return "*=" ; 05131 | case e_divass : return "/=" ; 05132 | case e_modass : return "%=" ; 05133 | case e_lt : return "<" ; 05134 | case e_lte : return "<=" ; 05135 | case e_eq : return "==" ; 05136 | case e_equal : return "=" ; 05137 | case e_ne : return "!=" ; 05138 | case e_nequal : return "<>" ; 05139 | case e_gte : return ">=" ; 05140 | case e_gt : return ">" ; 05141 | case e_and : return "and" ; 05142 | case e_or : return "or" ; 05143 | case e_xor : return "xor" ; 05144 | case e_nand : return "nand" 05145 | case e_nor : return "nor" ; 05146 | case e_xnor : return "xnor" 05147 | default : return "N/A" ; 05148 | } 05149 | } 05150 | 05151 | struct base_operation_t 05152 | { 05153 | base_operation_t(const operator_type t, const unsigned int& np) 05154 | : type(t) 05155 | , num_params(np) 05156 | {} 05157 | 05158 | operator_type type; 05159 | unsigned int num_params; 05160 | }; 05161 | 05162 | namespace loop_unroll 05163 | { 05164 | const unsigned int global_loop_batch_size = 05165 | #ifndef exprtk_disable_superscalar_unroll 05166 | 16; 05167 | #else 05168 | 4; 05169 | #endif 05170 | 05171 | struct details 05172 | { 05173 | explicit details(const std::size_t& vsize, 05174 | const unsigned int loop_batch_size = global_loop_batch_size) 05175 | : batch_size(loop_batch_size ) 05176 | , remainder (vsize % batch_size) 05177 | , upper_bound(static_cast<int>(vsize - (remainder ? loop_batch_size : 0))) 05178 | {} 05179 | 05180 | unsigned int batch_size; 05181 | int remainder; 05182 | int upper_bound; 05183 | }; 05184 | } 05185 | 05186 | #ifdef exprtk_enable_debugging 05187 | inline void dump_ptr(const std::string& s, const void* ptr, const std::size_t size = 0) 05188 | { 05189 | if (size) 05190 | exprtk_debug(("%s - addr: %p size: %d\n", 05191 | s.c_str(), 05192 | ptr, 05193 | static_cast<unsigned int>(size))); 05194 | else 05195 | exprtk_debug(("%s - addr: %p\n", s.c_str(), ptr)); 05196 | } 05197 | 05198 | template <typename T> 05199 | inline void dump_vector(const std::string& vec_name, const T* data, const std::size_t size) 05200 | { 05201 | printf("----- %s (%p) -----\n", 05202 | vec_name.c_str(), 05203 | static_cast<const void*>(data)); 05204 | printf("[ "); 05205 | for (std::size_t i = 0; i < size; ++i) 05206 | { 05207 | printf("%8.3f\t", data[i]); 05208 | } 05209 | printf(" ]\n"); 05210 | printf("---------------------\n"); 05211 | } 05212 | #else 05213 | inline void dump_ptr(const std::string&, const void*) {} 05214 | inline void dump_ptr(const std::string&, const void*, const std::size_t) {} 05215 | template <typename T> 05216 | inline void dump_vector(const std::string&, const T*, const std::size_t) {} 05217 | #endif 05218 | 05219 | template <typename T> 05220 | class vec_data_store 05221 | { 05222 | public: 05223 | 05224 | typedef vec_data_store<T> type; 05225 | typedef T* data_t; 05226 | 05227 | private: 05228 | 05229 | struct control_block 05230 | { 05231 | control_block() 05232 | : ref_count(1) 05233 | , size (0) 05234 | , data (0) 05235 | , destruct (true) 05236 | {} 05237 | 05238 | explicit control_block(const std::size_t& dsize) 05239 | : ref_count(1 ) 05240 | , size (dsize) 05241 | , data (0 ) 05242 | , destruct (true ) 05243 | { create_data(); } 05244 | 05245 | control_block(const std::size_t& dsize, data_t dptr, bool dstrct = false) 05246 | : ref_count(1 ) 05247 | , size (dsize ) 05248 | , data (dptr ) 05249 | , destruct (dstrct) 05250 | {} 05251 | 05252 | ~control_block() 05253 | { 05254 | if (data && destruct && (0 == ref_count)) 05255 | { 05256 | dump_ptr("~vec_data_store::control_block() data",data); 05257 | delete[] data; 05258 | data = reinterpret_cast<data_t>(0); 05259 | } 05260 | } 05261 | 05262 | static inline control_block* create(const std::size_t& dsize, data_t data_ptr = data_t(0), bool dstrct = false) 05263 | { 05264 | if (dsize) 05265 | { 05266 | if (0 == data_ptr) 05267 | return (new control_block(dsize)); 05268 | else 05269 | return (new control_block(dsize, data_ptr, dstrct)); 05270 | } 05271 | else 05272 | return (new control_block); 05273 | } 05274 | 05275 | static inline void destroy(control_block*& cntrl_blck) 05276 | { 05277 | if (cntrl_blck) 05278 | { 05279 | if ( 05280 | (0 != cntrl_blck->ref_count) && 05281 | (0 == --cntrl_blck->ref_count) 05282 | ) 05283 | { 05284 | delete cntrl_blck; 05285 | } 05286 | 05287 | cntrl_blck = 0; 05288 | } 05289 | } 05290 | 05291 | std::size_t ref_count; 05292 | std::size_t size; 05293 | data_t data; 05294 | bool destruct; 05295 | 05296 | private: 05297 | 05298 | control_block(const control_block&) exprtk_delete; 05299 | control_block& operator=(const control_block&) exprtk_delete; 05300 | 05301 | inline void create_data() 05302 | { 05303 | destruct = true; 05304 | data = new T[size]; 05305 | std::fill_n(data, size, T(0)); 05306 | dump_ptr("control_block::create_data() - data", data, size); 05307 | } 05308 | }; 05309 | 05310 | public: 05311 | 05312 | vec_data_store() 05313 | : control_block_(control_block::create(0)) 05314 | {} 05315 | 05316 | explicit vec_data_store(const std::size_t& size) 05317 | : control_block_(control_block::create(size,reinterpret_cast<data_t>(0),true)) 05318 | {} 05319 | 05320 | vec_data_store(const std::size_t& size, data_t data, bool dstrct = false) 05321 | : control_block_(control_block::create(size, data, dstrct)) 05322 | {} 05323 | 05324 | vec_data_store(const type& vds) 05325 | { 05326 | control_block_ = vds.control_block_; 05327 | control_block_->ref_count++; 05328 | } 05329 | 05330 | ~vec_data_store() 05331 | { 05332 | control_block::destroy(control_block_); 05333 | } 05334 | 05335 | type& operator=(const type& vds) 05336 | { 05337 | if (this != &vds) 05338 | { 05339 | const std::size_t final_size = min_size(control_block_, vds.control_block_); 05340 | 05341 | vds.control_block_->size = final_size; 05342 | control_block_->size = final_size; 05343 | 05344 | if (control_block_->destruct || (0 == control_block_->data)) 05345 | { 05346 | control_block::destroy(control_block_); 05347 | 05348 | control_block_ = vds.control_block_; 05349 | control_block_->ref_count++; 05350 | } 05351 | } 05352 | 05353 | return (*this); 05354 | } 05355 | 05356 | inline data_t data() 05357 | { 05358 | return control_block_->data; 05359 | } 05360 | 05361 | inline data_t data() const 05362 | { 05363 | return control_block_->data; 05364 | } 05365 | 05366 | inline std::size_t size() const 05367 | { 05368 | return control_block_->size; 05369 | } 05370 | 05371 | inline data_t& ref() 05372 | { 05373 | return control_block_->data; 05374 | } 05375 | 05376 | inline void dump() const 05377 | { 05378 | #ifdef exprtk_enable_debugging 05379 | exprtk_debug(("size: %d\taddress:%p\tdestruct:%c\n", 05380 | size(), 05381 | data(), 05382 | (control_block_->destruct ? 'T' : 'F'))); 05383 | 05384 | for (std::size_t i = 0; i < size(); ++i) 05385 | { 05386 | if (5 == i) 05387 | exprtk_debug(("\n")); 05388 | 05389 | exprtk_debug(("%15.10f ", data()[i])); 05390 | } 05391 | exprtk_debug(("\n")); 05392 | #endif 05393 | } 05394 | 05395 | static inline void match_sizes(type& vds0, type& vds1) 05396 | { 05397 | const std::size_t size = min_size(vds0.control_block_,vds1.control_block_); 05398 | vds0.control_block_->size = size; 05399 | vds1.control_block_->size = size; 05400 | } 05401 | 05402 | private: 05403 | 05404 | static inline std::size_t min_size(const control_block* cb0, const control_block* cb1) 05405 | { 05406 | const std::size_t size0 = cb0->size; 05407 | const std::size_t size1 = cb1->size; 05408 | 05409 | if (size0 && size1) 05410 | return std::min(size0,size1); 05411 | else 05412 | return (size0) ? size0 : size1; 05413 | } 05414 | 05415 | control_block* control_block_; 05416 | }; 05417 | 05418 | namespace numeric 05419 | { 05420 | namespace details 05421 | { 05422 | template <typename T> 05423 | inline T process_impl(const operator_type operation, const T arg) 05424 | { 05425 | switch (operation) 05426 | { 05427 | case e_abs : return numeric::abs (arg); 05428 | case e_acos : return numeric::acos (arg); 05429 | case e_acosh : return numeric::acosh(arg); 05430 | case e_asin : return numeric::asin (arg); 05431 | case e_asinh : return numeric::asinh(arg); 05432 | case e_atan : return numeric::atan (arg); 05433 | case e_atanh : return numeric::atanh(arg); 05434 | case e_ceil : return numeric::ceil (arg); 05435 | case e_cos : return numeric::cos (arg); 05436 | case e_cosh : return numeric::cosh (arg); 05437 | case e_exp : return numeric::exp (arg); 05438 | case e_expm1 : return numeric::expm1(arg); 05439 | case e_floor : return numeric::floor(arg); 05440 | case e_log : return numeric::log (arg); 05441 | case e_log10 : return numeric::log10(arg); 05442 | case e_log2 : return numeric::log2 (arg); 05443 | case e_log1p : return numeric::log1p(arg); 05444 | case e_neg : return numeric::neg (arg); 05445 | case e_pos : return numeric::pos (arg); 05446 | case e_round : return numeric::round(arg); 05447 | case e_sin : return numeric::sin (arg); 05448 | case e_sinc : return numeric::sinc (arg); 05449 | case e_sinh : return numeric::sinh (arg); 05450 | case e_sqrt : return numeric::sqrt (arg); 05451 | case e_tan : return numeric::tan (arg); 05452 | case e_tanh : return numeric::tanh (arg); 05453 | case e_cot : return numeric::cot (arg); 05454 | case e_sec : return numeric::sec (arg); 05455 | case e_csc : return numeric::csc (arg); 05456 | case e_r2d : return numeric::r2d (arg); 05457 | case e_d2r : return numeric::d2r (arg); 05458 | case e_d2g : return numeric::d2g (arg); 05459 | case e_g2d : return numeric::g2d (arg); 05460 | case e_notl : return numeric::notl (arg); 05461 | case e_sgn : return numeric::sgn (arg); 05462 | case e_erf : return numeric::erf (arg); 05463 | case e_erfc : return numeric::erfc (arg); 05464 | case e_ncdf : return numeric::ncdf (arg); 05465 | case e_frac : return numeric::frac (arg); 05466 | case e_trunc : return numeric::trunc(arg); 05467 | 05468 | default : exprtk_debug(("numeric::details::process_impl<T> - Invalid unary operation.\n")); 05469 | return std::numeric_limits<T>::quiet_NaN(); 05470 | } 05471 | } 05472 | 05473 | template <typename T> 05474 | inline T process_impl(const operator_type operation, const T arg0, const T arg1) 05475 | { 05476 | switch (operation) 05477 | { 05478 | case e_add : return (arg0 + arg1); 05479 | case e_sub : return (arg0 - arg1); 05480 | case e_mul : return (arg0 * arg1); 05481 | case e_div : return (arg0 / arg1); 05482 | case e_mod : return modulus<T>(arg0,arg1); 05483 | case e_pow : return pow<T>(arg0,arg1); 05484 | case e_atan2 : return atan2<T>(arg0,arg1); 05485 | case e_min : return std::min<T>(arg0,arg1); 05486 | case e_max : return std::max<T>(arg0,arg1); 05487 | case e_logn : return logn<T>(arg0,arg1); 05488 | case e_lt : return (arg0 < arg1) ? T(1) : T(0); 05489 | case e_lte : return (arg0 <= arg1) ? T(1) : T(0); 05490 | case e_eq : return std::equal_to<T>()(arg0,arg1) ? T(1) : T(0); 05491 | case e_ne : return std::not_equal_to<T>()(arg0,arg1) ? T(1) : T(0); 05492 | case e_gte : return (arg0 >= arg1) ? T(1) : T(0); 05493 | case e_gt : return (arg0 > arg1) ? T(1) : T(0); 05494 | case e_and : return and_opr <T>(arg0,arg1); 05495 | case e_nand : return nand_opr<T>(arg0,arg1); 05496 | case e_or : return or_opr <T>(arg0,arg1); 05497 | case e_nor : return nor_opr <T>(arg0,arg1); 05498 | case e_xor : return xor_opr <T>(arg0,arg1); 05499 | case e_xnor : return xnor_opr<T>(arg0,arg1); 05500 | case e_root : return root <T>(arg0,arg1); 05501 | case e_roundn : return roundn <T>(arg0,arg1); 05502 | case e_equal : return equal <T>(arg0,arg1); 05503 | case e_nequal : return nequal <T>(arg0,arg1); 05504 | case e_hypot : return hypot <T>(arg0,arg1); 05505 | case e_shr : return shr <T>(arg0,arg1); 05506 | case e_shl : return shl <T>(arg0,arg1); 05507 | 05508 | default : exprtk_debug(("numeric::details::process_impl<T> - Invalid binary operation.\n")); 05509 | return std::numeric_limits<T>::quiet_NaN(); 05510 | } 05511 | } 05512 | 05513 | template <typename T> 05514 | inline T process_impl(const operator_type operation, const T arg0, const T arg1, int_type_tag) 05515 | { 05516 | switch (operation) 05517 | { 05518 | case e_add : return (arg0 + arg1); 05519 | case e_sub : return (arg0 - arg1); 05520 | case e_mul : return (arg0 * arg1); 05521 | case e_div : return (arg0 / arg1); 05522 | case e_mod : return arg0 % arg1; 05523 | case e_pow : return pow<T>(arg0,arg1); 05524 | case e_min : return std::min<T>(arg0,arg1); 05525 | case e_max : return std::max<T>(arg0,arg1); 05526 | case e_logn : return logn<T>(arg0,arg1); 05527 | case e_lt : return (arg0 < arg1) ? T(1) : T(0); 05528 | case e_lte : return (arg0 <= arg1) ? T(1) : T(0); 05529 | case e_eq : return (arg0 == arg1) ? T(1) : T(0); 05530 | case e_ne : return (arg0 != arg1) ? T(1) : T(0); 05531 | case e_gte : return (arg0 >= arg1) ? T(1) : T(0); 05532 | case e_gt : return (arg0 > arg1) ? T(1) : T(0); 05533 | case e_and : return ((arg0 != T(0)) && (arg1 != T(0))) ? T(1) : T(0); 05534 | case e_nand : return ((arg0 != T(0)) && (arg1 != T(0))) ? T(0) : T(1); 05535 | case e_or : return ((arg0 != T(0)) || (arg1 != T(0))) ? T(1) : T(0); 05536 | case e_nor : return ((arg0 != T(0)) || (arg1 != T(0))) ? T(0) : T(1); 05537 | case e_xor : return arg0 ^ arg1; 05538 | case e_xnor : return !(arg0 ^ arg1); 05539 | case e_root : return root<T>(arg0,arg1); 05540 | case e_equal : return arg0 == arg1; 05541 | case e_nequal : return arg0 != arg1; 05542 | case e_hypot : return hypot<T>(arg0,arg1); 05543 | case e_shr : return arg0 >> arg1; 05544 | case e_shl : return arg0 << arg1; 05545 | 05546 | default : exprtk_debug(("numeric::details::process_impl<IntType> - Invalid binary operation.\n")); 05547 | return std::numeric_limits<T>::quiet_NaN(); 05548 | } 05549 | } 05550 | } 05551 | 05552 | template <typename T> 05553 | inline T process(const operator_type operation, const T arg) 05554 | { 05555 | return exprtk::details::numeric::details::process_impl(operation,arg); 05556 | } 05557 | 05558 | template <typename T> 05559 | inline T process(const operator_type operation, const T arg0, const T arg1) 05560 | { 05561 | return exprtk::details::numeric::details::process_impl(operation, arg0, arg1); 05562 | } 05563 | } 05564 | 05565 | template <typename Node> 05566 | struct node_collector_interface 05567 | { 05568 | typedef Node* node_ptr_t; 05569 | typedef Node** node_pp_t; 05570 | typedef std::vector<node_pp_t> noderef_list_t; 05571 | 05572 | virtual ~node_collector_interface() 05573 | {} 05574 | 05575 | virtual void collect_nodes(noderef_list_t&) 05576 | {} 05577 | }; 05578 | 05579 | template <typename Node> 05580 | struct node_depth_base; 05581 | 05582 | template <typename T> 05583 | class expression_node : public node_collector_interface<expression_node<T> > 05584 | , public node_depth_base<expression_node<T> > 05585 | { 05586 | public: 05587 | 05588 | enum node_type 05589 | { 05590 | e_none , e_null , e_constant , e_unary , 05591 | e_binary , e_binary_ext , e_trinary , e_quaternary , 05592 | e_vararg , e_conditional , e_while , e_repeat , 05593 | e_for , e_switch , e_mswitch , e_return , 05594 | e_retenv , e_variable , e_stringvar , e_stringconst , 05595 | e_stringvarrng , e_cstringvarrng , e_strgenrange , e_strconcat , 05596 | e_stringvarsize , e_strswap , e_stringsize , e_stringvararg , 05597 | e_function , e_vafunction , e_genfunction , e_strfunction , 05598 | e_strcondition , e_strccondition , e_add , e_sub , 05599 | e_mul , e_div , e_mod , e_pow , 05600 | e_lt , e_lte , e_gt , e_gte , 05601 | e_eq , e_ne , e_and , e_nand , 05602 | e_or , e_nor , e_xor , e_xnor , 05603 | e_in , e_like , e_ilike , e_inranges , 05604 | e_ipow , e_ipowinv , e_abs , e_acos , 05605 | e_acosh , e_asin , e_asinh , e_atan , 05606 | e_atanh , e_ceil , e_cos , e_cosh , 05607 | e_exp , e_expm1 , e_floor , e_log , 05608 | e_log10 , e_log2 , e_log1p , e_neg , 05609 | e_pos , e_round , e_sin , e_sinc , 05610 | e_sinh , e_sqrt , e_tan , e_tanh , 05611 | e_cot , e_sec , e_csc , e_r2d , 05612 | e_d2r , e_d2g , e_g2d , e_notl , 05613 | e_sgn , e_erf , e_erfc , e_ncdf , 05614 | e_frac , e_trunc , e_uvouv , e_vov , 05615 | e_cov , e_voc , e_vob , e_bov , 05616 | e_cob , e_boc , e_vovov , e_vovoc , 05617 | e_vocov , e_covov , e_covoc , e_vovovov , 05618 | e_vovovoc , e_vovocov , e_vocovov , e_covovov , 05619 | e_covocov , e_vocovoc , e_covovoc , e_vococov , 05620 | e_sf3ext , e_sf4ext , e_nulleq , e_strass , 05621 | e_vector , e_vecsize , e_vecelem , e_veccelem , 05622 | e_vecelemrtc , e_veccelemrtc , e_rbvecelem , e_rbvecelemrtc , 05623 | e_rbveccelem , e_rbveccelemrtc , e_vecinit , e_vecvalass , 05624 | e_vecvecass , e_vecopvalass , e_vecopvecass , e_vecfunc , 05625 | e_vecvecswap , e_vecvecineq , e_vecvalineq , e_valvecineq , 05626 | e_vecvecarith , e_vecvalarith , e_valvecarith , e_vecunaryop , 05627 | e_vecondition , e_break , e_continue , e_swap , 05628 | e_assert 05629 | }; 05630 | 05631 | typedef T value_type; 05632 | typedef expression_node<T>* expression_ptr; 05633 | typedef node_collector_interface<expression_node<T> > nci_t; 05634 | typedef typename nci_t::noderef_list_t noderef_list_t; 05635 | typedef node_depth_base<expression_node<T> > ndb_t; 05636 | 05637 | virtual ~expression_node() 05638 | {} 05639 | 05640 | inline virtual T value() const 05641 | { 05642 | return std::numeric_limits<T>::quiet_NaN(); 05643 | } 05644 | 05645 | inline virtual expression_node<T>* branch(const std::size_t& index = 0) const 05646 | { 05647 | return reinterpret_cast<expression_ptr>(index * 0); 05648 | } 05649 | 05650 | inline virtual node_type type() const 05651 | { 05652 | return e_none; 05653 | } 05654 | 05655 | inline virtual bool valid() const 05656 | { 05657 | return true; 05658 | } 05659 | }; // class expression_node 05660 | 05661 | template <typename T> 05662 | inline bool is_generally_string_node(const expression_node<T>* node); 05663 | 05664 | inline bool is_true(const double v) 05665 | { 05666 | return std::not_equal_to<double>()(0.0,v); 05667 | } 05668 | 05669 | inline bool is_true(const long double v) 05670 | { 05671 | return std::not_equal_to<long double>()(0.0L,v); 05672 | } 05673 | 05674 | inline bool is_true(const float v) 05675 | { 05676 | return std::not_equal_to<float>()(0.0f,v); 05677 | } 05678 | 05679 | template <typename T> 05680 | inline bool is_true(const expression_node<T>* node) 05681 | { 05682 | return std::not_equal_to<T>()(T(0),node->value()); 05683 | } 05684 | 05685 | template <typename T> 05686 | inline bool is_true(const std::pair<expression_node<T>*,bool>& node) 05687 | { 05688 | return std::not_equal_to<T>()(T(0),node.first->value()); 05689 | } 05690 | 05691 | template <typename T> 05692 | inline bool is_false(const expression_node<T>* node) 05693 | { 05694 | return std::equal_to<T>()(T(0),node->value()); 05695 | } 05696 | 05697 | template <typename T> 05698 | inline bool is_false(const std::pair<expression_node<T>*,bool>& node) 05699 | { 05700 | return std::equal_to<T>()(T(0),node.first->value()); 05701 | } 05702 | 05703 | template <typename T> 05704 | inline bool is_literal_node(const expression_node<T>* node) 05705 | { 05706 | return node && (details::expression_node<T>::e_constant == node->type()); 05707 | } 05708 | 05709 | template <typename T> 05710 | inline bool is_unary_node(const expression_node<T>* node) 05711 | { 05712 | return node && (details::expression_node<T>::e_unary == node->type()); 05713 | } 05714 | 05715 | template <typename T> 05716 | inline bool is_neg_unary_node(const expression_node<T>* node) 05717 | { 05718 | return node && (details::expression_node<T>::e_neg == node->type()); 05719 | } 05720 | 05721 | template <typename T> 05722 | inline bool is_binary_node(const expression_node<T>* node) 05723 | { 05724 | return node && (details::expression_node<T>::e_binary == node->type()); 05725 | } 05726 | 05727 | template <typename T> 05728 | inline bool is_variable_node(const expression_node<T>* node) 05729 | { 05730 | return node && (details::expression_node<T>::e_variable == node->type()); 05731 | } 05732 | 05733 | template <typename T> 05734 | inline bool is_ivariable_node(const expression_node<T>* node) 05735 | { 05736 | return node && 05737 | ( 05738 | details::expression_node<T>::e_variable == node->type() || 05739 | details::expression_node<T>::e_vecelem == node->type() || 05740 | details::expression_node<T>::e_veccelem == node->type() || 05741 | details::expression_node<T>::e_vecelemrtc == node->type() || 05742 | details::expression_node<T>::e_veccelemrtc == node->type() || 05743 | details::expression_node<T>::e_rbvecelem == node->type() || 05744 | details::expression_node<T>::e_rbveccelem == node->type() || 05745 | details::expression_node<T>::e_rbvecelemrtc == node->type() || 05746 | details::expression_node<T>::e_rbveccelemrtc == node->type() 05747 | ); 05748 | } 05749 | 05750 | template <typename T> 05751 | inline bool is_vector_elem_node(const expression_node<T>* node) 05752 | { 05753 | return node && (details::expression_node<T>::e_vecelem == node->type()); 05754 | } 05755 | 05756 | template <typename T> 05757 | inline bool is_vector_celem_node(const expression_node<T>* node) 05758 | { 05759 | return node && (details::expression_node<T>::e_veccelem == node->type()); 05760 | } 05761 | 05762 | template <typename T> 05763 | inline bool is_vector_elem_rtc_node(const expression_node<T>* node) 05764 | { 05765 | return node && (details::expression_node<T>::e_vecelemrtc == node->type()); 05766 | } 05767 | 05768 | template <typename T> 05769 | inline bool is_vector_celem_rtc_node(const expression_node<T>* node) 05770 | { 05771 | return node && (details::expression_node<T>::e_veccelemrtc == node->type()); 05772 | } 05773 | 05774 | template <typename T> 05775 | inline bool is_rebasevector_elem_node(const expression_node<T>* node) 05776 | { 05777 | return node && (details::expression_node<T>::e_rbvecelem == node->type()); 05778 | } 05779 | 05780 | template <typename T> 05781 | inline bool is_rebasevector_elem_rtc_node(const expression_node<T>* node) 05782 | { 05783 | return node && (details::expression_node<T>::e_rbvecelemrtc == node->type()); 05784 | } 05785 | 05786 | template <typename T> 05787 | inline bool is_rebasevector_celem_rtc_node(const expression_node<T>* node) 05788 | { 05789 | return node && (details::expression_node<T>::e_rbveccelemrtc == node->type()); 05790 | } 05791 | 05792 | template <typename T> 05793 | inline bool is_rebasevector_celem_node(const expression_node<T>* node) 05794 | { 05795 | return node && (details::expression_node<T>::e_rbveccelem == node->type()); 05796 | } 05797 | 05798 | template <typename T> 05799 | inline bool is_vector_node(const expression_node<T>* node) 05800 | { 05801 | return node && (details::expression_node<T>::e_vector == node->type()); 05802 | } 05803 | 05804 | template <typename T> 05805 | inline bool is_ivector_node(const expression_node<T>* node) 05806 | { 05807 | if (node) 05808 | { 05809 | switch (node->type()) 05810 | { 05811 | case details::expression_node<T>::e_vector : 05812 | case details::expression_node<T>::e_vecvalass : 05813 | case details::expression_node<T>::e_vecvecass : 05814 | case details::expression_node<T>::e_vecopvalass : 05815 | case details::expression_node<T>::e_vecopvecass : 05816 | case details::expression_node<T>::e_vecvecswap : 05817 | case details::expression_node<T>::e_vecvecarith : 05818 | case details::expression_node<T>::e_vecvalarith : 05819 | case details::expression_node<T>::e_valvecarith : 05820 | case details::expression_node<T>::e_vecunaryop : 05821 | case details::expression_node<T>::e_vecondition : return true; 05822 | default : return false; 05823 | } 05824 | } 05825 | else 05826 | return false; 05827 | } 05828 | 05829 | template <typename T> 05830 | inline bool is_constant_node(const expression_node<T>* node) 05831 | { 05832 | return node && 05833 | ( 05834 | details::expression_node<T>::e_constant == node->type() || 05835 | details::expression_node<T>::e_stringconst == node->type() 05836 | ); 05837 | } 05838 | 05839 | template <typename T> 05840 | inline bool is_null_node(const expression_node<T>* node) 05841 | { 05842 | return node && (details::expression_node<T>::e_null == node->type()); 05843 | } 05844 | 05845 | template <typename T> 05846 | inline bool is_break_node(const expression_node<T>* node) 05847 | { 05848 | return node && (details::expression_node<T>::e_break == node->type()); 05849 | } 05850 | 05851 | template <typename T> 05852 | inline bool is_continue_node(const expression_node<T>* node) 05853 | { 05854 | return node && (details::expression_node<T>::e_continue == node->type()); 05855 | } 05856 | 05857 | template <typename T> 05858 | inline bool is_swap_node(const expression_node<T>* node) 05859 | { 05860 | return node && (details::expression_node<T>::e_swap == node->type()); 05861 | } 05862 | 05863 | template <typename T> 05864 | inline bool is_function(const expression_node<T>* node) 05865 | { 05866 | return node && (details::expression_node<T>::e_function == node->type()); 05867 | } 05868 | 05869 | template <typename T> 05870 | inline bool is_vararg_node(const expression_node<T>* node) 05871 | { 05872 | return node && (details::expression_node<T>::e_vararg == node->type()); 05873 | } 05874 | 05875 | template <typename T> 05876 | inline bool is_return_node(const expression_node<T>* node) 05877 | { 05878 | return node && (details::expression_node<T>::e_return == node->type()); 05879 | } 05880 | 05881 | template <typename T> class unary_node; 05882 | 05883 | template <typename T> 05884 | inline bool is_negate_node(const expression_node<T>* node) 05885 | { 05886 | if (node && is_unary_node(node)) 05887 | { 05888 | return (details::e_neg == static_cast<const unary_node<T>*>(node)->operation()); 05889 | } 05890 | else 05891 | return false; 05892 | } 05893 | 05894 | template <typename T> 05895 | inline bool is_assert_node(const expression_node<T>* node) 05896 | { 05897 | return node && (details::expression_node<T>::e_assert == node->type()); 05898 | } 05899 | 05900 | template <typename T> 05901 | inline bool branch_deletable(const expression_node<T>* node) 05902 | { 05903 | return (0 != node) && 05904 | !is_variable_node(node) && 05905 | !is_string_node (node) ; 05906 | } 05907 | 05908 | template <std::size_t N, typename T> 05909 | inline bool all_nodes_valid(expression_node<T>* const (&b)[N]) 05910 | { 05911 | for (std::size_t i = 0; i < N; ++i) 05912 | { 05913 | if (0 == b[i]) return false; 05914 | } 05915 | 05916 | return true; 05917 | } 05918 | 05919 | template <typename T, 05920 | typename Allocator, 05921 | template <typename, typename> class Sequence> 05922 | inline bool all_nodes_valid(const Sequence<expression_node<T>*,Allocator>& b) 05923 | { 05924 | for (std::size_t i = 0; i < b.size(); ++i) 05925 | { 05926 | if (0 == b[i]) return false; 05927 | } 05928 | 05929 | return true; 05930 | } 05931 | 05932 | template <std::size_t N, typename T> 05933 | inline bool all_nodes_variables(expression_node<T>* const (&b)[N]) 05934 | { 05935 | for (std::size_t i = 0; i < N; ++i) 05936 | { 05937 | if (0 == b[i]) 05938 | return false; 05939 | else if (!is_variable_node(b[i])) 05940 | return false; 05941 | } 05942 | 05943 | return true; 05944 | } 05945 | 05946 | template <typename T, 05947 | typename Allocator, 05948 | template <typename, typename> class Sequence> 05949 | inline bool all_nodes_variables(const Sequence<expression_node<T>*,Allocator>& b) 05950 | { 05951 | for (std::size_t i = 0; i < b.size(); ++i) 05952 | { 05953 | if (0 == b[i]) 05954 | return false; 05955 | else if (!is_variable_node(b[i])) 05956 | return false; 05957 | } 05958 | 05959 | return true; 05960 | } 05961 | 05962 | template <typename Node> 05963 | class node_collection_destructor 05964 | { 05965 | public: 05966 | 05967 | typedef node_collector_interface<Node> nci_t; 05968 | 05969 | typedef typename nci_t::node_ptr_t node_ptr_t; 05970 | typedef typename nci_t::node_pp_t node_pp_t; 05971 | typedef typename nci_t::noderef_list_t noderef_list_t; 05972 | 05973 | static void delete_nodes(node_ptr_t& root) 05974 | { 05975 | std::vector<node_pp_t> node_delete_list; 05976 | node_delete_list.reserve(1000); 05977 | 05978 | collect_nodes(root, node_delete_list); 05979 | 05980 | for (std::size_t i = 0; i < node_delete_list.size(); ++i) 05981 | { 05982 | node_ptr_t& node = *node_delete_list[i]; 05983 | exprtk_debug(("ncd::delete_nodes() - deleting: %p\n", reinterpret_cast<void*>(node))); 05984 | delete node; 05985 | node = reinterpret_cast<node_ptr_t>(0); 05986 | } 05987 | } 05988 | 05989 | private: 05990 | 05991 | static void collect_nodes(node_ptr_t& root, noderef_list_t& node_delete_list) 05992 | { 05993 | std::deque<node_ptr_t> node_list; 05994 | node_list.push_back(root); 05995 | node_delete_list.push_back(&root); 05996 | 05997 | noderef_list_t child_node_delete_list; 05998 | child_node_delete_list.reserve(1000); 05999 | 06000 | while (!node_list.empty()) 06001 | { 06002 | node_list.front()->collect_nodes(child_node_delete_list); 06003 | 06004 | if (!child_node_delete_list.empty()) 06005 | { 06006 | for (std::size_t i = 0; i < child_node_delete_list.size(); ++i) 06007 | { 06008 | node_pp_t& node = child_node_delete_list[i]; 06009 | 06010 | if (0 == (*node)) 06011 | { 06012 | exprtk_debug(("ncd::collect_nodes() - null node encountered.\n")); 06013 | } 06014 | 06015 | node_list.push_back(*node); 06016 | } 06017 | 06018 | node_delete_list.insert( 06019 | node_delete_list.end(), 06020 | child_node_delete_list.begin(), child_node_delete_list.end()); 06021 | 06022 | child_node_delete_list.clear(); 06023 | } 06024 | 06025 | node_list.pop_front(); 06026 | } 06027 | 06028 | std::reverse(node_delete_list.begin(), node_delete_list.end()); 06029 | } 06030 | }; 06031 | 06032 | template <typename NodeAllocator, typename T, std::size_t N> 06033 | inline void free_all_nodes(NodeAllocator& node_allocator, expression_node<T>* (&b)[N]) 06034 | { 06035 | for (std::size_t i = 0; i < N; ++i) 06036 | { 06037 | free_node(node_allocator,b[i]); 06038 | } 06039 | } 06040 | 06041 | template <typename NodeAllocator, 06042 | typename T, 06043 | typename Allocator, 06044 | template <typename, typename> class Sequence> 06045 | inline void free_all_nodes(NodeAllocator& node_allocator, Sequence<expression_node<T>*,Allocator>& b) 06046 | { 06047 | for (std::size_t i = 0; i < b.size(); ++i) 06048 | { 06049 | free_node(node_allocator,b[i]); 06050 | } 06051 | 06052 | b.clear(); 06053 | } 06054 | 06055 | template <typename NodeAllocator, typename T> 06056 | inline void free_node(NodeAllocator&, expression_node<T>*& node) 06057 | { 06058 | if ((0 == node) || is_variable_node(node) || is_string_node(node)) 06059 | { 06060 | return; 06061 | } 06062 | 06063 | node_collection_destructor<expression_node<T> > 06064 | ::delete_nodes(node); 06065 | } 06066 | 06067 | template <typename T> 06068 | inline void destroy_node(expression_node<T>*& node) 06069 | { 06070 | if (0 != node) 06071 | { 06072 | node_collection_destructor<expression_node<T> > 06073 | ::delete_nodes(node); 06074 | } 06075 | } 06076 | 06077 | template <typename Node> 06078 | struct node_depth_base 06079 | { 06080 | typedef Node* node_ptr_t; 06081 | typedef std::pair<node_ptr_t,bool> nb_pair_t; 06082 | 06083 | node_depth_base() 06084 | : depth_set(false) 06085 | , depth(0) 06086 | {} 06087 | 06088 | virtual ~node_depth_base() 06089 | {} 06090 | 06091 | virtual std::size_t node_depth() const { return 1; } 06092 | 06093 | std::size_t compute_node_depth(const Node* const& node) const 06094 | { 06095 | if (!depth_set) 06096 | { 06097 | depth = 1 + (node ? node->node_depth() : 0); 06098 | depth_set = true; 06099 | } 06100 | 06101 | return depth; 06102 | } 06103 | 06104 | std::size_t compute_node_depth(const nb_pair_t& branch) const 06105 | { 06106 | if (!depth_set) 06107 | { 06108 | depth = 1 + (branch.first ? branch.first->node_depth() : 0); 06109 | depth_set = true; 06110 | } 06111 | 06112 | return depth; 06113 | } 06114 | 06115 | template <std::size_t N> 06116 | std::size_t compute_node_depth(const nb_pair_t (&branch)[N]) const 06117 | { 06118 | if (!depth_set) 06119 | { 06120 | depth = 0; 06121 | 06122 | for (std::size_t i = 0; i < N; ++i) 06123 | { 06124 | if (branch[i].first) 06125 | { 06126 | depth = std::max(depth,branch[i].first->node_depth()); 06127 | } 06128 | } 06129 | 06130 | depth += 1; 06131 | depth_set = true; 06132 | } 06133 | 06134 | return depth; 06135 | } 06136 | 06137 | template <typename BranchType> 06138 | std::size_t max_node_depth(const BranchType& n0, const BranchType& n1) const 06139 | { 06140 | return std::max(compute_node_depth(n0), compute_node_depth(n1)); 06141 | } 06142 | 06143 | template <typename BranchType> 06144 | std::size_t max_node_depth(const BranchType& n0, const BranchType& n1, const BranchType& n2) const 06145 | { 06146 | return std::max(compute_node_depth(n0), 06147 | std::max(compute_node_depth(n1), compute_node_depth(n2))); 06148 | } 06149 | 06150 | template <typename BranchType> 06151 | std::size_t max_node_depth(const BranchType& n0, const BranchType& n1, 06152 | const BranchType& n2, const BranchType& n3) const 06153 | { 06154 | return std::max( 06155 | std::max(compute_node_depth(n0), compute_node_depth(n1)), 06156 | std::max(compute_node_depth(n2), compute_node_depth(n3))); 06157 | } 06158 | 06159 | template <typename BranchType> 06160 | std::size_t compute_node_depth(const BranchType& n0, const BranchType& n1) const 06161 | { 06162 | if (!depth_set) 06163 | { 06164 | depth = 1 + max_node_depth(n0, n1); 06165 | depth_set = true; 06166 | } 06167 | 06168 | return depth; 06169 | } 06170 | 06171 | template <typename BranchType> 06172 | std::size_t compute_node_depth(const BranchType& n0, const BranchType& n1, 06173 | const BranchType& n2) const 06174 | { 06175 | if (!depth_set) 06176 | { 06177 | depth = 1 + max_node_depth(n0, n1, n2); 06178 | depth_set = true; 06179 | } 06180 | 06181 | return depth; 06182 | } 06183 | 06184 | template <typename BranchType> 06185 | std::size_t compute_node_depth(const BranchType& n0, const BranchType& n1, 06186 | const BranchType& n2, const BranchType& n3) const 06187 | { 06188 | if (!depth_set) 06189 | { 06190 | depth = 1 + max_node_depth(n0, n1, n2, n3); 06191 | depth_set = true; 06192 | } 06193 | 06194 | return depth; 06195 | } 06196 | 06197 | template <typename Allocator, 06198 | template <typename, typename> class Sequence> 06199 | std::size_t compute_node_depth(const Sequence<node_ptr_t, Allocator>& branch_list) const 06200 | { 06201 | if (!depth_set) 06202 | { 06203 | for (std::size_t i = 0; i < branch_list.size(); ++i) 06204 | { 06205 | if (branch_list[i]) 06206 | { 06207 | depth = std::max(depth, compute_node_depth(branch_list[i])); 06208 | } 06209 | } 06210 | 06211 | depth_set = true; 06212 | } 06213 | 06214 | return depth; 06215 | } 06216 | 06217 | template <typename Allocator, 06218 | template <typename, typename> class Sequence> 06219 | std::size_t compute_node_depth(const Sequence<nb_pair_t,Allocator>& branch_list) const 06220 | { 06221 | if (!depth_set) 06222 | { 06223 | for (std::size_t i = 0; i < branch_list.size(); ++i) 06224 | { 06225 | if (branch_list[i].first) 06226 | { 06227 | depth = std::max(depth, compute_node_depth(branch_list[i].first)); 06228 | } 06229 | } 06230 | 06231 | depth_set = true; 06232 | } 06233 | 06234 | return depth; 06235 | } 06236 | 06237 | mutable bool depth_set; 06238 | mutable std::size_t depth; 06239 | 06240 | template <typename NodeSequence> 06241 | void collect(node_ptr_t const& node, 06242 | const bool deletable, 06243 | NodeSequence& delete_node_list) const 06244 | { 06245 | if ((0 != node) && deletable) 06246 | { 06247 | delete_node_list.push_back(const_cast<node_ptr_t*>(&node)); 06248 | } 06249 | } 06250 | 06251 | template <typename NodeSequence> 06252 | void collect(const nb_pair_t& branch, 06253 | NodeSequence& delete_node_list) const 06254 | { 06255 | collect(branch.first, branch.second, delete_node_list); 06256 | } 06257 | 06258 | template <typename NodeSequence> 06259 | void collect(Node*& node, 06260 | NodeSequence& delete_node_list) const 06261 | { 06262 | collect(node, branch_deletable(node), delete_node_list); 06263 | } 06264 | 06265 | template <std::size_t N, typename NodeSequence> 06266 | void collect(const nb_pair_t(&branch)[N], 06267 | NodeSequence& delete_node_list) const 06268 | { 06269 | for (std::size_t i = 0; i < N; ++i) 06270 | { 06271 | collect(branch[i].first, branch[i].second, delete_node_list); 06272 | } 06273 | } 06274 | 06275 | template <typename Allocator, 06276 | template <typename, typename> class Sequence, 06277 | typename NodeSequence> 06278 | void collect(const Sequence<nb_pair_t, Allocator>& branch, 06279 | NodeSequence& delete_node_list) const 06280 | { 06281 | for (std::size_t i = 0; i < branch.size(); ++i) 06282 | { 06283 | collect(branch[i].first, branch[i].second, delete_node_list); 06284 | } 06285 | } 06286 | 06287 | template <typename Allocator, 06288 | template <typename, typename> class Sequence, 06289 | typename NodeSequence> 06290 | void collect(const Sequence<node_ptr_t, Allocator>& branch_list, 06291 | NodeSequence& delete_node_list) const 06292 | { 06293 | for (std::size_t i = 0; i < branch_list.size(); ++i) 06294 | { 06295 | collect(branch_list[i], branch_deletable(branch_list[i]), delete_node_list); 06296 | } 06297 | } 06298 | 06299 | template <typename Boolean, 06300 | typename AllocatorT, 06301 | typename AllocatorB, 06302 | template <typename, typename> class Sequence, 06303 | typename NodeSequence> 06304 | void collect(const Sequence<node_ptr_t, AllocatorT>& branch_list, 06305 | const Sequence<Boolean, AllocatorB>& branch_deletable_list, 06306 | NodeSequence& delete_node_list) const 06307 | { 06308 | for (std::size_t i = 0; i < branch_list.size(); ++i) 06309 | { 06310 | collect(branch_list[i], branch_deletable_list[i], delete_node_list); 06311 | } 06312 | } 06313 | }; 06314 | 06315 | template <typename Type> 06316 | class vector_holder 06317 | { 06318 | private: 06319 | 06320 | typedef Type value_type; 06321 | typedef value_type* value_ptr; 06322 | typedef const value_ptr const_value_ptr; 06323 | typedef vector_holder<Type> vector_holder_t; 06324 | 06325 | class vector_holder_base 06326 | { 06327 | public: 06328 | 06329 | virtual ~vector_holder_base() 06330 | {} 06331 | 06332 | inline value_ptr operator[](const std::size_t& index) const 06333 | { 06334 | return value_at(index); 06335 | } 06336 | 06337 | inline std::size_t size() const 06338 | { 06339 | return vector_size(); 06340 | } 06341 | 06342 | inline std::size_t base_size() const 06343 | { 06344 | return vector_base_size(); 06345 | } 06346 | 06347 | inline value_ptr data() const 06348 | { 06349 | return value_at(0); 06350 | } 06351 | 06352 | virtual inline bool rebaseable() const 06353 | { 06354 | return false; 06355 | } 06356 | 06357 | virtual void set_ref(value_ptr*) 06358 | {} 06359 | 06360 | virtual void remove_ref(value_ptr*) 06361 | {} 06362 | 06363 | virtual vector_view<Type>* rebaseable_instance() 06364 | { 06365 | return reinterpret_cast<vector_view<Type>*>(0); 06366 | } 06367 | 06368 | protected: 06369 | 06370 | virtual value_ptr value_at(const std::size_t&) const = 0; 06371 | virtual std::size_t vector_size() const = 0; 06372 | virtual std::size_t vector_base_size() const = 0; 06373 | }; 06374 | 06375 | class array_vector_impl exprtk_final : public vector_holder_base 06376 | { 06377 | public: 06378 | 06379 | array_vector_impl(const Type* vec, const std::size_t& vec_size) 06380 | : vec_(vec) 06381 | , size_(vec_size) 06382 | {} 06383 | 06384 | protected: 06385 | 06386 | value_ptr value_at(const std::size_t& index) const exprtk_override 06387 | { 06388 | assert(index < size_); 06389 | return const_cast<const_value_ptr>(vec_ + index); 06390 | } 06391 | 06392 | std::size_t vector_size() const exprtk_override 06393 | { 06394 | return size_; 06395 | } 06396 | 06397 | std::size_t vector_base_size() const exprtk_override 06398 | { 06399 | return vector_size(); 06400 | } 06401 | 06402 | private: 06403 | 06404 | array_vector_impl(const array_vector_impl&) exprtk_delete; 06405 | array_vector_impl& operator=(const array_vector_impl&) exprtk_delete; 06406 | 06407 | const Type* vec_; 06408 | const std::size_t size_; 06409 | }; 06410 | 06411 | template <typename Allocator, 06412 | template <typename, typename> class Sequence> 06413 | class sequence_vector_impl exprtk_final : public vector_holder_base 06414 | { 06415 | public: 06416 | 06417 | typedef Sequence<Type,Allocator> sequence_t; 06418 | 06419 | explicit sequence_vector_impl(sequence_t& seq) 06420 | : sequence_(seq) 06421 | {} 06422 | 06423 | protected: 06424 | 06425 | value_ptr value_at(const std::size_t& index) const exprtk_override 06426 | { 06427 | assert(index < sequence_.size()); 06428 | return (&sequence_[index]); 06429 | } 06430 | 06431 | std::size_t vector_size() const exprtk_override 06432 | { 06433 | return sequence_.size(); 06434 | } 06435 | 06436 | std::size_t vector_base_size() const exprtk_override 06437 | { 06438 | return vector_size(); 06439 | } 06440 | 06441 | private: 06442 | 06443 | sequence_vector_impl(const sequence_vector_impl&) exprtk_delete; 06444 | sequence_vector_impl& operator=(const sequence_vector_impl&) exprtk_delete; 06445 | 06446 | sequence_t& sequence_; 06447 | }; 06448 | 06449 | class vector_view_impl exprtk_final : public vector_holder_base 06450 | { 06451 | public: 06452 | 06453 | typedef exprtk::vector_view<Type> vector_view_t; 06454 | 06455 | explicit vector_view_impl(vector_view_t& vec_view) 06456 | : vec_view_(vec_view) 06457 | { 06458 | assert(vec_view_.size() > 0); 06459 | } 06460 | 06461 | void set_ref(value_ptr* ref) exprtk_override 06462 | { 06463 | vec_view_.set_ref(ref); 06464 | } 06465 | 06466 | void remove_ref(value_ptr* ref) exprtk_override 06467 | { 06468 | vec_view_.remove_ref(ref); 06469 | } 06470 | 06471 | bool rebaseable() const exprtk_override 06472 | { 06473 | return true; 06474 | } 06475 | 06476 | vector_view<Type>* rebaseable_instance() exprtk_override 06477 | { 06478 | return &vec_view_; 06479 | } 06480 | 06481 | protected: 06482 | 06483 | value_ptr value_at(const std::size_t& index) const exprtk_override 06484 | { 06485 | assert(index < vec_view_.size()); 06486 | return (&vec_view_[index]); 06487 | } 06488 | 06489 | std::size_t vector_size() const exprtk_override 06490 | { 06491 | return vec_view_.size(); 06492 | } 06493 | 06494 | std::size_t vector_base_size() const exprtk_override 06495 | { 06496 | return vec_view_.base_size(); 06497 | } 06498 | 06499 | private: 06500 | 06501 | vector_view_impl(const vector_view_impl&) exprtk_delete; 06502 | vector_view_impl& operator=(const vector_view_impl&) exprtk_delete; 06503 | 06504 | vector_view_t& vec_view_; 06505 | }; 06506 | 06507 | class resizable_vector_impl exprtk_final : public vector_holder_base 06508 | { 06509 | public: 06510 | 06511 | resizable_vector_impl(vector_holder& vec_view_holder, 06512 | const Type* vec, 06513 | const std::size_t& vec_size) 06514 | : vec_(vec) 06515 | , size_(vec_size) 06516 | , vec_view_holder_(*vec_view_holder.rebaseable_instance()) 06517 | { 06518 | assert(vec_view_holder.rebaseable_instance()); 06519 | assert(size_ <= vector_base_size()); 06520 | } 06521 | 06522 | virtual ~resizable_vector_impl() exprtk_override 06523 | {} 06524 | 06525 | protected: 06526 | 06527 | value_ptr value_at(const std::size_t& index) const exprtk_override 06528 | { 06529 | assert(index < vector_size()); 06530 | return const_cast<const_value_ptr>(vec_ + index); 06531 | } 06532 | 06533 | std::size_t vector_size() const exprtk_override 06534 | { 06535 | return vec_view_holder_.size(); 06536 | } 06537 | 06538 | std::size_t vector_base_size() const exprtk_override 06539 | { 06540 | return vec_view_holder_.base_size(); 06541 | } 06542 | 06543 | bool rebaseable() const exprtk_override 06544 | { 06545 | return true; 06546 | } 06547 | 06548 | virtual vector_view<Type>* rebaseable_instance() exprtk_override 06549 | { 06550 | return &vec_view_holder_; 06551 | } 06552 | 06553 | private: 06554 | 06555 | resizable_vector_impl(const resizable_vector_impl&) exprtk_delete; 06556 | resizable_vector_impl& operator=(const resizable_vector_impl&) exprtk_delete; 06557 | 06558 | const Type* vec_; 06559 | const std::size_t size_; 06560 | vector_view<Type>& vec_view_holder_; 06561 | }; 06562 | 06563 | public: 06564 | 06565 | typedef typename details::vec_data_store<Type> vds_t; 06566 | 06567 | vector_holder(Type* vec, const std::size_t& vec_size) 06568 | : vector_holder_base_(new(buffer)array_vector_impl(vec,vec_size)) 06569 | {} 06570 | 06571 | explicit vector_holder(const vds_t& vds) 06572 | : vector_holder_base_(new(buffer)array_vector_impl(vds.data(),vds.size())) 06573 | {} 06574 | 06575 | template <typename Allocator> 06576 | explicit vector_holder(std::vector<Type,Allocator>& vec) 06577 | : vector_holder_base_(new(buffer)sequence_vector_impl<Allocator,std::vector>(vec)) 06578 | {} 06579 | 06580 | explicit vector_holder(exprtk::vector_view<Type>& vec) 06581 | : vector_holder_base_(new(buffer)vector_view_impl(vec)) 06582 | {} 06583 | 06584 | explicit vector_holder(vector_holder_t& vec_holder, const vds_t& vds) 06585 | : vector_holder_base_(new(buffer)resizable_vector_impl(vec_holder, vds.data(), vds.size())) 06586 | {} 06587 | 06588 | inline value_ptr operator[](const std::size_t& index) const 06589 | { 06590 | return (*vector_holder_base_)[index]; 06591 | } 06592 | 06593 | inline std::size_t size() const 06594 | { 06595 | return vector_holder_base_->size(); 06596 | } 06597 | 06598 | inline std::size_t base_size() const 06599 | { 06600 | return vector_holder_base_->base_size(); 06601 | } 06602 | 06603 | inline value_ptr data() const 06604 | { 06605 | return vector_holder_base_->data(); 06606 | } 06607 | 06608 | void set_ref(value_ptr* ref) 06609 | { 06610 | if (rebaseable()) 06611 | { 06612 | vector_holder_base_->set_ref(ref); 06613 | } 06614 | } 06615 | 06616 | void remove_ref(value_ptr* ref) 06617 | { 06618 | if (rebaseable()) 06619 | { 06620 | vector_holder_base_->remove_ref(ref); 06621 | } 06622 | } 06623 | 06624 | bool rebaseable() const 06625 | { 06626 | return vector_holder_base_->rebaseable(); 06627 | } 06628 | 06629 | vector_view<Type>* rebaseable_instance() 06630 | { 06631 | return vector_holder_base_->rebaseable_instance(); 06632 | } 06633 | 06634 | private: 06635 | 06636 | vector_holder(const vector_holder<Type>&) exprtk_delete; 06637 | vector_holder<Type>& operator=(const vector_holder<Type>&) exprtk_delete; 06638 | 06639 | mutable vector_holder_base* vector_holder_base_; 06640 | uchar_t buffer[64]; 06641 | }; 06642 | 06643 | template <typename T> 06644 | class null_node exprtk_final : public expression_node<T> 06645 | { 06646 | public: 06647 | 06648 | inline T value() const exprtk_override 06649 | { 06650 | return std::numeric_limits<T>::quiet_NaN(); 06651 | } 06652 | 06653 | inline typename expression_node<T>::node_type type() const exprtk_override 06654 | { 06655 | return expression_node<T>::e_null; 06656 | } 06657 | }; 06658 | 06659 | template <typename T, std::size_t N> 06660 | inline void construct_branch_pair(std::pair<expression_node<T>*,bool> (&branch)[N], 06661 | expression_node<T>* b, 06662 | const std::size_t& index) 06663 | { 06664 | if (b && (index < N)) 06665 | { 06666 | branch[index] = std::make_pair(b,branch_deletable(b)); 06667 | } 06668 | } 06669 | 06670 | template <typename T> 06671 | inline void construct_branch_pair(std::pair<expression_node<T>*,bool>& branch, expression_node<T>* b) 06672 | { 06673 | if (b) 06674 | { 06675 | branch = std::make_pair(b,branch_deletable(b)); 06676 | } 06677 | } 06678 | 06679 | template <std::size_t N, typename T> 06680 | inline void init_branches(std::pair<expression_node<T>*,bool> (&branch)[N], 06681 | expression_node<T>* b0, 06682 | expression_node<T>* b1 = reinterpret_cast<expression_node<T>*>(0), 06683 | expression_node<T>* b2 = reinterpret_cast<expression_node<T>*>(0), 06684 | expression_node<T>* b3 = reinterpret_cast<expression_node<T>*>(0), 06685 | expression_node<T>* b4 = reinterpret_cast<expression_node<T>*>(0), 06686 | expression_node<T>* b5 = reinterpret_cast<expression_node<T>*>(0), 06687 | expression_node<T>* b6 = reinterpret_cast<expression_node<T>*>(0), 06688 | expression_node<T>* b7 = reinterpret_cast<expression_node<T>*>(0), 06689 | expression_node<T>* b8 = reinterpret_cast<expression_node<T>*>(0), 06690 | expression_node<T>* b9 = reinterpret_cast<expression_node<T>*>(0)) 06691 | { 06692 | construct_branch_pair(branch, b0, 0); 06693 | construct_branch_pair(branch, b1, 1); 06694 | construct_branch_pair(branch, b2, 2); 06695 | construct_branch_pair(branch, b3, 3); 06696 | construct_branch_pair(branch, b4, 4); 06697 | construct_branch_pair(branch, b5, 5); 06698 | construct_branch_pair(branch, b6, 6); 06699 | construct_branch_pair(branch, b7, 7); 06700 | construct_branch_pair(branch, b8, 8); 06701 | construct_branch_pair(branch, b9, 9); 06702 | } 06703 | 06704 | template <typename T> 06705 | class null_eq_node exprtk_final : public expression_node<T> 06706 | { 06707 | public: 06708 | 06709 | typedef expression_node<T>* expression_ptr; 06710 | typedef std::pair<expression_ptr,bool> branch_t; 06711 | 06712 | explicit null_eq_node(expression_ptr branch, const bool equality = true) 06713 | : equality_(equality) 06714 | { 06715 | construct_branch_pair(branch_, branch); 06716 | assert(valid()); 06717 | } 06718 | 06719 | inline T value() const exprtk_override 06720 | { 06721 | const T v = branch_.first->value(); 06722 | const bool result = details::numeric::is_nan(v); 06723 | 06724 | if (result) 06725 | return equality_ ? T(1) : T(0); 06726 | else 06727 | return equality_ ? T(0) : T(1); 06728 | } 06729 | 06730 | inline typename expression_node<T>::node_type type() const exprtk_override 06731 | { 06732 | return expression_node<T>::e_nulleq; 06733 | } 06734 | 06735 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 06736 | { 06737 | return branch_.first; 06738 | } 06739 | 06740 | inline bool valid() const exprtk_override 06741 | { 06742 | return branch_.first; 06743 | } 06744 | 06745 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 06746 | { 06747 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 06748 | } 06749 | 06750 | std::size_t node_depth() const exprtk_override 06751 | { 06752 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 06753 | } 06754 | 06755 | private: 06756 | 06757 | bool equality_; 06758 | branch_t branch_; 06759 | }; 06760 | 06761 | template <typename T> 06762 | class literal_node exprtk_final : public expression_node<T> 06763 | { 06764 | public: 06765 | 06766 | explicit literal_node(const T& v) 06767 | : value_(v) 06768 | {} 06769 | 06770 | inline T value() const exprtk_override 06771 | { 06772 | return value_; 06773 | } 06774 | 06775 | inline typename expression_node<T>::node_type type() const exprtk_override 06776 | { 06777 | return expression_node<T>::e_constant; 06778 | } 06779 | 06780 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 06781 | { 06782 | return reinterpret_cast<expression_node<T>*>(0); 06783 | } 06784 | 06785 | private: 06786 | 06787 | literal_node(const literal_node<T>&) exprtk_delete; 06788 | literal_node<T>& operator=(const literal_node<T>&) exprtk_delete; 06789 | 06790 | const T value_; 06791 | }; 06792 | 06793 | template <typename T> 06794 | struct range_pack; 06795 | 06796 | template <typename T> 06797 | struct range_data_type; 06798 | 06799 | template <typename T> 06800 | class range_interface 06801 | { 06802 | public: 06803 | 06804 | typedef range_pack<T> range_t; 06805 | 06806 | virtual ~range_interface() 06807 | {} 06808 | 06809 | virtual range_t& range_ref() = 0; 06810 | 06811 | virtual const range_t& range_ref() const = 0; 06812 | }; 06813 | 06814 | #ifndef exprtk_disable_string_capabilities 06815 | template <typename T> 06816 | class string_base_node 06817 | { 06818 | public: 06819 | 06820 | typedef range_data_type<T> range_data_type_t; 06821 | 06822 | virtual ~string_base_node() 06823 | {} 06824 | 06825 | virtual std::string str () const = 0; 06826 | 06827 | virtual char_cptr base() const = 0; 06828 | 06829 | virtual std::size_t size() const = 0; 06830 | }; 06831 | 06832 | template <typename T> 06833 | class string_literal_node exprtk_final 06834 | : public expression_node <T> 06835 | , public string_base_node<T> 06836 | , public range_interface <T> 06837 | { 06838 | public: 06839 | 06840 | typedef range_pack<T> range_t; 06841 | 06842 | explicit string_literal_node(const std::string& v) 06843 | : value_(v) 06844 | { 06845 | rp_.n0_c = std::make_pair<bool,std::size_t>(true, 0); 06846 | rp_.n1_c = std::make_pair<bool,std::size_t>(true, v.size()); 06847 | rp_.cache.first = rp_.n0_c.second; 06848 | rp_.cache.second = rp_.n1_c.second; 06849 | } 06850 | 06851 | inline T value() const exprtk_override 06852 | { 06853 | return std::numeric_limits<T>::quiet_NaN(); 06854 | } 06855 | 06856 | inline typename expression_node<T>::node_type type() const exprtk_override 06857 | { 06858 | return expression_node<T>::e_stringconst; 06859 | } 06860 | 06861 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 06862 | { 06863 | return reinterpret_cast<expression_node<T>*>(0); 06864 | } 06865 | 06866 | std::string str() const exprtk_override 06867 | { 06868 | return value_; 06869 | } 06870 | 06871 | char_cptr base() const exprtk_override 06872 | { 06873 | return value_.data(); 06874 | } 06875 | 06876 | std::size_t size() const exprtk_override 06877 | { 06878 | return value_.size(); 06879 | } 06880 | 06881 | range_t& range_ref() exprtk_override 06882 | { 06883 | return rp_; 06884 | } 06885 | 06886 | const range_t& range_ref() const exprtk_override 06887 | { 06888 | return rp_; 06889 | } 06890 | 06891 | private: 06892 | 06893 | string_literal_node(const string_literal_node<T>&) exprtk_delete; 06894 | string_literal_node<T>& operator=(const string_literal_node<T>&) exprtk_delete; 06895 | 06896 | const std::string value_; 06897 | range_t rp_; 06898 | }; 06899 | #endif 06900 | 06901 | template <typename T> 06902 | class unary_node : public expression_node<T> 06903 | { 06904 | public: 06905 | 06906 | typedef expression_node<T>* expression_ptr; 06907 | typedef std::pair<expression_ptr,bool> branch_t; 06908 | 06909 | unary_node(const operator_type& opr, expression_ptr branch) 06910 | : operation_(opr) 06911 | { 06912 | construct_branch_pair(branch_,branch); 06913 | assert(valid()); 06914 | } 06915 | 06916 | inline T value() const exprtk_override 06917 | { 06918 | return numeric::process<T> 06919 | (operation_,branch_.first->value()); 06920 | } 06921 | 06922 | inline typename expression_node<T>::node_type type() const exprtk_override 06923 | { 06924 | return expression_node<T>::e_unary; 06925 | } 06926 | 06927 | inline operator_type operation() 06928 | { 06929 | return operation_; 06930 | } 06931 | 06932 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 06933 | { 06934 | return branch_.first; 06935 | } 06936 | 06937 | inline bool valid() const exprtk_override 06938 | { 06939 | return branch_.first && branch_.first->valid(); 06940 | } 06941 | 06942 | inline void release() 06943 | { 06944 | branch_.second = false; 06945 | } 06946 | 06947 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 06948 | { 06949 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 06950 | } 06951 | 06952 | std::size_t node_depth() const exprtk_final 06953 | { 06954 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 06955 | } 06956 | 06957 | private: 06958 | 06959 | operator_type operation_; 06960 | branch_t branch_; 06961 | }; 06962 | 06963 | template <typename T> 06964 | class binary_node : public expression_node<T> 06965 | { 06966 | public: 06967 | 06968 | typedef expression_node<T>* expression_ptr; 06969 | typedef std::pair<expression_ptr,bool> branch_t; 06970 | 06971 | binary_node(const operator_type& opr, 06972 | expression_ptr branch0, 06973 | expression_ptr branch1) 06974 | : operation_(opr) 06975 | { 06976 | init_branches<2>(branch_, branch0, branch1); 06977 | assert(valid()); 06978 | } 06979 | 06980 | inline T value() const exprtk_override 06981 | { 06982 | return numeric::process<T> 06983 | ( 06984 | operation_, 06985 | branch_[0].first->value(), 06986 | branch_[1].first->value() 06987 | ); 06988 | } 06989 | 06990 | inline typename expression_node<T>::node_type type() const exprtk_override 06991 | { 06992 | return expression_node<T>::e_binary; 06993 | } 06994 | 06995 | inline operator_type operation() 06996 | { 06997 | return operation_; 06998 | } 06999 | 07000 | inline expression_node<T>* branch(const std::size_t& index = 0) const exprtk_override 07001 | { 07002 | assert(index < 2); 07003 | return branch_[index].first; 07004 | } 07005 | 07006 | inline bool valid() const exprtk_override 07007 | { 07008 | return 07009 | branch_[0].first && branch_[0].first->valid() && 07010 | branch_[1].first && branch_[1].first->valid() ; 07011 | } 07012 | 07013 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07014 | { 07015 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 07016 | } 07017 | 07018 | std::size_t node_depth() const exprtk_final 07019 | { 07020 | return expression_node<T>::ndb_t::template compute_node_depth<2>(branch_); 07021 | } 07022 | 07023 | private: 07024 | 07025 | operator_type operation_; 07026 | branch_t branch_[2]; 07027 | }; 07028 | 07029 | template <typename T, typename Operation> 07030 | class binary_ext_node exprtk_final : public expression_node<T> 07031 | { 07032 | public: 07033 | 07034 | typedef expression_node<T>* expression_ptr; 07035 | typedef std::pair<expression_ptr,bool> branch_t; 07036 | 07037 | binary_ext_node(expression_ptr branch0, expression_ptr branch1) 07038 | { 07039 | init_branches<2>(branch_, branch0, branch1); 07040 | assert(valid()); 07041 | } 07042 | 07043 | inline T value() const exprtk_override 07044 | { 07045 | const T arg0 = branch_[0].first->value(); 07046 | const T arg1 = branch_[1].first->value(); 07047 | return Operation::process(arg0,arg1); 07048 | } 07049 | 07050 | inline typename expression_node<T>::node_type type() const exprtk_override 07051 | { 07052 | return expression_node<T>::e_binary_ext; 07053 | } 07054 | 07055 | inline operator_type operation() 07056 | { 07057 | return Operation::operation(); 07058 | } 07059 | 07060 | inline expression_node<T>* branch(const std::size_t& index = 0) const exprtk_override 07061 | { 07062 | assert(index < 2); 07063 | return branch_[index].first; 07064 | } 07065 | 07066 | inline bool valid() const exprtk_override 07067 | { 07068 | return 07069 | branch_[0].first && branch_[0].first->valid() && 07070 | branch_[1].first && branch_[1].first->valid() ; 07071 | } 07072 | 07073 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07074 | { 07075 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 07076 | } 07077 | 07078 | std::size_t node_depth() const exprtk_override 07079 | { 07080 | return expression_node<T>::ndb_t::template compute_node_depth<2>(branch_); 07081 | } 07082 | 07083 | protected: 07084 | 07085 | branch_t branch_[2]; 07086 | }; 07087 | 07088 | template <typename T> 07089 | class trinary_node : public expression_node<T> 07090 | { 07091 | public: 07092 | 07093 | typedef expression_node<T>* expression_ptr; 07094 | typedef std::pair<expression_ptr,bool> branch_t; 07095 | 07096 | trinary_node(const operator_type& opr, 07097 | expression_ptr branch0, 07098 | expression_ptr branch1, 07099 | expression_ptr branch2) 07100 | : operation_(opr) 07101 | { 07102 | init_branches<3>(branch_, branch0, branch1, branch2); 07103 | assert(valid()); 07104 | } 07105 | 07106 | inline T value() const exprtk_override 07107 | { 07108 | const T arg0 = branch_[0].first->value(); 07109 | const T arg1 = branch_[1].first->value(); 07110 | const T arg2 = branch_[2].first->value(); 07111 | 07112 | switch (operation_) 07113 | { 07114 | case e_inrange : return (arg1 < arg0) ? T(0) : ((arg1 > arg2) ? T(0) : T(1)); 07115 | 07116 | case e_clamp : return (arg1 < arg0) ? arg0 : (arg1 > arg2 ? arg2 : arg1); 07117 | 07118 | case e_iclamp : if ((arg1 <= arg0) || (arg1 >= arg2)) 07119 | return arg1; 07120 | else 07121 | return ((T(2) * arg1 <= (arg2 + arg0)) ? arg0 : arg2); 07122 | 07123 | default : exprtk_debug(("trinary_node::value() - Error: Invalid operation\n")); 07124 | return std::numeric_limits<T>::quiet_NaN(); 07125 | } 07126 | } 07127 | 07128 | inline typename expression_node<T>::node_type type() const exprtk_override 07129 | { 07130 | return expression_node<T>::e_trinary; 07131 | } 07132 | 07133 | inline bool valid() const exprtk_override 07134 | { 07135 | return 07136 | branch_[0].first && branch_[0].first->valid() && 07137 | branch_[1].first && branch_[1].first->valid() && 07138 | branch_[2].first && branch_[2].first->valid() ; 07139 | } 07140 | 07141 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07142 | { 07143 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 07144 | } 07145 | 07146 | std::size_t node_depth() const exprtk_override exprtk_final 07147 | { 07148 | return expression_node<T>::ndb_t::template compute_node_depth<3>(branch_); 07149 | } 07150 | 07151 | protected: 07152 | 07153 | operator_type operation_; 07154 | branch_t branch_[3]; 07155 | }; 07156 | 07157 | template <typename T> 07158 | class quaternary_node : public expression_node<T> 07159 | { 07160 | public: 07161 | 07162 | typedef expression_node<T>* expression_ptr; 07163 | typedef std::pair<expression_ptr,bool> branch_t; 07164 | 07165 | quaternary_node(const operator_type& opr, 07166 | expression_ptr branch0, 07167 | expression_ptr branch1, 07168 | expression_ptr branch2, 07169 | expression_ptr branch3) 07170 | : operation_(opr) 07171 | { 07172 | init_branches<4>(branch_, branch0, branch1, branch2, branch3); 07173 | } 07174 | 07175 | inline T value() const exprtk_override 07176 | { 07177 | return std::numeric_limits<T>::quiet_NaN(); 07178 | } 07179 | 07180 | inline typename expression_node<T>::node_type type() const exprtk_override 07181 | { 07182 | return expression_node<T>::e_quaternary; 07183 | } 07184 | 07185 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07186 | { 07187 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 07188 | } 07189 | 07190 | std::size_t node_depth() const exprtk_override exprtk_final 07191 | { 07192 | return expression_node<T>::ndb_t::template compute_node_depth<4>(branch_); 07193 | } 07194 | 07195 | inline bool valid() const exprtk_override 07196 | { 07197 | return 07198 | branch_[0].first && branch_[0].first->valid() && 07199 | branch_[1].first && branch_[1].first->valid() && 07200 | branch_[2].first && branch_[2].first->valid() && 07201 | branch_[3].first && branch_[3].first->valid() ; 07202 | } 07203 | 07204 | protected: 07205 | 07206 | operator_type operation_; 07207 | branch_t branch_[4]; 07208 | }; 07209 | 07210 | template <typename T> 07211 | class conditional_node exprtk_final : public expression_node<T> 07212 | { 07213 | public: 07214 | 07215 | typedef expression_node<T>* expression_ptr; 07216 | typedef std::pair<expression_ptr,bool> branch_t; 07217 | 07218 | conditional_node(expression_ptr condition, 07219 | expression_ptr consequent, 07220 | expression_ptr alternative) 07221 | { 07222 | construct_branch_pair(condition_ , condition ); 07223 | construct_branch_pair(consequent_ , consequent ); 07224 | construct_branch_pair(alternative_, alternative); 07225 | assert(valid()); 07226 | } 07227 | 07228 | inline T value() const exprtk_override 07229 | { 07230 | if (is_true(condition_)) 07231 | return consequent_.first->value(); 07232 | else 07233 | return alternative_.first->value(); 07234 | } 07235 | 07236 | inline typename expression_node<T>::node_type type() const exprtk_override 07237 | { 07238 | return expression_node<T>::e_conditional; 07239 | } 07240 | 07241 | inline bool valid() const exprtk_override 07242 | { 07243 | return 07244 | condition_ .first && condition_ .first->valid() && 07245 | consequent_ .first && consequent_ .first->valid() && 07246 | alternative_.first && alternative_.first->valid() ; 07247 | } 07248 | 07249 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07250 | { 07251 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 07252 | expression_node<T>::ndb_t::collect(consequent_ , node_delete_list); 07253 | expression_node<T>::ndb_t::collect(alternative_ , node_delete_list); 07254 | } 07255 | 07256 | std::size_t node_depth() const exprtk_override 07257 | { 07258 | return expression_node<T>::ndb_t::compute_node_depth 07259 | (condition_, consequent_, alternative_); 07260 | } 07261 | 07262 | private: 07263 | 07264 | branch_t condition_; 07265 | branch_t consequent_; 07266 | branch_t alternative_; 07267 | }; 07268 | 07269 | template <typename T> 07270 | class cons_conditional_node exprtk_final : public expression_node<T> 07271 | { 07272 | public: 07273 | 07274 | // Consequent only conditional statement node 07275 | typedef expression_node<T>* expression_ptr; 07276 | typedef std::pair<expression_ptr,bool> branch_t; 07277 | 07278 | cons_conditional_node(expression_ptr condition, 07279 | expression_ptr consequent) 07280 | { 07281 | construct_branch_pair(condition_ , condition ); 07282 | construct_branch_pair(consequent_, consequent); 07283 | assert(valid()); 07284 | } 07285 | 07286 | inline T value() const exprtk_override 07287 | { 07288 | if (is_true(condition_)) 07289 | return consequent_.first->value(); 07290 | else 07291 | return std::numeric_limits<T>::quiet_NaN(); 07292 | } 07293 | 07294 | inline typename expression_node<T>::node_type type() const exprtk_override 07295 | { 07296 | return expression_node<T>::e_conditional; 07297 | } 07298 | 07299 | inline bool valid() const exprtk_override 07300 | { 07301 | return 07302 | condition_ .first && condition_ .first->valid() && 07303 | consequent_.first && consequent_.first->valid() ; 07304 | } 07305 | 07306 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07307 | { 07308 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 07309 | expression_node<T>::ndb_t::collect(consequent_ , node_delete_list); 07310 | } 07311 | 07312 | std::size_t node_depth() const exprtk_override 07313 | { 07314 | return expression_node<T>::ndb_t:: 07315 | compute_node_depth(condition_, consequent_); 07316 | } 07317 | 07318 | private: 07319 | 07320 | branch_t condition_; 07321 | branch_t consequent_; 07322 | }; 07323 | 07324 | #ifndef exprtk_disable_break_continue 07325 | template <typename T> 07326 | class break_exception 07327 | { 07328 | public: 07329 | 07330 | explicit break_exception(const T& v) 07331 | : value(v) 07332 | {} 07333 | 07334 | T value; 07335 | }; 07336 | 07337 | class continue_exception {}; 07338 | 07339 | template <typename T> 07340 | class break_node exprtk_final : public expression_node<T> 07341 | { 07342 | public: 07343 | 07344 | typedef expression_node<T>* expression_ptr; 07345 | typedef std::pair<expression_ptr,bool> branch_t; 07346 | 07347 | explicit break_node(expression_ptr ret = expression_ptr(0)) 07348 | { 07349 | construct_branch_pair(return_, ret); 07350 | } 07351 | 07352 | inline T value() const exprtk_override 07353 | { 07354 | const T result = return_.first ? 07355 | return_.first->value() : 07356 | std::numeric_limits<T>::quiet_NaN(); 07357 | 07358 | throw break_exception<T>(result); 07359 | 07360 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 07361 | return std::numeric_limits<T>::quiet_NaN(); 07362 | #endif 07363 | } 07364 | 07365 | inline typename expression_node<T>::node_type type() const exprtk_override 07366 | { 07367 | return expression_node<T>::e_break; 07368 | } 07369 | 07370 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07371 | { 07372 | expression_node<T>::ndb_t::collect(return_, node_delete_list); 07373 | } 07374 | 07375 | std::size_t node_depth() const exprtk_override 07376 | { 07377 | return expression_node<T>::ndb_t::compute_node_depth(return_); 07378 | } 07379 | 07380 | private: 07381 | 07382 | branch_t return_; 07383 | }; 07384 | 07385 | template <typename T> 07386 | class continue_node exprtk_final : public expression_node<T> 07387 | { 07388 | public: 07389 | 07390 | inline T value() const exprtk_override 07391 | { 07392 | throw continue_exception(); 07393 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 07394 | return std::numeric_limits<T>::quiet_NaN(); 07395 | #endif 07396 | } 07397 | 07398 | inline typename expression_node<T>::node_type type() const exprtk_override 07399 | { 07400 | return expression_node<T>::e_break; 07401 | } 07402 | }; 07403 | #endif 07404 | 07405 | struct loop_runtime_checker 07406 | { 07407 | loop_runtime_checker(loop_runtime_check_ptr loop_runtime_check, 07408 | loop_runtime_check::loop_types lp_typ = loop_runtime_check::e_invalid) 07409 | : iteration_count_(0) 07410 | , loop_runtime_check_(loop_runtime_check) 07411 | , max_loop_iterations_(loop_runtime_check_->max_loop_iterations) 07412 | , loop_type_(lp_typ) 07413 | { 07414 | assert(loop_runtime_check_); 07415 | } 07416 | 07417 | inline void reset(const _uint64_t initial_value = 0) const 07418 | { 07419 | iteration_count_ = initial_value; 07420 | } 07421 | 07422 | inline bool check() const 07423 | { 07424 | assert(loop_runtime_check_); 07425 | 07426 | if ( 07427 | (++iteration_count_ <= max_loop_iterations_) && 07428 | loop_runtime_check_->check() 07429 | ) 07430 | { 07431 | return true; 07432 | } 07433 | 07434 | loop_runtime_check::violation_context ctxt; 07435 | ctxt.loop = loop_type_; 07436 | ctxt.violation = loop_runtime_check::e_iteration_count; 07437 | 07438 | loop_runtime_check_->handle_runtime_violation(ctxt); 07439 | 07440 | return false; 07441 | } 07442 | 07443 | bool valid() const 07444 | { 07445 | return 0 != loop_runtime_check_; 07446 | } 07447 | 07448 | mutable _uint64_t iteration_count_; 07449 | mutable loop_runtime_check_ptr loop_runtime_check_; 07450 | const details::_uint64_t& max_loop_iterations_; 07451 | loop_runtime_check::loop_types loop_type_; 07452 | }; 07453 | 07454 | template <typename T> 07455 | class while_loop_node : public expression_node<T> 07456 | { 07457 | public: 07458 | 07459 | typedef expression_node<T>* expression_ptr; 07460 | typedef std::pair<expression_ptr,bool> branch_t; 07461 | 07462 | while_loop_node(expression_ptr condition, 07463 | expression_ptr loop_body) 07464 | { 07465 | construct_branch_pair(condition_, condition); 07466 | construct_branch_pair(loop_body_, loop_body); 07467 | assert(valid()); 07468 | } 07469 | 07470 | inline T value() const exprtk_override 07471 | { 07472 | T result = T(0); 07473 | 07474 | while (is_true(condition_)) 07475 | { 07476 | result = loop_body_.first->value(); 07477 | } 07478 | 07479 | return result; 07480 | } 07481 | 07482 | inline typename expression_node<T>::node_type type() const exprtk_override 07483 | { 07484 | return expression_node<T>::e_while; 07485 | } 07486 | 07487 | inline bool valid() const exprtk_override 07488 | { 07489 | return 07490 | condition_.first && condition_.first->valid() && 07491 | loop_body_.first && loop_body_.first->valid() ; 07492 | } 07493 | 07494 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07495 | { 07496 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 07497 | expression_node<T>::ndb_t::collect(loop_body_ , node_delete_list); 07498 | } 07499 | 07500 | std::size_t node_depth() const exprtk_override 07501 | { 07502 | return expression_node<T>::ndb_t::compute_node_depth(condition_, loop_body_); 07503 | } 07504 | 07505 | protected: 07506 | 07507 | branch_t condition_; 07508 | branch_t loop_body_; 07509 | }; 07510 | 07511 | template <typename T> 07512 | class while_loop_rtc_node exprtk_final 07513 | : public while_loop_node<T> 07514 | , public loop_runtime_checker 07515 | { 07516 | public: 07517 | 07518 | typedef while_loop_node<T> parent_t; 07519 | typedef expression_node<T>* expression_ptr; 07520 | 07521 | while_loop_rtc_node(expression_ptr condition, 07522 | expression_ptr loop_body, 07523 | loop_runtime_check_ptr loop_rt_chk) 07524 | : parent_t(condition, loop_body) 07525 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_while_loop) 07526 | { 07527 | assert(valid()); 07528 | } 07529 | 07530 | inline T value() const exprtk_override 07531 | { 07532 | 07533 | T result = T(0); 07534 | 07535 | loop_runtime_checker::reset(); 07536 | 07537 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 07538 | { 07539 | result = parent_t::loop_body_.first->value(); 07540 | } 07541 | 07542 | return result; 07543 | } 07544 | 07545 | using parent_t::valid; 07546 | 07547 | bool valid() const exprtk_override exprtk_final 07548 | { 07549 | return parent_t::valid() && 07550 | loop_runtime_checker::valid(); 07551 | } 07552 | }; 07553 | 07554 | template <typename T> 07555 | class repeat_until_loop_node : public expression_node<T> 07556 | { 07557 | public: 07558 | 07559 | typedef expression_node<T>* expression_ptr; 07560 | typedef std::pair<expression_ptr,bool> branch_t; 07561 | 07562 | repeat_until_loop_node(expression_ptr condition, 07563 | expression_ptr loop_body) 07564 | { 07565 | construct_branch_pair(condition_, condition); 07566 | construct_branch_pair(loop_body_, loop_body); 07567 | assert(valid()); 07568 | } 07569 | 07570 | inline T value() const exprtk_override 07571 | { 07572 | T result = T(0); 07573 | 07574 | do 07575 | { 07576 | result = loop_body_.first->value(); 07577 | } 07578 | while (is_false(condition_.first)); 07579 | 07580 | return result; 07581 | } 07582 | 07583 | inline typename expression_node<T>::node_type type() const exprtk_override 07584 | { 07585 | return expression_node<T>::e_repeat; 07586 | } 07587 | 07588 | inline bool valid() const exprtk_override 07589 | { 07590 | return 07591 | condition_.first && condition_.first->valid() && 07592 | loop_body_.first && loop_body_.first->valid() ; 07593 | } 07594 | 07595 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07596 | { 07597 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 07598 | expression_node<T>::ndb_t::collect(loop_body_ , node_delete_list); 07599 | } 07600 | 07601 | std::size_t node_depth() const exprtk_override 07602 | { 07603 | return expression_node<T>::ndb_t::compute_node_depth(condition_, loop_body_); 07604 | } 07605 | 07606 | protected: 07607 | 07608 | branch_t condition_; 07609 | branch_t loop_body_; 07610 | }; 07611 | 07612 | template <typename T> 07613 | class repeat_until_loop_rtc_node exprtk_final 07614 | : public repeat_until_loop_node<T> 07615 | , public loop_runtime_checker 07616 | { 07617 | public: 07618 | 07619 | typedef repeat_until_loop_node<T> parent_t; 07620 | typedef expression_node<T>* expression_ptr; 07621 | 07622 | repeat_until_loop_rtc_node(expression_ptr condition, 07623 | expression_ptr loop_body, 07624 | loop_runtime_check_ptr loop_rt_chk) 07625 | : parent_t(condition, loop_body) 07626 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_repeat_until_loop) 07627 | { 07628 | assert(valid()); 07629 | } 07630 | 07631 | inline T value() const exprtk_override 07632 | { 07633 | T result = T(0); 07634 | 07635 | loop_runtime_checker::reset(1); 07636 | 07637 | do 07638 | { 07639 | result = parent_t::loop_body_.first->value(); 07640 | } 07641 | while (is_false(parent_t::condition_.first) && loop_runtime_checker::check()); 07642 | 07643 | return result; 07644 | } 07645 | 07646 | using parent_t::valid; 07647 | 07648 | inline bool valid() const exprtk_override exprtk_final 07649 | { 07650 | return parent_t::valid() && 07651 | loop_runtime_checker::valid(); 07652 | } 07653 | }; 07654 | 07655 | template <typename T> 07656 | class for_loop_node : public expression_node<T> 07657 | { 07658 | public: 07659 | 07660 | typedef expression_node<T>* expression_ptr; 07661 | typedef std::pair<expression_ptr,bool> branch_t; 07662 | 07663 | for_loop_node(expression_ptr initialiser, 07664 | expression_ptr condition, 07665 | expression_ptr incrementor, 07666 | expression_ptr loop_body) 07667 | { 07668 | construct_branch_pair(initialiser_, initialiser); 07669 | construct_branch_pair(condition_ , condition ); 07670 | construct_branch_pair(incrementor_, incrementor); 07671 | construct_branch_pair(loop_body_ , loop_body ); 07672 | assert(valid()); 07673 | } 07674 | 07675 | inline T value() const exprtk_override 07676 | { 07677 | T result = T(0); 07678 | 07679 | if (initialiser_.first) 07680 | initialiser_.first->value(); 07681 | 07682 | if (incrementor_.first) 07683 | { 07684 | while (is_true(condition_)) 07685 | { 07686 | result = loop_body_.first->value(); 07687 | incrementor_.first->value(); 07688 | } 07689 | } 07690 | else 07691 | { 07692 | while (is_true(condition_)) 07693 | { 07694 | result = loop_body_.first->value(); 07695 | } 07696 | } 07697 | 07698 | return result; 07699 | } 07700 | 07701 | inline typename expression_node<T>::node_type type() const exprtk_override 07702 | { 07703 | return expression_node<T>::e_for; 07704 | } 07705 | 07706 | inline bool valid() const exprtk_override 07707 | { 07708 | return condition_.first && loop_body_.first; 07709 | } 07710 | 07711 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 07712 | { 07713 | expression_node<T>::ndb_t::collect(initialiser_ , node_delete_list); 07714 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 07715 | expression_node<T>::ndb_t::collect(incrementor_ , node_delete_list); 07716 | expression_node<T>::ndb_t::collect(loop_body_ , node_delete_list); 07717 | } 07718 | 07719 | std::size_t node_depth() const exprtk_override 07720 | { 07721 | return expression_node<T>::ndb_t::compute_node_depth 07722 | (initialiser_, condition_, incrementor_, loop_body_); 07723 | } 07724 | 07725 | protected: 07726 | 07727 | branch_t initialiser_; 07728 | branch_t condition_ ; 07729 | branch_t incrementor_; 07730 | branch_t loop_body_ ; 07731 | }; 07732 | 07733 | template <typename T> 07734 | class for_loop_rtc_node exprtk_final 07735 | : public for_loop_node<T> 07736 | , public loop_runtime_checker 07737 | { 07738 | public: 07739 | 07740 | typedef for_loop_node<T> parent_t; 07741 | typedef expression_node<T>* expression_ptr; 07742 | 07743 | for_loop_rtc_node(expression_ptr initialiser, 07744 | expression_ptr condition, 07745 | expression_ptr incrementor, 07746 | expression_ptr loop_body, 07747 | loop_runtime_check_ptr loop_rt_chk) 07748 | : parent_t(initialiser, condition, incrementor, loop_body) 07749 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_for_loop) 07750 | { 07751 | assert(valid()); 07752 | } 07753 | 07754 | inline T value() const exprtk_override 07755 | { 07756 | T result = T(0); 07757 | 07758 | loop_runtime_checker::reset(); 07759 | 07760 | if (parent_t::initialiser_.first) 07761 | parent_t::initialiser_.first->value(); 07762 | 07763 | if (parent_t::incrementor_.first) 07764 | { 07765 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 07766 | { 07767 | result = parent_t::loop_body_.first->value(); 07768 | parent_t::incrementor_.first->value(); 07769 | } 07770 | } 07771 | else 07772 | { 07773 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 07774 | { 07775 | result = parent_t::loop_body_.first->value(); 07776 | } 07777 | } 07778 | 07779 | return result; 07780 | } 07781 | 07782 | using parent_t::valid; 07783 | 07784 | inline bool valid() const exprtk_override exprtk_final 07785 | { 07786 | return parent_t::valid() && 07787 | loop_runtime_checker::valid(); 07788 | } 07789 | }; 07790 | 07791 | #ifndef exprtk_disable_break_continue 07792 | template <typename T> 07793 | class while_loop_bc_node : public while_loop_node<T> 07794 | { 07795 | public: 07796 | 07797 | typedef while_loop_node<T> parent_t; 07798 | typedef expression_node<T>* expression_ptr; 07799 | 07800 | while_loop_bc_node(expression_ptr condition, 07801 | expression_ptr loop_body) 07802 | : parent_t(condition, loop_body) 07803 | { 07804 | assert(parent_t::valid()); 07805 | } 07806 | 07807 | inline T value() const exprtk_override 07808 | { 07809 | T result = T(0); 07810 | 07811 | while (is_true(parent_t::condition_)) 07812 | { 07813 | try 07814 | { 07815 | result = parent_t::loop_body_.first->value(); 07816 | } 07817 | catch(const break_exception<T>& e) 07818 | { 07819 | return e.value; 07820 | } 07821 | catch(const continue_exception&) 07822 | {} 07823 | } 07824 | 07825 | return result; 07826 | } 07827 | }; 07828 | 07829 | template <typename T> 07830 | class while_loop_bc_rtc_node exprtk_final 07831 | : public while_loop_bc_node<T> 07832 | , public loop_runtime_checker 07833 | { 07834 | public: 07835 | 07836 | typedef while_loop_bc_node<T> parent_t; 07837 | typedef expression_node<T>* expression_ptr; 07838 | 07839 | while_loop_bc_rtc_node(expression_ptr condition, 07840 | expression_ptr loop_body, 07841 | loop_runtime_check_ptr loop_rt_chk) 07842 | : parent_t(condition, loop_body) 07843 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_while_loop) 07844 | { 07845 | assert(valid()); 07846 | } 07847 | 07848 | inline T value() const exprtk_override 07849 | { 07850 | T result = T(0); 07851 | 07852 | loop_runtime_checker::reset(); 07853 | 07854 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 07855 | { 07856 | try 07857 | { 07858 | result = parent_t::loop_body_.first->value(); 07859 | } 07860 | catch(const break_exception<T>& e) 07861 | { 07862 | return e.value; 07863 | } 07864 | catch(const continue_exception&) 07865 | {} 07866 | } 07867 | 07868 | return result; 07869 | } 07870 | 07871 | using parent_t::valid; 07872 | 07873 | inline bool valid() const exprtk_override exprtk_final 07874 | { 07875 | return parent_t::valid() && 07876 | loop_runtime_checker::valid(); 07877 | } 07878 | }; 07879 | 07880 | template <typename T> 07881 | class repeat_until_loop_bc_node : public repeat_until_loop_node<T> 07882 | { 07883 | public: 07884 | 07885 | typedef repeat_until_loop_node<T> parent_t; 07886 | typedef expression_node<T>* expression_ptr; 07887 | 07888 | repeat_until_loop_bc_node(expression_ptr condition, 07889 | expression_ptr loop_body) 07890 | : parent_t(condition, loop_body) 07891 | { 07892 | assert(parent_t::valid()); 07893 | } 07894 | 07895 | inline T value() const exprtk_override 07896 | { 07897 | T result = T(0); 07898 | 07899 | do 07900 | { 07901 | try 07902 | { 07903 | result = parent_t::loop_body_.first->value(); 07904 | } 07905 | catch(const break_exception<T>& e) 07906 | { 07907 | return e.value; 07908 | } 07909 | catch(const continue_exception&) 07910 | {} 07911 | } 07912 | while (is_false(parent_t::condition_.first)); 07913 | 07914 | return result; 07915 | } 07916 | }; 07917 | 07918 | template <typename T> 07919 | class repeat_until_loop_bc_rtc_node exprtk_final 07920 | : public repeat_until_loop_bc_node<T> 07921 | , public loop_runtime_checker 07922 | { 07923 | public: 07924 | 07925 | typedef repeat_until_loop_bc_node<T> parent_t; 07926 | typedef expression_node<T>* expression_ptr; 07927 | 07928 | repeat_until_loop_bc_rtc_node(expression_ptr condition, 07929 | expression_ptr loop_body, 07930 | loop_runtime_check_ptr loop_rt_chk) 07931 | : parent_t(condition, loop_body) 07932 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_repeat_until_loop) 07933 | { 07934 | assert(valid()); 07935 | } 07936 | 07937 | inline T value() const exprtk_override 07938 | { 07939 | T result = T(0); 07940 | 07941 | loop_runtime_checker::reset(); 07942 | 07943 | do 07944 | { 07945 | try 07946 | { 07947 | result = parent_t::loop_body_.first->value(); 07948 | } 07949 | catch(const break_exception<T>& e) 07950 | { 07951 | return e.value; 07952 | } 07953 | catch(const continue_exception&) 07954 | {} 07955 | } 07956 | while (is_false(parent_t::condition_.first) && loop_runtime_checker::check()); 07957 | 07958 | return result; 07959 | } 07960 | 07961 | using parent_t::valid; 07962 | 07963 | inline bool valid() const exprtk_override exprtk_final 07964 | { 07965 | return parent_t::valid() && 07966 | loop_runtime_checker::valid(); 07967 | } 07968 | }; 07969 | 07970 | template <typename T> 07971 | class for_loop_bc_node : public for_loop_node<T> 07972 | { 07973 | public: 07974 | 07975 | typedef for_loop_node<T> parent_t; 07976 | typedef expression_node<T>* expression_ptr; 07977 | 07978 | for_loop_bc_node(expression_ptr initialiser, 07979 | expression_ptr condition, 07980 | expression_ptr incrementor, 07981 | expression_ptr loop_body) 07982 | : parent_t(initialiser, condition, incrementor, loop_body) 07983 | { 07984 | assert(parent_t::valid()); 07985 | } 07986 | 07987 | inline T value() const exprtk_override 07988 | { 07989 | T result = T(0); 07990 | 07991 | if (parent_t::initialiser_.first) 07992 | parent_t::initialiser_.first->value(); 07993 | 07994 | if (parent_t::incrementor_.first) 07995 | { 07996 | while (is_true(parent_t::condition_)) 07997 | { 07998 | try 07999 | { 08000 | result = parent_t::loop_body_.first->value(); 08001 | } 08002 | catch(const break_exception<T>& e) 08003 | { 08004 | return e.value; 08005 | } 08006 | catch(const continue_exception&) 08007 | {} 08008 | 08009 | parent_t::incrementor_.first->value(); 08010 | } 08011 | } 08012 | else 08013 | { 08014 | while (is_true(parent_t::condition_)) 08015 | { 08016 | try 08017 | { 08018 | result = parent_t::loop_body_.first->value(); 08019 | } 08020 | catch(const break_exception<T>& e) 08021 | { 08022 | return e.value; 08023 | } 08024 | catch(const continue_exception&) 08025 | {} 08026 | } 08027 | } 08028 | 08029 | return result; 08030 | } 08031 | }; 08032 | 08033 | template <typename T> 08034 | class for_loop_bc_rtc_node exprtk_final 08035 | : public for_loop_bc_node<T> 08036 | , public loop_runtime_checker 08037 | { 08038 | public: 08039 | 08040 | typedef for_loop_bc_node<T> parent_t; 08041 | typedef expression_node<T>* expression_ptr; 08042 | 08043 | for_loop_bc_rtc_node(expression_ptr initialiser, 08044 | expression_ptr condition, 08045 | expression_ptr incrementor, 08046 | expression_ptr loop_body, 08047 | loop_runtime_check_ptr loop_rt_chk) 08048 | : parent_t(initialiser, condition, incrementor, loop_body) 08049 | , loop_runtime_checker(loop_rt_chk, loop_runtime_check::e_for_loop) 08050 | { 08051 | assert(valid()); 08052 | } 08053 | 08054 | inline T value() const exprtk_override 08055 | { 08056 | T result = T(0); 08057 | 08058 | loop_runtime_checker::reset(); 08059 | 08060 | if (parent_t::initialiser_.first) 08061 | parent_t::initialiser_.first->value(); 08062 | 08063 | if (parent_t::incrementor_.first) 08064 | { 08065 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 08066 | { 08067 | try 08068 | { 08069 | result = parent_t::loop_body_.first->value(); 08070 | } 08071 | catch(const break_exception<T>& e) 08072 | { 08073 | return e.value; 08074 | } 08075 | catch(const continue_exception&) 08076 | {} 08077 | 08078 | parent_t::incrementor_.first->value(); 08079 | } 08080 | } 08081 | else 08082 | { 08083 | while (is_true(parent_t::condition_) && loop_runtime_checker::check()) 08084 | { 08085 | try 08086 | { 08087 | result = parent_t::loop_body_.first->value(); 08088 | } 08089 | catch(const break_exception<T>& e) 08090 | { 08091 | return e.value; 08092 | } 08093 | catch(const continue_exception&) 08094 | {} 08095 | } 08096 | } 08097 | 08098 | return result; 08099 | } 08100 | 08101 | using parent_t::valid; 08102 | 08103 | inline bool valid() const exprtk_override exprtk_final 08104 | { 08105 | return parent_t::valid() && 08106 | loop_runtime_checker::valid(); 08107 | } 08108 | }; 08109 | #endif 08110 | 08111 | template <typename T> 08112 | class switch_node : public expression_node<T> 08113 | { 08114 | public: 08115 | 08116 | typedef expression_node<T>* expression_ptr; 08117 | typedef std::pair<expression_ptr,bool> branch_t; 08118 | 08119 | template <typename Allocator, 08120 | template <typename, typename> class Sequence> 08121 | explicit switch_node(const Sequence<expression_ptr,Allocator>& arg_list) 08122 | { 08123 | if (1 != (arg_list.size() & 1)) 08124 | return; 08125 | 08126 | arg_list_.resize(arg_list.size()); 08127 | 08128 | for (std::size_t i = 0; i < arg_list.size(); ++i) 08129 | { 08130 | if (arg_list[i] && arg_list[i]->valid()) 08131 | { 08132 | construct_branch_pair(arg_list_[i], arg_list[i]); 08133 | } 08134 | else 08135 | { 08136 | arg_list_.clear(); 08137 | return; 08138 | } 08139 | } 08140 | 08141 | assert(valid()); 08142 | } 08143 | 08144 | inline T value() const exprtk_override 08145 | { 08146 | const std::size_t upper_bound = (arg_list_.size() - 1); 08147 | 08148 | for (std::size_t i = 0; i < upper_bound; i += 2) 08149 | { 08150 | expression_ptr condition = arg_list_[i ].first; 08151 | expression_ptr consequent = arg_list_[i + 1].first; 08152 | 08153 | if (is_true(condition)) 08154 | { 08155 | return consequent->value(); 08156 | } 08157 | } 08158 | 08159 | return arg_list_[upper_bound].first->value(); 08160 | } 08161 | 08162 | inline typename expression_node<T>::node_type type() const exprtk_override exprtk_final 08163 | { 08164 | return expression_node<T>::e_switch; 08165 | } 08166 | 08167 | inline bool valid() const exprtk_override 08168 | { 08169 | return !arg_list_.empty(); 08170 | } 08171 | 08172 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 08173 | { 08174 | expression_node<T>::ndb_t::collect(arg_list_, node_delete_list); 08175 | } 08176 | 08177 | std::size_t node_depth() const exprtk_override exprtk_final 08178 | { 08179 | return expression_node<T>::ndb_t::compute_node_depth(arg_list_); 08180 | } 08181 | 08182 | protected: 08183 | 08184 | std::vector<branch_t> arg_list_; 08185 | }; 08186 | 08187 | template <typename T, typename Switch_N> 08188 | class switch_n_node exprtk_final : public switch_node<T> 08189 | { 08190 | public: 08191 | 08192 | typedef expression_node<T>* expression_ptr; 08193 | 08194 | template <typename Allocator, 08195 | template <typename, typename> class Sequence> 08196 | explicit switch_n_node(const Sequence<expression_ptr,Allocator>& arg_list) 08197 | : switch_node<T>(arg_list) 08198 | {} 08199 | 08200 | inline T value() const exprtk_override 08201 | { 08202 | return Switch_N::process(switch_node<T>::arg_list_); 08203 | } 08204 | }; 08205 | 08206 | template <typename T> 08207 | class multi_switch_node exprtk_final : public expression_node<T> 08208 | { 08209 | public: 08210 | 08211 | typedef expression_node<T>* expression_ptr; 08212 | typedef std::pair<expression_ptr,bool> branch_t; 08213 | 08214 | template <typename Allocator, 08215 | template <typename, typename> class Sequence> 08216 | explicit multi_switch_node(const Sequence<expression_ptr,Allocator>& arg_list) 08217 | { 08218 | if (0 != (arg_list.size() & 1)) 08219 | return; 08220 | 08221 | arg_list_.resize(arg_list.size()); 08222 | 08223 | for (std::size_t i = 0; i < arg_list.size(); ++i) 08224 | { 08225 | if (arg_list[i] && arg_list[i]->valid()) 08226 | { 08227 | construct_branch_pair(arg_list_[i], arg_list[i]); 08228 | } 08229 | else 08230 | { 08231 | arg_list_.clear(); 08232 | return; 08233 | } 08234 | } 08235 | 08236 | assert(valid()); 08237 | } 08238 | 08239 | inline T value() const exprtk_override 08240 | { 08241 | const std::size_t upper_bound = (arg_list_.size() - 1); 08242 | 08243 | T result = T(0); 08244 | 08245 | for (std::size_t i = 0; i < upper_bound; i += 2) 08246 | { 08247 | expression_ptr condition = arg_list_[i ].first; 08248 | expression_ptr consequent = arg_list_[i + 1].first; 08249 | 08250 | if (is_true(condition)) 08251 | { 08252 | result = consequent->value(); 08253 | } 08254 | } 08255 | 08256 | return result; 08257 | } 08258 | 08259 | inline typename expression_node<T>::node_type type() const exprtk_override 08260 | { 08261 | return expression_node<T>::e_mswitch; 08262 | } 08263 | 08264 | inline bool valid() const exprtk_override 08265 | { 08266 | return !arg_list_.empty() && (0 == (arg_list_.size() % 2)); 08267 | } 08268 | 08269 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 08270 | { 08271 | expression_node<T>::ndb_t::collect(arg_list_, node_delete_list); 08272 | } 08273 | 08274 | std::size_t node_depth() const exprtk_override exprtk_final 08275 | { 08276 | return expression_node<T>::ndb_t::compute_node_depth(arg_list_); 08277 | } 08278 | 08279 | private: 08280 | 08281 | std::vector<branch_t> arg_list_; 08282 | }; 08283 | 08284 | template <typename T> 08285 | class ivariable 08286 | { 08287 | public: 08288 | 08289 | virtual ~ivariable() 08290 | {} 08291 | 08292 | virtual T& ref() = 0; 08293 | virtual const T& ref() const = 0; 08294 | }; 08295 | 08296 | template <typename T> 08297 | class variable_node exprtk_final 08298 | : public expression_node<T> 08299 | , public ivariable <T> 08300 | { 08301 | public: 08302 | 08303 | static T null_value; 08304 | 08305 | explicit variable_node() 08306 | : value_(&null_value) 08307 | {} 08308 | 08309 | explicit variable_node(T& v) 08310 | : value_(&v) 08311 | {} 08312 | 08313 | inline bool operator <(const variable_node<T>& v) const 08314 | { 08315 | return this < (&v); 08316 | } 08317 | 08318 | inline T value() const exprtk_override 08319 | { 08320 | return (*value_); 08321 | } 08322 | 08323 | inline T& ref() exprtk_override 08324 | { 08325 | return (*value_); 08326 | } 08327 | 08328 | inline const T& ref() const exprtk_override 08329 | { 08330 | return (*value_); 08331 | } 08332 | 08333 | inline typename expression_node<T>::node_type type() const exprtk_override 08334 | { 08335 | return expression_node<T>::e_variable; 08336 | } 08337 | 08338 | private: 08339 | 08340 | T* value_; 08341 | }; 08342 | 08343 | template <typename T> 08344 | T variable_node<T>::null_value = T(std::numeric_limits<T>::quiet_NaN()); 08345 | 08346 | template <typename T> 08347 | struct range_pack 08348 | { 08349 | typedef expression_node<T>* expression_node_ptr; 08350 | typedef std::pair<std::size_t,std::size_t> cached_range_t; 08351 | 08352 | range_pack() 08353 | : n0_e (std::make_pair(false,expression_node_ptr(0))) 08354 | , n1_e (std::make_pair(false,expression_node_ptr(0))) 08355 | , n0_c (std::make_pair(false,0)) 08356 | , n1_c (std::make_pair(false,0)) 08357 | , cache(std::make_pair(0,0)) 08358 | {} 08359 | 08360 | void clear() 08361 | { 08362 | n0_e = std::make_pair(false,expression_node_ptr(0)); 08363 | n1_e = std::make_pair(false,expression_node_ptr(0)); 08364 | n0_c = std::make_pair(false,0); 08365 | n1_c = std::make_pair(false,0); 08366 | cache = std::make_pair(0,0); 08367 | } 08368 | 08369 | void free() 08370 | { 08371 | if (n0_e.first && n0_e.second) 08372 | { 08373 | n0_e.first = false; 08374 | 08375 | if ( 08376 | !is_variable_node(n0_e.second) && 08377 | !is_string_node (n0_e.second) 08378 | ) 08379 | { 08380 | destroy_node(n0_e.second); 08381 | } 08382 | } 08383 | 08384 | if (n1_e.first && n1_e.second) 08385 | { 08386 | n1_e.first = false; 08387 | 08388 | if ( 08389 | !is_variable_node(n1_e.second) && 08390 | !is_string_node (n1_e.second) 08391 | ) 08392 | { 08393 | destroy_node(n1_e.second); 08394 | } 08395 | } 08396 | } 08397 | 08398 | bool const_range() const 08399 | { 08400 | return ( n0_c.first && n1_c.first) && 08401 | (!n0_e.first && !n1_e.first); 08402 | } 08403 | 08404 | bool var_range() const 08405 | { 08406 | return ( n0_e.first && n1_e.first) && 08407 | (!n0_c.first && !n1_c.first); 08408 | } 08409 | 08410 | bool operator() (std::size_t& r0, std::size_t& r1, 08411 | const std::size_t& size = std::numeric_limits<std::size_t>::max()) const 08412 | { 08413 | if (n0_c.first) 08414 | r0 = n0_c.second; 08415 | else if (n0_e.first) 08416 | { 08417 | r0 = static_cast<std::size_t>(details::numeric::to_int64(n0_e.second->value())); 08418 | } 08419 | else 08420 | return false; 08421 | 08422 | if (n1_c.first) 08423 | r1 = n1_c.second; 08424 | else if (n1_e.first) 08425 | { 08426 | r1 = static_cast<std::size_t>(details::numeric::to_int64(n1_e.second->value())); 08427 | } 08428 | else 08429 | return false; 08430 | 08431 | if ( 08432 | (std::numeric_limits<std::size_t>::max() != size) && 08433 | (std::numeric_limits<std::size_t>::max() == r1 ) 08434 | ) 08435 | { 08436 | r1 = size; 08437 | } 08438 | 08439 | cache.first = r0; 08440 | cache.second = r1; 08441 | 08442 | #ifndef exprtk_enable_range_runtime_checks 08443 | return (r0 <= r1); 08444 | #else 08445 | return range_runtime_check(r0, r1, size); 08446 | #endif 08447 | } 08448 | 08449 | inline std::size_t const_size() const 08450 | { 08451 | return (n1_c.second - n0_c.second); 08452 | } 08453 | 08454 | inline std::size_t cache_size() const 08455 | { 08456 | return (cache.second - cache.first); 08457 | } 08458 | 08459 | std::pair<bool,expression_node_ptr> n0_e; 08460 | std::pair<bool,expression_node_ptr> n1_e; 08461 | std::pair<bool,std::size_t > n0_c; 08462 | std::pair<bool,std::size_t > n1_c; 08463 | mutable cached_range_t cache; 08464 | 08465 | #ifdef exprtk_enable_range_runtime_checks 08466 | bool range_runtime_check(const std::size_t r0, 08467 | const std::size_t r1, 08468 | const std::size_t size) const 08469 | { 08470 | if (r0 > size) 08471 | { 08472 | throw std::runtime_error("range error: (r0 < 0) || (r0 > size)"); 08473 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 08474 | return false; 08475 | #endif 08476 | } 08477 | 08478 | if (r1 > size) 08479 | { 08480 | throw std::runtime_error("range error: (r1 < 0) || (r1 > size)"); 08481 | #if !defined(_MSC_VER) && !defined(__NVCOMPILER) 08482 | return false; 08483 | #endif 08484 | } 08485 | 08486 | return (r0 <= r1); 08487 | } 08488 | #endif 08489 | }; 08490 | 08491 | template <typename T> 08492 | class string_base_node; 08493 | 08494 | template <typename T> 08495 | struct range_data_type 08496 | { 08497 | typedef range_pack<T> range_t; 08498 | typedef string_base_node<T>* strbase_ptr_t; 08499 | 08500 | range_data_type() 08501 | : range(0) 08502 | , data (0) 08503 | , size (0) 08504 | , type_size(0) 08505 | , str_node (0) 08506 | {} 08507 | 08508 | range_t* range; 08509 | void* data; 08510 | std::size_t size; 08511 | std::size_t type_size; 08512 | strbase_ptr_t str_node; 08513 | }; 08514 | 08515 | template <typename T> class vector_node; 08516 | 08517 | template <typename T> 08518 | class vector_interface 08519 | { 08520 | public: 08521 | 08522 | typedef vector_node<T>* vector_node_ptr; 08523 | typedef vec_data_store<T> vds_t; 08524 | 08525 | virtual ~vector_interface() 08526 | {} 08527 | 08528 | virtual std::size_t size () const = 0; 08529 | 08530 | virtual std::size_t base_size() const = 0; 08531 | 08532 | virtual vector_node_ptr vec () const = 0; 08533 | 08534 | virtual vector_node_ptr vec () = 0; 08535 | 08536 | virtual vds_t& vds () = 0; 08537 | 08538 | virtual const vds_t& vds () const = 0; 08539 | 08540 | virtual bool side_effect () const { return false; } 08541 | }; 08542 | 08543 | template <typename T> 08544 | class vector_node exprtk_final 08545 | : public expression_node <T> 08546 | , public vector_interface<T> 08547 | { 08548 | public: 08549 | 08550 | typedef expression_node<T>* expression_ptr; 08551 | typedef vector_holder<T> vector_holder_t; 08552 | typedef vector_node<T>* vector_node_ptr; 08553 | typedef vec_data_store<T> vds_t; 08554 | 08555 | explicit vector_node(vector_holder_t* vh) 08556 | : vector_holder_(vh) 08557 | , vds_((*vector_holder_).size(),(*vector_holder_)[0]) 08558 | { 08559 | vector_holder_->set_ref(&vds_.ref()); 08560 | } 08561 | 08562 | vector_node(const vds_t& vds, vector_holder_t* vh) 08563 | : vector_holder_(vh) 08564 | , vds_(vds) 08565 | {} 08566 | 08567 | ~vector_node() exprtk_override 08568 | { 08569 | assert(valid()); 08570 | vector_holder_->remove_ref(&vds_.ref()); 08571 | } 08572 | 08573 | inline T value() const exprtk_override 08574 | { 08575 | return vds().data()[0]; 08576 | } 08577 | 08578 | vector_node_ptr vec() const exprtk_override 08579 | { 08580 | return const_cast<vector_node_ptr>(this); 08581 | } 08582 | 08583 | vector_node_ptr vec() exprtk_override 08584 | { 08585 | return this; 08586 | } 08587 | 08588 | inline typename expression_node<T>::node_type type() const exprtk_override 08589 | { 08590 | return expression_node<T>::e_vector; 08591 | } 08592 | 08593 | inline bool valid() const exprtk_override 08594 | { 08595 | return vector_holder_; 08596 | } 08597 | 08598 | std::size_t size() const exprtk_override 08599 | { 08600 | return vec_holder().size(); 08601 | } 08602 | 08603 | std::size_t base_size() const exprtk_override 08604 | { 08605 | return vec_holder().base_size(); 08606 | } 08607 | 08608 | vds_t& vds() exprtk_override 08609 | { 08610 | return vds_; 08611 | } 08612 | 08613 | const vds_t& vds() const exprtk_override 08614 | { 08615 | return vds_; 08616 | } 08617 | 08618 | inline vector_holder_t& vec_holder() 08619 | { 08620 | return (*vector_holder_); 08621 | } 08622 | 08623 | inline vector_holder_t& vec_holder() const 08624 | { 08625 | return (*vector_holder_); 08626 | } 08627 | 08628 | private: 08629 | 08630 | vector_holder_t* vector_holder_; 08631 | vds_t vds_; 08632 | }; 08633 | 08634 | template <typename T> 08635 | class vector_size_node exprtk_final 08636 | : public expression_node <T> 08637 | { 08638 | public: 08639 | 08640 | typedef expression_node<T>* expression_ptr; 08641 | typedef vector_holder<T> vector_holder_t; 08642 | 08643 | explicit vector_size_node(vector_holder_t* vh) 08644 | : vector_holder_(vh) 08645 | {} 08646 | 08647 | ~vector_size_node() exprtk_override 08648 | { 08649 | assert(valid()); 08650 | } 08651 | 08652 | inline T value() const exprtk_override 08653 | { 08654 | assert(vector_holder_); 08655 | return static_cast<T>(vector_holder_->size()); 08656 | } 08657 | 08658 | inline typename expression_node<T>::node_type type() const exprtk_override 08659 | { 08660 | return expression_node<T>::e_vecsize; 08661 | } 08662 | 08663 | inline bool valid() const exprtk_override 08664 | { 08665 | return vector_holder_ && vector_holder_->size(); 08666 | } 08667 | 08668 | inline vector_holder_t* vec_holder() 08669 | { 08670 | return vector_holder_; 08671 | } 08672 | 08673 | private: 08674 | 08675 | vector_holder_t* vector_holder_; 08676 | }; 08677 | 08678 | template <typename T> 08679 | class vector_elem_node exprtk_final 08680 | : public expression_node<T> 08681 | , public ivariable <T> 08682 | { 08683 | public: 08684 | 08685 | typedef expression_node<T>* expression_ptr; 08686 | typedef vector_holder<T> vector_holder_t; 08687 | typedef vector_holder_t* vector_holder_ptr; 08688 | typedef std::pair<expression_ptr,bool> branch_t; 08689 | 08690 | vector_elem_node(expression_ptr vec_node, 08691 | expression_ptr index, 08692 | vector_holder_ptr vec_holder) 08693 | : vector_holder_(vec_holder) 08694 | , vector_base_((*vec_holder)[0]) 08695 | { 08696 | construct_branch_pair(vector_node_, vec_node); 08697 | construct_branch_pair(index_ , index ); 08698 | assert(valid()); 08699 | } 08700 | 08701 | inline T value() const exprtk_override 08702 | { 08703 | return *access_vector(); 08704 | } 08705 | 08706 | inline T& ref() exprtk_override 08707 | { 08708 | return *access_vector(); 08709 | } 08710 | 08711 | inline const T& ref() const exprtk_override 08712 | { 08713 | return *access_vector(); 08714 | } 08715 | 08716 | inline typename expression_node<T>::node_type type() const exprtk_override 08717 | { 08718 | return expression_node<T>::e_vecelem; 08719 | } 08720 | 08721 | inline bool valid() const exprtk_override 08722 | { 08723 | return 08724 | vector_holder_ && 08725 | index_.first && 08726 | vector_node_.first && 08727 | index_.first->valid() && 08728 | vector_node_.first->valid(); 08729 | } 08730 | 08731 | inline vector_holder_t& vec_holder() 08732 | { 08733 | return (*vector_holder_); 08734 | } 08735 | 08736 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 08737 | { 08738 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 08739 | expression_node<T>::ndb_t::collect(index_ , node_delete_list); 08740 | } 08741 | 08742 | std::size_t node_depth() const exprtk_override 08743 | { 08744 | return expression_node<T>::ndb_t::compute_node_depth 08745 | (vector_node_, index_); 08746 | } 08747 | 08748 | private: 08749 | 08750 | inline T* access_vector() const 08751 | { 08752 | vector_node_.first->value(); 08753 | return (vector_base_ + details::numeric::to_uint64(index_.first->value())); 08754 | } 08755 | 08756 | vector_holder_ptr vector_holder_; 08757 | T* vector_base_; 08758 | branch_t vector_node_; 08759 | branch_t index_; 08760 | }; 08761 | 08762 | template <typename T> 08763 | class vector_celem_node exprtk_final 08764 | : public expression_node<T> 08765 | , public ivariable <T> 08766 | { 08767 | public: 08768 | 08769 | typedef expression_node<T>* expression_ptr; 08770 | typedef vector_holder<T> vector_holder_t; 08771 | typedef vector_holder_t* vector_holder_ptr; 08772 | typedef std::pair<expression_ptr,bool> branch_t; 08773 | 08774 | vector_celem_node(expression_ptr vec_node, 08775 | const std::size_t index, 08776 | vector_holder_ptr vec_holder) 08777 | : index_(index) 08778 | , vector_holder_(vec_holder) 08779 | , vector_base_((*vec_holder)[0]) 08780 | { 08781 | construct_branch_pair(vector_node_, vec_node); 08782 | assert(valid()); 08783 | } 08784 | 08785 | inline T value() const exprtk_override 08786 | { 08787 | return *access_vector(); 08788 | } 08789 | 08790 | inline T& ref() exprtk_override 08791 | { 08792 | return *access_vector(); 08793 | } 08794 | 08795 | inline const T& ref() const exprtk_override 08796 | { 08797 | return *access_vector(); 08798 | } 08799 | 08800 | inline typename expression_node<T>::node_type type() const exprtk_override 08801 | { 08802 | return expression_node<T>::e_veccelem; 08803 | } 08804 | 08805 | inline bool valid() const exprtk_override 08806 | { 08807 | return 08808 | vector_holder_ && 08809 | vector_node_.first && 08810 | vector_node_.first->valid(); 08811 | } 08812 | 08813 | inline vector_holder_t& vec_holder() 08814 | { 08815 | return (*vector_holder_); 08816 | } 08817 | 08818 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 08819 | { 08820 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 08821 | } 08822 | 08823 | std::size_t node_depth() const exprtk_override 08824 | { 08825 | return expression_node<T>::ndb_t::compute_node_depth(vector_node_); 08826 | } 08827 | 08828 | private: 08829 | 08830 | inline T* access_vector() const 08831 | { 08832 | vector_node_.first->value(); 08833 | return (vector_base_ + index_); 08834 | } 08835 | 08836 | const std::size_t index_; 08837 | vector_holder_ptr vector_holder_; 08838 | T* vector_base_; 08839 | branch_t vector_node_; 08840 | }; 08841 | 08842 | template <typename T> 08843 | class vector_elem_rtc_node exprtk_final 08844 | : public expression_node<T> 08845 | , public ivariable <T> 08846 | { 08847 | public: 08848 | 08849 | typedef expression_node<T>* expression_ptr; 08850 | typedef vector_holder<T> vector_holder_t; 08851 | typedef vector_holder_t* vector_holder_ptr; 08852 | typedef std::pair<expression_ptr,bool> branch_t; 08853 | 08854 | vector_elem_rtc_node(expression_ptr vec_node, 08855 | expression_ptr index, 08856 | vector_holder_ptr vec_holder, 08857 | vector_access_runtime_check_ptr vec_rt_chk) 08858 | : vector_holder_(vec_holder) 08859 | , vector_base_((*vec_holder)[0]) 08860 | , vec_rt_chk_(vec_rt_chk) 08861 | , max_vector_index_(vector_holder_->size() - 1) 08862 | { 08863 | construct_branch_pair(vector_node_, vec_node); 08864 | construct_branch_pair(index_ , index ); 08865 | assert(valid()); 08866 | } 08867 | 08868 | inline T value() const exprtk_override 08869 | { 08870 | return *access_vector(); 08871 | } 08872 | 08873 | inline T& ref() exprtk_override 08874 | { 08875 | return *access_vector(); 08876 | } 08877 | 08878 | inline const T& ref() const exprtk_override 08879 | { 08880 | return *access_vector(); 08881 | } 08882 | 08883 | inline typename expression_node<T>::node_type type() const exprtk_override 08884 | { 08885 | return expression_node<T>::e_vecelemrtc; 08886 | } 08887 | 08888 | inline bool valid() const exprtk_override 08889 | { 08890 | return 08891 | vector_holder_ && 08892 | index_.first && 08893 | vector_node_.first && 08894 | index_.first->valid() && 08895 | vector_node_.first->valid(); 08896 | } 08897 | 08898 | inline vector_holder_t& vec_holder() 08899 | { 08900 | return (*vector_holder_); 08901 | } 08902 | 08903 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 08904 | { 08905 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 08906 | expression_node<T>::ndb_t::collect(index_, node_delete_list); 08907 | } 08908 | 08909 | std::size_t node_depth() const exprtk_override 08910 | { 08911 | return expression_node<T>::ndb_t::compute_node_depth 08912 | (vector_node_, index_); 08913 | } 08914 | 08915 | private: 08916 | 08917 | inline T* access_vector() const 08918 | { 08919 | const _uint64_t index = details::numeric::to_uint64(index_.first->value()); 08920 | vector_node_.first->value(); 08921 | 08922 | if (index <= max_vector_index_) 08923 | { 08924 | return (vector_holder_->data() + index); 08925 | } 08926 | 08927 | assert(vec_rt_chk_); 08928 | 08929 | vector_access_runtime_check::violation_context context; 08930 | context.base_ptr = reinterpret_cast<void*>(vector_base_); 08931 | context.end_ptr = reinterpret_cast<void*>(vector_base_ + vector_holder_->size()); 08932 | context.access_ptr = reinterpret_cast<void*>(vector_base_ + index); 08933 | context.type_size = sizeof(T); 08934 | 08935 | return vec_rt_chk_->handle_runtime_violation(context) ? 08936 | reinterpret_cast<T*>(context.access_ptr) : 08937 | vector_base_ ; 08938 | } 08939 | 08940 | vector_holder_ptr vector_holder_; 08941 | T* vector_base_; 08942 | branch_t vector_node_; 08943 | branch_t index_; 08944 | vector_access_runtime_check_ptr vec_rt_chk_; 08945 | const std::size_t max_vector_index_; 08946 | }; 08947 | 08948 | template <typename T> 08949 | class vector_celem_rtc_node exprtk_final 08950 | : public expression_node<T> 08951 | , public ivariable <T> 08952 | { 08953 | public: 08954 | 08955 | typedef expression_node<T>* expression_ptr; 08956 | typedef vector_holder<T> vector_holder_t; 08957 | typedef vector_holder_t* vector_holder_ptr; 08958 | typedef std::pair<expression_ptr,bool> branch_t; 08959 | 08960 | vector_celem_rtc_node(expression_ptr vec_node, 08961 | const std::size_t index, 08962 | vector_holder_ptr vec_holder, 08963 | vector_access_runtime_check_ptr vec_rt_chk) 08964 | : index_(index) 08965 | , max_vector_index_(vec_holder->size() - 1) 08966 | , vector_holder_(vec_holder) 08967 | , vector_base_((*vec_holder)[0]) 08968 | , vec_rt_chk_(vec_rt_chk) 08969 | { 08970 | construct_branch_pair(vector_node_, vec_node); 08971 | assert(valid()); 08972 | } 08973 | 08974 | inline T value() const exprtk_override 08975 | { 08976 | return *access_vector(); 08977 | } 08978 | 08979 | inline T& ref() exprtk_override 08980 | { 08981 | return *access_vector(); 08982 | } 08983 | 08984 | inline const T& ref() const exprtk_override 08985 | { 08986 | return *access_vector(); 08987 | } 08988 | 08989 | inline typename expression_node<T>::node_type type() const exprtk_override 08990 | { 08991 | return expression_node<T>::e_veccelemrtc; 08992 | } 08993 | 08994 | inline bool valid() const exprtk_override 08995 | { 08996 | return 08997 | vector_holder_ && 08998 | vector_node_.first && 08999 | vector_node_.first->valid(); 09000 | } 09001 | 09002 | inline vector_holder_t& vec_holder() 09003 | { 09004 | return (*vector_holder_); 09005 | } 09006 | 09007 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09008 | { 09009 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 09010 | } 09011 | 09012 | std::size_t node_depth() const exprtk_override 09013 | { 09014 | return expression_node<T>::ndb_t::compute_node_depth(vector_node_); 09015 | } 09016 | 09017 | private: 09018 | 09019 | inline T* access_vector() const 09020 | { 09021 | vector_node_.first->value(); 09022 | 09023 | if (index_ <= max_vector_index_) 09024 | { 09025 | return (vector_holder_->data() + index_); 09026 | } 09027 | 09028 | assert(vec_rt_chk_); 09029 | 09030 | vector_access_runtime_check::violation_context context; 09031 | context.base_ptr = reinterpret_cast<void*>(vector_base_); 09032 | context.end_ptr = reinterpret_cast<void*>(vector_base_ + vector_holder_->size()); 09033 | context.access_ptr = reinterpret_cast<void*>(vector_base_ + index_); 09034 | context.type_size = sizeof(T); 09035 | 09036 | return vec_rt_chk_->handle_runtime_violation(context) ? 09037 | reinterpret_cast<T*>(context.access_ptr) : 09038 | vector_base_ ; 09039 | } 09040 | 09041 | const std::size_t index_; 09042 | const std::size_t max_vector_index_; 09043 | vector_holder_ptr vector_holder_; 09044 | T* vector_base_; 09045 | branch_t vector_node_; 09046 | vector_access_runtime_check_ptr vec_rt_chk_; 09047 | }; 09048 | 09049 | template <typename T> 09050 | class rebasevector_elem_node exprtk_final 09051 | : public expression_node<T> 09052 | , public ivariable <T> 09053 | { 09054 | public: 09055 | 09056 | typedef expression_node<T>* expression_ptr; 09057 | typedef vector_holder<T> vector_holder_t; 09058 | typedef vector_holder_t* vector_holder_ptr; 09059 | typedef vec_data_store<T> vds_t; 09060 | typedef std::pair<expression_ptr,bool> branch_t; 09061 | 09062 | rebasevector_elem_node(expression_ptr vec_node, 09063 | expression_ptr index, 09064 | vector_holder_ptr vec_holder) 09065 | : vector_holder_(vec_holder) 09066 | { 09067 | construct_branch_pair(vector_node_, vec_node); 09068 | construct_branch_pair(index_ , index ); 09069 | assert(valid()); 09070 | } 09071 | 09072 | inline T value() const exprtk_override 09073 | { 09074 | return *access_vector(); 09075 | } 09076 | 09077 | inline T& ref() exprtk_override 09078 | { 09079 | return *access_vector(); 09080 | } 09081 | 09082 | inline const T& ref() const exprtk_override 09083 | { 09084 | return *access_vector(); 09085 | } 09086 | 09087 | inline typename expression_node<T>::node_type type() const exprtk_override 09088 | { 09089 | return expression_node<T>::e_rbvecelem; 09090 | } 09091 | 09092 | inline bool valid() const exprtk_override 09093 | { 09094 | return 09095 | vector_holder_ && 09096 | index_.first && 09097 | vector_node_.first && 09098 | index_.first->valid() && 09099 | vector_node_.first->valid(); 09100 | } 09101 | 09102 | inline vector_holder_t& vec_holder() 09103 | { 09104 | return (*vector_holder_); 09105 | } 09106 | 09107 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09108 | { 09109 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 09110 | expression_node<T>::ndb_t::collect(index_, node_delete_list); 09111 | } 09112 | 09113 | std::size_t node_depth() const exprtk_override 09114 | { 09115 | return expression_node<T>::ndb_t::compute_node_depth 09116 | (vector_node_, index_); 09117 | } 09118 | 09119 | private: 09120 | 09121 | inline T* access_vector() const 09122 | { 09123 | vector_node_.first->value(); 09124 | return (vector_holder_->data() + details::numeric::to_uint64(index_.first->value())); 09125 | } 09126 | 09127 | vector_holder_ptr vector_holder_; 09128 | branch_t vector_node_; 09129 | branch_t index_; 09130 | }; 09131 | 09132 | template <typename T> 09133 | class rebasevector_celem_node exprtk_final 09134 | : public expression_node<T> 09135 | , public ivariable <T> 09136 | { 09137 | public: 09138 | 09139 | typedef expression_node<T>* expression_ptr; 09140 | typedef vector_holder<T> vector_holder_t; 09141 | typedef vector_holder_t* vector_holder_ptr; 09142 | typedef std::pair<expression_ptr,bool> branch_t; 09143 | 09144 | rebasevector_celem_node(expression_ptr vec_node, 09145 | const std::size_t index, 09146 | vector_holder_ptr vec_holder) 09147 | : index_(index) 09148 | , vector_holder_(vec_holder) 09149 | { 09150 | construct_branch_pair(vector_node_, vec_node); 09151 | assert(valid()); 09152 | } 09153 | 09154 | inline T value() const exprtk_override 09155 | { 09156 | vector_node_.first->value(); 09157 | return ref(); 09158 | } 09159 | 09160 | inline T& ref() exprtk_override 09161 | { 09162 | return *(vector_holder_->data() + index_); 09163 | } 09164 | 09165 | inline const T& ref() const exprtk_override 09166 | { 09167 | return *(vector_holder_->data() + index_); 09168 | } 09169 | 09170 | inline typename expression_node<T>::node_type type() const exprtk_override 09171 | { 09172 | return expression_node<T>::e_rbveccelem; 09173 | } 09174 | 09175 | inline bool valid() const exprtk_override 09176 | { 09177 | return 09178 | vector_holder_ && 09179 | vector_node_.first && 09180 | vector_node_.first->valid(); 09181 | } 09182 | 09183 | inline vector_holder_t& vec_holder() 09184 | { 09185 | return (*vector_holder_); 09186 | } 09187 | 09188 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09189 | { 09190 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 09191 | } 09192 | 09193 | std::size_t node_depth() const exprtk_override 09194 | { 09195 | return expression_node<T>::ndb_t::compute_node_depth(vector_node_); 09196 | } 09197 | 09198 | private: 09199 | 09200 | const std::size_t index_; 09201 | vector_holder_ptr vector_holder_; 09202 | branch_t vector_node_; 09203 | }; 09204 | 09205 | template <typename T> 09206 | class rebasevector_elem_rtc_node exprtk_final 09207 | : public expression_node<T> 09208 | , public ivariable <T> 09209 | { 09210 | public: 09211 | 09212 | typedef expression_node<T>* expression_ptr; 09213 | typedef vector_holder<T> vector_holder_t; 09214 | typedef vector_holder_t* vector_holder_ptr; 09215 | typedef std::pair<expression_ptr,bool> branch_t; 09216 | 09217 | rebasevector_elem_rtc_node(expression_ptr vec_node, 09218 | expression_ptr index, 09219 | vector_holder_ptr vec_holder, 09220 | vector_access_runtime_check_ptr vec_rt_chk) 09221 | : vector_holder_(vec_holder) 09222 | , vec_rt_chk_(vec_rt_chk) 09223 | { 09224 | construct_branch_pair(vector_node_, vec_node); 09225 | construct_branch_pair(index_ , index ); 09226 | assert(valid()); 09227 | } 09228 | 09229 | inline T value() const exprtk_override 09230 | { 09231 | return *access_vector(); 09232 | } 09233 | 09234 | inline T& ref() exprtk_override 09235 | { 09236 | return *access_vector(); 09237 | } 09238 | 09239 | inline const T& ref() const exprtk_override 09240 | { 09241 | return *access_vector(); 09242 | } 09243 | 09244 | inline typename expression_node<T>::node_type type() const exprtk_override 09245 | { 09246 | return expression_node<T>::e_rbvecelemrtc; 09247 | } 09248 | 09249 | inline bool valid() const exprtk_override 09250 | { 09251 | return 09252 | vector_holder_ && 09253 | index_.first && 09254 | vector_node_.first && 09255 | index_.first->valid() && 09256 | vector_node_.first->valid(); 09257 | } 09258 | 09259 | inline vector_holder_t& vec_holder() 09260 | { 09261 | return (*vector_holder_); 09262 | } 09263 | 09264 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09265 | { 09266 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 09267 | expression_node<T>::ndb_t::collect(index_ , node_delete_list); 09268 | } 09269 | 09270 | std::size_t node_depth() const exprtk_override 09271 | { 09272 | return expression_node<T>::ndb_t::compute_node_depth 09273 | (vector_node_, index_); 09274 | } 09275 | 09276 | private: 09277 | 09278 | inline T* access_vector() const 09279 | { 09280 | vector_node_.first->value(); 09281 | const _uint64_t index = details::numeric::to_uint64(index_.first->value()); 09282 | 09283 | if (index <= (vector_holder_->size() - 1)) 09284 | { 09285 | return (vector_holder_->data() + index); 09286 | } 09287 | 09288 | assert(vec_rt_chk_); 09289 | 09290 | vector_access_runtime_check::violation_context context; 09291 | context.base_ptr = reinterpret_cast<void*>(vector_holder_->data()); 09292 | context.end_ptr = reinterpret_cast<void*>(vector_holder_->data() + vector_holder_->size()); 09293 | context.access_ptr = reinterpret_cast<void*>(vector_holder_->data() + index); 09294 | context.type_size = sizeof(T); 09295 | 09296 | return vec_rt_chk_->handle_runtime_violation(context) ? 09297 | reinterpret_cast<T*>(context.access_ptr) : 09298 | vector_holder_->data() ; 09299 | } 09300 | 09301 | vector_holder_ptr vector_holder_; 09302 | branch_t vector_node_; 09303 | branch_t index_; 09304 | vector_access_runtime_check_ptr vec_rt_chk_; 09305 | }; 09306 | 09307 | template <typename T> 09308 | class rebasevector_celem_rtc_node exprtk_final 09309 | : public expression_node<T> 09310 | , public ivariable <T> 09311 | { 09312 | public: 09313 | 09314 | typedef expression_node<T>* expression_ptr; 09315 | typedef vector_holder<T> vector_holder_t; 09316 | typedef vector_holder_t* vector_holder_ptr; 09317 | typedef std::pair<expression_ptr,bool> branch_t; 09318 | 09319 | rebasevector_celem_rtc_node(expression_ptr vec_node, 09320 | const std::size_t index, 09321 | vector_holder_ptr vec_holder, 09322 | vector_access_runtime_check_ptr vec_rt_chk) 09323 | : index_(index) 09324 | , vector_holder_(vec_holder) 09325 | , vector_base_((*vec_holder)[0]) 09326 | , vec_rt_chk_(vec_rt_chk) 09327 | { 09328 | construct_branch_pair(vector_node_, vec_node); 09329 | assert(valid()); 09330 | } 09331 | 09332 | inline T value() const exprtk_override 09333 | { 09334 | return *access_vector(); 09335 | } 09336 | 09337 | inline T& ref() exprtk_override 09338 | { 09339 | return *access_vector(); 09340 | } 09341 | 09342 | inline const T& ref() const exprtk_override 09343 | { 09344 | return *access_vector(); 09345 | } 09346 | 09347 | inline typename expression_node<T>::node_type type() const exprtk_override 09348 | { 09349 | return expression_node<T>::e_rbveccelemrtc; 09350 | } 09351 | 09352 | inline bool valid() const exprtk_override 09353 | { 09354 | return 09355 | vector_holder_ && 09356 | vector_node_.first && 09357 | vector_node_.first->valid(); 09358 | } 09359 | 09360 | inline vector_holder_t& vec_holder() 09361 | { 09362 | return (*vector_holder_); 09363 | } 09364 | 09365 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09366 | { 09367 | expression_node<T>::ndb_t::collect(vector_node_, node_delete_list); 09368 | } 09369 | 09370 | std::size_t node_depth() const exprtk_override 09371 | { 09372 | return expression_node<T>::ndb_t::compute_node_depth(vector_node_); 09373 | } 09374 | 09375 | private: 09376 | 09377 | inline T* access_vector() const 09378 | { 09379 | vector_node_.first->value(); 09380 | 09381 | if (index_ <= vector_holder_->size() - 1) 09382 | { 09383 | return (vector_holder_->data() + index_); 09384 | } 09385 | 09386 | assert(vec_rt_chk_); 09387 | 09388 | vector_access_runtime_check::violation_context context; 09389 | context.base_ptr = reinterpret_cast<void*>(vector_base_); 09390 | context.end_ptr = reinterpret_cast<void*>(vector_base_ + vector_holder_->size()); 09391 | context.access_ptr = reinterpret_cast<void*>(vector_base_ + index_); 09392 | context.type_size = sizeof(T); 09393 | 09394 | return vec_rt_chk_->handle_runtime_violation(context) ? 09395 | reinterpret_cast<T*>(context.access_ptr) : 09396 | vector_base_ ; 09397 | } 09398 | 09399 | const std::size_t index_; 09400 | vector_holder_ptr vector_holder_; 09401 | T* vector_base_; 09402 | branch_t vector_node_; 09403 | vector_access_runtime_check_ptr vec_rt_chk_; 09404 | }; 09405 | 09406 | template <typename T> 09407 | class vector_initialisation_node exprtk_final : public expression_node<T> 09408 | { 09409 | public: 09410 | 09411 | typedef expression_node<T>* expression_ptr; 09412 | 09413 | vector_initialisation_node(T* vector_base, 09414 | const std::size_t& size, 09415 | const std::vector<expression_ptr>& initialiser_list, 09416 | const bool single_value_initialse) 09417 | : vector_base_(vector_base) 09418 | , initialiser_list_(initialiser_list) 09419 | , size_(size) 09420 | , single_value_initialse_(single_value_initialse) 09421 | , zero_value_initialse_(false) 09422 | , const_nonzero_literal_value_initialse_(false) 09423 | , single_initialiser_value_(T(0)) 09424 | { 09425 | if (single_value_initialse_) 09426 | { 09427 | if (initialiser_list_.empty()) 09428 | zero_value_initialse_ = true; 09429 | else if ( 09430 | (initialiser_list_.size() == 1) && 09431 | details::is_constant_node(initialiser_list_[0]) && 09432 | (T(0) == initialiser_list_[0]->value()) 09433 | ) 09434 | { 09435 | zero_value_initialse_ = true; 09436 | } 09437 | else 09438 | { 09439 | assert(initialiser_list_.size() == 1); 09440 | 09441 | if (details::is_constant_node(initialiser_list_[0])) 09442 | { 09443 | const_nonzero_literal_value_initialse_ = true; 09444 | single_initialiser_value_ = initialiser_list_[0]->value(); 09445 | assert(T(0) != single_initialiser_value_); 09446 | } 09447 | } 09448 | } 09449 | } 09450 | 09451 | inline T value() const exprtk_override 09452 | { 09453 | if (single_value_initialse_) 09454 | { 09455 | if (zero_value_initialse_) 09456 | { 09457 | details::set_zero_value(vector_base_, size_); 09458 | } 09459 | else if (const_nonzero_literal_value_initialse_) 09460 | { 09461 | for (std::size_t i = 0; i < size_; ++i) 09462 | { 09463 | *(vector_base_ + i) = single_initialiser_value_; 09464 | } 09465 | } 09466 | else 09467 | { 09468 | for (std::size_t i = 0; i < size_; ++i) 09469 | { 09470 | *(vector_base_ + i) = initialiser_list_[0]->value(); 09471 | } 09472 | } 09473 | } 09474 | else 09475 | { 09476 | const std::size_t initialiser_list_size = initialiser_list_.size(); 09477 | 09478 | for (std::size_t i = 0; i < initialiser_list_size; ++i) 09479 | { 09480 | *(vector_base_ + i) = initialiser_list_[i]->value(); 09481 | } 09482 | 09483 | if (initialiser_list_size < size_) 09484 | { 09485 | details::set_zero_value( 09486 | vector_base_ + initialiser_list_size, 09487 | (size_ - initialiser_list_size)); 09488 | } 09489 | } 09490 | 09491 | return *(vector_base_); 09492 | } 09493 | 09494 | inline typename expression_node<T>::node_type type() const exprtk_override 09495 | { 09496 | return expression_node<T>::e_vecinit; 09497 | } 09498 | 09499 | inline bool valid() const exprtk_override 09500 | { 09501 | return vector_base_; 09502 | } 09503 | 09504 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09505 | { 09506 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09507 | } 09508 | 09509 | std::size_t node_depth() const exprtk_override 09510 | { 09511 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09512 | } 09513 | 09514 | private: 09515 | 09516 | vector_initialisation_node(const vector_initialisation_node<T>&) exprtk_delete; 09517 | vector_initialisation_node<T>& operator=(const vector_initialisation_node<T>&) exprtk_delete; 09518 | 09519 | mutable T* vector_base_; 09520 | std::vector<expression_ptr> initialiser_list_; 09521 | const std::size_t size_; 09522 | const bool single_value_initialse_; 09523 | bool zero_value_initialse_; 09524 | bool const_nonzero_literal_value_initialse_; 09525 | T single_initialiser_value_; 09526 | }; 09527 | 09528 | template <typename T> 09529 | class vector_init_zero_value_node exprtk_final : public expression_node<T> 09530 | { 09531 | public: 09532 | 09533 | typedef expression_node<T>* expression_ptr; 09534 | 09535 | vector_init_zero_value_node(T* vector_base, 09536 | const std::size_t& size, 09537 | const std::vector<expression_ptr>& initialiser_list) 09538 | : vector_base_(vector_base) 09539 | , size_(size) 09540 | , initialiser_list_(initialiser_list) 09541 | {} 09542 | 09543 | inline T value() const exprtk_override 09544 | { 09545 | details::set_zero_value(vector_base_, size_); 09546 | return *(vector_base_); 09547 | } 09548 | 09549 | inline typename expression_node<T>::node_type type() const exprtk_override 09550 | { 09551 | return expression_node<T>::e_vecinit; 09552 | } 09553 | 09554 | inline bool valid() const exprtk_override 09555 | { 09556 | return vector_base_; 09557 | } 09558 | 09559 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09560 | { 09561 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09562 | } 09563 | 09564 | std::size_t node_depth() const exprtk_override 09565 | { 09566 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09567 | } 09568 | 09569 | private: 09570 | 09571 | vector_init_zero_value_node(const vector_init_zero_value_node<T>&) exprtk_delete; 09572 | vector_init_zero_value_node<T>& operator=(const vector_init_zero_value_node<T>&) exprtk_delete; 09573 | 09574 | mutable T* vector_base_; 09575 | const std::size_t size_; 09576 | std::vector<expression_ptr> initialiser_list_; 09577 | }; 09578 | 09579 | template <typename T> 09580 | class vector_init_single_constvalue_node exprtk_final : public expression_node<T> 09581 | { 09582 | public: 09583 | 09584 | typedef expression_node<T>* expression_ptr; 09585 | 09586 | vector_init_single_constvalue_node(T* vector_base, 09587 | const std::size_t& size, 09588 | const std::vector<expression_ptr>& initialiser_list) 09589 | : vector_base_(vector_base) 09590 | , size_(size) 09591 | , initialiser_list_(initialiser_list) 09592 | { 09593 | single_initialiser_value_ = initialiser_list_[0]->value(); 09594 | assert(valid()); 09595 | } 09596 | 09597 | inline T value() const exprtk_override 09598 | { 09599 | for (std::size_t i = 0; i < size_; ++i) 09600 | { 09601 | *(vector_base_ + i) = single_initialiser_value_; 09602 | } 09603 | 09604 | return *(vector_base_); 09605 | } 09606 | 09607 | inline typename expression_node<T>::node_type type() const exprtk_override 09608 | { 09609 | return expression_node<T>::e_vecinit; 09610 | } 09611 | 09612 | inline bool valid() const exprtk_override 09613 | { 09614 | return vector_base_ && 09615 | (initialiser_list_.size() == 1) && 09616 | (details::is_constant_node(initialiser_list_[0])) && 09617 | (single_initialiser_value_ != T(0)); 09618 | } 09619 | 09620 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09621 | { 09622 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09623 | } 09624 | 09625 | std::size_t node_depth() const exprtk_override 09626 | { 09627 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09628 | } 09629 | 09630 | private: 09631 | 09632 | vector_init_single_constvalue_node(const vector_init_single_constvalue_node<T>&) exprtk_delete; 09633 | vector_init_single_constvalue_node<T>& operator=(const vector_init_single_constvalue_node<T>&) exprtk_delete; 09634 | 09635 | mutable T* vector_base_; 09636 | const std::size_t size_; 09637 | std::vector<expression_ptr> initialiser_list_; 09638 | T single_initialiser_value_; 09639 | }; 09640 | 09641 | template <typename T> 09642 | class vector_init_single_value_node exprtk_final : public expression_node<T> 09643 | { 09644 | public: 09645 | 09646 | typedef expression_node<T>* expression_ptr; 09647 | 09648 | vector_init_single_value_node(T* vector_base, 09649 | const std::size_t& size, 09650 | const std::vector<expression_ptr>& initialiser_list) 09651 | : vector_base_(vector_base) 09652 | , size_(size) 09653 | , initialiser_list_(initialiser_list) 09654 | { 09655 | assert(valid()); 09656 | } 09657 | 09658 | inline T value() const exprtk_override 09659 | { 09660 | expression_node<T>& node = *initialiser_list_[0]; 09661 | 09662 | for (std::size_t i = 0; i < size_; ++i) 09663 | { 09664 | *(vector_base_ + i) = node.value(); 09665 | } 09666 | 09667 | return *(vector_base_); 09668 | } 09669 | 09670 | inline typename expression_node<T>::node_type type() const exprtk_override 09671 | { 09672 | return expression_node<T>::e_vecinit; 09673 | } 09674 | 09675 | inline bool valid() const exprtk_override 09676 | { 09677 | return vector_base_ && 09678 | (initialiser_list_.size() == 1) && 09679 | !details::is_constant_node(initialiser_list_[0]); 09680 | } 09681 | 09682 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09683 | { 09684 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09685 | } 09686 | 09687 | std::size_t node_depth() const exprtk_override 09688 | { 09689 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09690 | } 09691 | 09692 | private: 09693 | 09694 | vector_init_single_value_node(const vector_init_single_value_node<T>&) exprtk_delete; 09695 | vector_init_single_value_node<T>& operator=(const vector_init_single_value_node<T>&) exprtk_delete; 09696 | 09697 | mutable T* vector_base_; 09698 | const std::size_t size_; 09699 | std::vector<expression_ptr> initialiser_list_; 09700 | }; 09701 | 09702 | template <typename T> 09703 | class vector_init_iota_constconst_node exprtk_final : public expression_node<T> 09704 | { 09705 | public: 09706 | 09707 | typedef expression_node<T>* expression_ptr; 09708 | 09709 | vector_init_iota_constconst_node(T* vector_base, 09710 | const std::size_t& size, 09711 | const std::vector<expression_ptr>& initialiser_list) 09712 | : vector_base_(vector_base) 09713 | , size_(size) 09714 | , initialiser_list_(initialiser_list) 09715 | { 09716 | base_value_ = initialiser_list_[0]->value(); 09717 | increment_value_ = initialiser_list_[1]->value(); 09718 | 09719 | assert(valid()); 09720 | } 09721 | 09722 | inline T value() const exprtk_override 09723 | { 09724 | T value = base_value_; 09725 | 09726 | for (std::size_t i = 0; i < size_; ++i, value += increment_value_) 09727 | { 09728 | *(vector_base_ + i) = value; 09729 | } 09730 | 09731 | return *(vector_base_); 09732 | } 09733 | 09734 | inline typename expression_node<T>::node_type type() const exprtk_override 09735 | { 09736 | return expression_node<T>::e_vecinit; 09737 | } 09738 | 09739 | inline bool valid() const exprtk_override 09740 | { 09741 | return vector_base_ && 09742 | (initialiser_list_.size() == 2) && 09743 | (details::is_constant_node(initialiser_list_[0])) && 09744 | (details::is_constant_node(initialiser_list_[1])) ; 09745 | } 09746 | 09747 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09748 | { 09749 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09750 | } 09751 | 09752 | std::size_t node_depth() const exprtk_override 09753 | { 09754 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09755 | } 09756 | 09757 | private: 09758 | 09759 | vector_init_iota_constconst_node(const vector_init_iota_constconst_node<T>&) exprtk_delete; 09760 | vector_init_iota_constconst_node<T>& operator=(const vector_init_iota_constconst_node<T>&) exprtk_delete; 09761 | 09762 | mutable T* vector_base_; 09763 | const std::size_t size_; 09764 | std::vector<expression_ptr> initialiser_list_; 09765 | T base_value_; 09766 | T increment_value_; 09767 | }; 09768 | 09769 | template <typename T> 09770 | class vector_init_iota_constnconst_node exprtk_final : public expression_node<T> 09771 | { 09772 | public: 09773 | 09774 | typedef expression_node<T>* expression_ptr; 09775 | 09776 | vector_init_iota_constnconst_node(T* vector_base, 09777 | const std::size_t& size, 09778 | const std::vector<expression_ptr>& initialiser_list) 09779 | : vector_base_(vector_base) 09780 | , size_(size) 09781 | , initialiser_list_(initialiser_list) 09782 | { 09783 | assert(valid()); 09784 | base_value_ = initialiser_list_[0]->value(); 09785 | } 09786 | 09787 | inline T value() const exprtk_override 09788 | { 09789 | T value = base_value_; 09790 | expression_node<T>& increment = *initialiser_list_[1]; 09791 | 09792 | for (std::size_t i = 0; i < size_; ++i, value += increment.value()) 09793 | { 09794 | *(vector_base_ + i) = value; 09795 | } 09796 | 09797 | return *(vector_base_); 09798 | } 09799 | 09800 | inline typename expression_node<T>::node_type type() const exprtk_override 09801 | { 09802 | return expression_node<T>::e_vecinit; 09803 | } 09804 | 09805 | inline bool valid() const exprtk_override 09806 | { 09807 | return vector_base_ && 09808 | (initialiser_list_.size() == 2) && 09809 | ( details::is_constant_node(initialiser_list_[0])) && 09810 | (!details::is_constant_node(initialiser_list_[1])); 09811 | } 09812 | 09813 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09814 | { 09815 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09816 | } 09817 | 09818 | std::size_t node_depth() const exprtk_override 09819 | { 09820 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09821 | } 09822 | 09823 | private: 09824 | 09825 | vector_init_iota_constnconst_node(const vector_init_iota_constnconst_node<T>&) exprtk_delete; 09826 | vector_init_iota_constnconst_node<T>& operator=(const vector_init_iota_constnconst_node<T>&) exprtk_delete; 09827 | 09828 | mutable T* vector_base_; 09829 | const std::size_t size_; 09830 | std::vector<expression_ptr> initialiser_list_; 09831 | T base_value_; 09832 | }; 09833 | 09834 | template <typename T> 09835 | class vector_init_iota_nconstconst_node exprtk_final : public expression_node<T> 09836 | { 09837 | public: 09838 | 09839 | typedef expression_node<T>* expression_ptr; 09840 | 09841 | vector_init_iota_nconstconst_node(T* vector_base, 09842 | const std::size_t& size, 09843 | const std::vector<expression_ptr>& initialiser_list) 09844 | : vector_base_(vector_base) 09845 | , size_(size) 09846 | , initialiser_list_(initialiser_list) 09847 | { 09848 | assert(valid()); 09849 | } 09850 | 09851 | inline T value() const exprtk_override 09852 | { 09853 | T value = initialiser_list_[0]->value(); 09854 | const T increment = initialiser_list_[1]->value(); 09855 | 09856 | for (std::size_t i = 0; i < size_; ++i, value += increment) 09857 | { 09858 | *(vector_base_ + i) = value; 09859 | } 09860 | 09861 | return *(vector_base_); 09862 | } 09863 | 09864 | inline typename expression_node<T>::node_type type() const exprtk_override 09865 | { 09866 | return expression_node<T>::e_vecinit; 09867 | } 09868 | 09869 | inline bool valid() const exprtk_override 09870 | { 09871 | return vector_base_ && 09872 | (initialiser_list_.size() == 2) && 09873 | (!details::is_constant_node(initialiser_list_[0])) && 09874 | (details::is_constant_node(initialiser_list_[1])); 09875 | } 09876 | 09877 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09878 | { 09879 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09880 | } 09881 | 09882 | std::size_t node_depth() const exprtk_override 09883 | { 09884 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09885 | } 09886 | 09887 | private: 09888 | 09889 | vector_init_iota_nconstconst_node(const vector_init_iota_nconstconst_node<T>&) exprtk_delete; 09890 | vector_init_iota_nconstconst_node<T>& operator=(const vector_init_iota_nconstconst_node<T>&) exprtk_delete; 09891 | 09892 | mutable T* vector_base_; 09893 | const std::size_t size_; 09894 | std::vector<expression_ptr> initialiser_list_; 09895 | }; 09896 | 09897 | template <typename T> 09898 | class vector_init_iota_nconstnconst_node exprtk_final : public expression_node<T> 09899 | { 09900 | public: 09901 | 09902 | typedef expression_node<T>* expression_ptr; 09903 | 09904 | vector_init_iota_nconstnconst_node(T* vector_base, 09905 | const std::size_t& size, 09906 | const std::vector<expression_ptr>& initialiser_list) 09907 | : vector_base_(vector_base) 09908 | , size_(size) 09909 | , initialiser_list_(initialiser_list) 09910 | { 09911 | assert(valid()); 09912 | } 09913 | 09914 | inline T value() const exprtk_override 09915 | { 09916 | T value = initialiser_list_[0]->value(); 09917 | expression_node<T>& increment = *initialiser_list_[1]; 09918 | 09919 | for (std::size_t i = 0; i < size_; ++i, value += increment.value()) 09920 | { 09921 | *(vector_base_ + i) = value; 09922 | } 09923 | 09924 | return *(vector_base_); 09925 | } 09926 | 09927 | inline typename expression_node<T>::node_type type() const exprtk_override 09928 | { 09929 | return expression_node<T>::e_vecinit; 09930 | } 09931 | 09932 | inline bool valid() const exprtk_override 09933 | { 09934 | return vector_base_ && 09935 | (initialiser_list_.size() == 2) && 09936 | (!details::is_constant_node(initialiser_list_[0])) && 09937 | (!details::is_constant_node(initialiser_list_[1])); 09938 | } 09939 | 09940 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 09941 | { 09942 | expression_node<T>::ndb_t::collect(initialiser_list_, node_delete_list); 09943 | } 09944 | 09945 | std::size_t node_depth() const exprtk_override 09946 | { 09947 | return expression_node<T>::ndb_t::compute_node_depth(initialiser_list_); 09948 | } 09949 | 09950 | private: 09951 | 09952 | vector_init_iota_nconstnconst_node(const vector_init_iota_nconstnconst_node<T>&) exprtk_delete; 09953 | vector_init_iota_nconstnconst_node<T>& operator=(const vector_init_iota_nconstnconst_node<T>&) exprtk_delete; 09954 | 09955 | mutable T* vector_base_; 09956 | const std::size_t size_; 09957 | std::vector<expression_ptr> initialiser_list_; 09958 | }; 09959 | 09960 | template <typename T> 09961 | class swap_node exprtk_final : public expression_node<T> 09962 | { 09963 | public: 09964 | 09965 | typedef expression_node<T>* expression_ptr; 09966 | typedef variable_node<T>* variable_node_ptr; 09967 | 09968 | swap_node(variable_node_ptr var0, variable_node_ptr var1) 09969 | : var0_(var0) 09970 | , var1_(var1) 09971 | {} 09972 | 09973 | inline T value() const exprtk_override 09974 | { 09975 | std::swap(var0_->ref(),var1_->ref()); 09976 | return var1_->ref(); 09977 | } 09978 | 09979 | inline typename expression_node<T>::node_type type() const exprtk_override 09980 | { 09981 | return expression_node<T>::e_swap; 09982 | } 09983 | 09984 | private: 09985 | 09986 | variable_node_ptr var0_; 09987 | variable_node_ptr var1_; 09988 | }; 09989 | 09990 | template <typename T> 09991 | class swap_generic_node exprtk_final : public binary_node<T> 09992 | { 09993 | public: 09994 | 09995 | typedef expression_node<T>* expression_ptr; 09996 | typedef ivariable<T>* ivariable_ptr; 09997 | 09998 | swap_generic_node(expression_ptr var0, expression_ptr var1) 09999 | : binary_node<T>(details::e_swap, var0, var1) 10000 | , var0_(dynamic_cast<ivariable_ptr>(var0)) 10001 | , var1_(dynamic_cast<ivariable_ptr>(var1)) 10002 | {} 10003 | 10004 | inline T value() const exprtk_override 10005 | { 10006 | std::swap(var0_->ref(),var1_->ref()); 10007 | return var1_->ref(); 10008 | } 10009 | 10010 | inline typename expression_node<T>::node_type type() const exprtk_override 10011 | { 10012 | return expression_node<T>::e_swap; 10013 | } 10014 | 10015 | private: 10016 | 10017 | ivariable_ptr var0_; 10018 | ivariable_ptr var1_; 10019 | }; 10020 | 10021 | template <typename T> 10022 | class swap_vecvec_node exprtk_final 10023 | : public binary_node <T> 10024 | , public vector_interface<T> 10025 | { 10026 | public: 10027 | 10028 | typedef expression_node<T>* expression_ptr; 10029 | typedef vector_node <T>* vector_node_ptr; 10030 | typedef vec_data_store <T> vds_t; 10031 | 10032 | using binary_node<T>::branch; 10033 | 10034 | swap_vecvec_node(expression_ptr branch0, 10035 | expression_ptr branch1) 10036 | : binary_node<T>(details::e_swap, branch0, branch1) 10037 | , vec0_node_ptr_(0) 10038 | , vec1_node_ptr_(0) 10039 | , initialised_ (false) 10040 | { 10041 | if (is_ivector_node(branch(0))) 10042 | { 10043 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 10044 | 10045 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(0)))) 10046 | { 10047 | vec0_node_ptr_ = vi->vec(); 10048 | vds() = vi->vds(); 10049 | } 10050 | } 10051 | 10052 | if (is_ivector_node(branch(1))) 10053 | { 10054 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 10055 | 10056 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(1)))) 10057 | { 10058 | vec1_node_ptr_ = vi->vec(); 10059 | } 10060 | } 10061 | 10062 | if (vec0_node_ptr_ && vec1_node_ptr_) 10063 | { 10064 | initialised_ = size() <= base_size(); 10065 | } 10066 | 10067 | assert(valid()); 10068 | } 10069 | 10070 | inline T value() const exprtk_override 10071 | { 10072 | binary_node<T>::branch(0)->value(); 10073 | binary_node<T>::branch(1)->value(); 10074 | 10075 | T* vec0 = vec0_node_ptr_->vds().data(); 10076 | T* vec1 = vec1_node_ptr_->vds().data(); 10077 | 10078 | assert(size() <= base_size()); 10079 | const std::size_t n = size(); 10080 | 10081 | for (std::size_t i = 0; i < n; ++i) 10082 | { 10083 | std::swap(vec0[i],vec1[i]); 10084 | } 10085 | 10086 | return vec1_node_ptr_->value(); 10087 | } 10088 | 10089 | vector_node_ptr vec() const exprtk_override 10090 | { 10091 | return vec0_node_ptr_; 10092 | } 10093 | 10094 | vector_node_ptr vec() exprtk_override 10095 | { 10096 | return vec0_node_ptr_; 10097 | } 10098 | 10099 | inline typename expression_node<T>::node_type type() const exprtk_override 10100 | { 10101 | return expression_node<T>::e_vecvecswap; 10102 | } 10103 | 10104 | inline bool valid() const exprtk_override 10105 | { 10106 | return initialised_ && binary_node<T>::valid(); 10107 | } 10108 | 10109 | std::size_t size() const exprtk_override 10110 | { 10111 | return std::min( 10112 | vec0_node_ptr_->vec_holder().size(), 10113 | vec1_node_ptr_->vec_holder().size()); 10114 | } 10115 | 10116 | std::size_t base_size() const exprtk_override 10117 | { 10118 | return std::min( 10119 | vec0_node_ptr_->vec_holder().base_size(), 10120 | vec1_node_ptr_->vec_holder().base_size()); 10121 | } 10122 | 10123 | vds_t& vds() exprtk_override 10124 | { 10125 | return vds_; 10126 | } 10127 | 10128 | const vds_t& vds() const exprtk_override 10129 | { 10130 | return vds_; 10131 | } 10132 | 10133 | private: 10134 | 10135 | vector_node<T>* vec0_node_ptr_; 10136 | vector_node<T>* vec1_node_ptr_; 10137 | bool initialised_; 10138 | vds_t vds_; 10139 | }; 10140 | 10141 | #ifndef exprtk_disable_string_capabilities 10142 | template <typename T> 10143 | class stringvar_node exprtk_final 10144 | : public expression_node <T> 10145 | , public string_base_node<T> 10146 | , public range_interface <T> 10147 | { 10148 | public: 10149 | 10150 | typedef typename range_interface<T>::range_t range_t; 10151 | 10152 | static std::string null_value; 10153 | 10154 | explicit stringvar_node() 10155 | : value_(&null_value) 10156 | {} 10157 | 10158 | explicit stringvar_node(std::string& v) 10159 | : value_(&v) 10160 | { 10161 | rp_.n0_c = std::make_pair<bool,std::size_t>(true,0); 10162 | rp_.n1_c = std::make_pair<bool,std::size_t>(true,v.size()); 10163 | rp_.cache.first = rp_.n0_c.second; 10164 | rp_.cache.second = rp_.n1_c.second; 10165 | } 10166 | 10167 | inline bool operator <(const stringvar_node<T>& v) const 10168 | { 10169 | return this < (&v); 10170 | } 10171 | 10172 | inline T value() const exprtk_override 10173 | { 10174 | rp_.n1_c.second = (*value_).size(); 10175 | rp_.cache.second = rp_.n1_c.second; 10176 | 10177 | return std::numeric_limits<T>::quiet_NaN(); 10178 | } 10179 | 10180 | std::string str() const exprtk_override 10181 | { 10182 | return ref(); 10183 | } 10184 | 10185 | char_cptr base() const exprtk_override 10186 | { 10187 | return &(*value_)[0]; 10188 | } 10189 | 10190 | std::size_t size() const exprtk_override 10191 | { 10192 | return ref().size(); 10193 | } 10194 | 10195 | std::string& ref() 10196 | { 10197 | return (*value_); 10198 | } 10199 | 10200 | const std::string& ref() const 10201 | { 10202 | return (*value_); 10203 | } 10204 | 10205 | range_t& range_ref() exprtk_override 10206 | { 10207 | return rp_; 10208 | } 10209 | 10210 | const range_t& range_ref() const exprtk_override 10211 | { 10212 | return rp_; 10213 | } 10214 | 10215 | inline typename expression_node<T>::node_type type() const exprtk_override 10216 | { 10217 | return expression_node<T>::e_stringvar; 10218 | } 10219 | 10220 | void rebase(std::string& s) 10221 | { 10222 | value_ = &s; 10223 | rp_.n0_c = std::make_pair<bool,std::size_t>(true,0); 10224 | rp_.n1_c = std::make_pair<bool,std::size_t>(true,value_->size() - 1); 10225 | rp_.cache.first = rp_.n0_c.second; 10226 | rp_.cache.second = rp_.n1_c.second; 10227 | } 10228 | 10229 | private: 10230 | 10231 | std::string* value_; 10232 | mutable range_t rp_; 10233 | }; 10234 | 10235 | template <typename T> 10236 | std::string stringvar_node<T>::null_value = std::string(""); 10237 | 10238 | template <typename T> 10239 | class string_range_node exprtk_final 10240 | : public expression_node <T> 10241 | , public string_base_node<T> 10242 | , public range_interface <T> 10243 | { 10244 | public: 10245 | 10246 | typedef typename range_interface<T>::range_t range_t; 10247 | 10248 | static std::string null_value; 10249 | 10250 | explicit string_range_node(std::string& v, const range_t& rp) 10251 | : value_(&v) 10252 | , rp_(rp) 10253 | {} 10254 | 10255 | virtual ~string_range_node() 10256 | { 10257 | rp_.free(); 10258 | } 10259 | 10260 | inline bool operator <(const string_range_node<T>& v) const 10261 | { 10262 | return this < (&v); 10263 | } 10264 | 10265 | inline T value() const exprtk_override 10266 | { 10267 | return std::numeric_limits<T>::quiet_NaN(); 10268 | } 10269 | 10270 | inline std::string str() const exprtk_override 10271 | { 10272 | return (*value_); 10273 | } 10274 | 10275 | char_cptr base() const exprtk_override 10276 | { 10277 | return &(*value_)[0]; 10278 | } 10279 | 10280 | std::size_t size() const exprtk_override 10281 | { 10282 | return ref().size(); 10283 | } 10284 | 10285 | inline range_t range() const 10286 | { 10287 | return rp_; 10288 | } 10289 | 10290 | inline virtual std::string& ref() 10291 | { 10292 | return (*value_); 10293 | } 10294 | 10295 | inline virtual const std::string& ref() const 10296 | { 10297 | return (*value_); 10298 | } 10299 | 10300 | inline range_t& range_ref() exprtk_override 10301 | { 10302 | return rp_; 10303 | } 10304 | 10305 | inline const range_t& range_ref() const exprtk_override 10306 | { 10307 | return rp_; 10308 | } 10309 | 10310 | inline typename expression_node<T>::node_type type() const exprtk_override 10311 | { 10312 | return expression_node<T>::e_stringvarrng; 10313 | } 10314 | 10315 | private: 10316 | 10317 | std::string* value_; 10318 | range_t rp_; 10319 | }; 10320 | 10321 | template <typename T> 10322 | std::string string_range_node<T>::null_value = std::string(""); 10323 | 10324 | template <typename T> 10325 | class const_string_range_node exprtk_final 10326 | : public expression_node <T> 10327 | , public string_base_node<T> 10328 | , public range_interface <T> 10329 | { 10330 | public: 10331 | 10332 | typedef typename range_interface<T>::range_t range_t; 10333 | 10334 | explicit const_string_range_node(const std::string& v, const range_t& rp) 10335 | : value_(v) 10336 | , rp_(rp) 10337 | {} 10338 | 10339 | ~const_string_range_node() exprtk_override 10340 | { 10341 | rp_.free(); 10342 | } 10343 | 10344 | inline T value() const exprtk_override 10345 | { 10346 | return std::numeric_limits<T>::quiet_NaN(); 10347 | } 10348 | 10349 | std::string str() const exprtk_override 10350 | { 10351 | return value_; 10352 | } 10353 | 10354 | char_cptr base() const exprtk_override 10355 | { 10356 | return value_.data(); 10357 | } 10358 | 10359 | std::size_t size() const exprtk_override 10360 | { 10361 | return value_.size(); 10362 | } 10363 | 10364 | range_t range() const 10365 | { 10366 | return rp_; 10367 | } 10368 | 10369 | range_t& range_ref() exprtk_override 10370 | { 10371 | return rp_; 10372 | } 10373 | 10374 | const range_t& range_ref() const exprtk_override 10375 | { 10376 | return rp_; 10377 | } 10378 | 10379 | inline typename expression_node<T>::node_type type() const exprtk_override 10380 | { 10381 | return expression_node<T>::e_cstringvarrng; 10382 | } 10383 | 10384 | private: 10385 | 10386 | const_string_range_node(const const_string_range_node<T>&) exprtk_delete; 10387 | const_string_range_node<T>& operator=(const const_string_range_node<T>&) exprtk_delete; 10388 | 10389 | const std::string value_; 10390 | range_t rp_; 10391 | }; 10392 | 10393 | template <typename T> 10394 | class generic_string_range_node exprtk_final 10395 | : public expression_node <T> 10396 | , public string_base_node<T> 10397 | , public range_interface <T> 10398 | { 10399 | public: 10400 | 10401 | typedef expression_node <T>* expression_ptr; 10402 | typedef stringvar_node <T>* strvar_node_ptr; 10403 | typedef string_base_node<T>* str_base_ptr; 10404 | typedef typename range_interface<T>::range_t range_t; 10405 | typedef range_t* range_ptr; 10406 | typedef range_interface<T> irange_t; 10407 | typedef irange_t* irange_ptr; 10408 | typedef std::pair<expression_ptr,bool> branch_t; 10409 | 10410 | generic_string_range_node(expression_ptr str_branch, const range_t& brange) 10411 | : initialised_(false) 10412 | , str_base_ptr_ (0) 10413 | , str_range_ptr_(0) 10414 | , base_range_(brange) 10415 | { 10416 | range_.n0_c = std::make_pair<bool,std::size_t>(true,0); 10417 | range_.n1_c = std::make_pair<bool,std::size_t>(true,0); 10418 | range_.cache.first = range_.n0_c.second; 10419 | range_.cache.second = range_.n1_c.second; 10420 | 10421 | construct_branch_pair(branch_, str_branch); 10422 | 10423 | if (is_generally_string_node(branch_.first)) 10424 | { 10425 | str_base_ptr_ = dynamic_cast<str_base_ptr>(branch_.first); 10426 | 10427 | if (0 == str_base_ptr_) 10428 | return; 10429 | 10430 | str_range_ptr_ = dynamic_cast<irange_ptr>(branch_.first); 10431 | 10432 | if (0 == str_range_ptr_) 10433 | return; 10434 | } 10435 | 10436 | initialised_ = (str_base_ptr_ && str_range_ptr_); 10437 | assert(valid()); 10438 | } 10439 | 10440 | ~generic_string_range_node() exprtk_override 10441 | { 10442 | base_range_.free(); 10443 | } 10444 | 10445 | inline T value() const exprtk_override 10446 | { 10447 | branch_.first->value(); 10448 | 10449 | std::size_t str_r0 = 0; 10450 | std::size_t str_r1 = 0; 10451 | 10452 | std::size_t r0 = 0; 10453 | std::size_t r1 = 0; 10454 | 10455 | const range_t& range = str_range_ptr_->range_ref(); 10456 | 10457 | const std::size_t base_str_size = str_base_ptr_->size(); 10458 | 10459 | if ( 10460 | range (str_r0, str_r1, base_str_size ) && 10461 | base_range_(r0 , r1 , base_str_size - str_r0) 10462 | ) 10463 | { 10464 | const std::size_t size = r1 - r0; 10465 | 10466 | range_.n1_c.second = size; 10467 | range_.cache.second = range_.n1_c.second; 10468 | 10469 | value_.assign(str_base_ptr_->base() + str_r0 + r0, size); 10470 | } 10471 | 10472 | return std::numeric_limits<T>::quiet_NaN(); 10473 | } 10474 | 10475 | std::string str() const exprtk_override 10476 | { 10477 | return value_; 10478 | } 10479 | 10480 | char_cptr base() const exprtk_override 10481 | { 10482 | return &value_[0]; 10483 | } 10484 | 10485 | std::size_t size() const exprtk_override 10486 | { 10487 | return value_.size(); 10488 | } 10489 | 10490 | range_t& range_ref() exprtk_override 10491 | { 10492 | return range_; 10493 | } 10494 | 10495 | const range_t& range_ref() const exprtk_override 10496 | { 10497 | return range_; 10498 | } 10499 | 10500 | inline typename expression_node<T>::node_type type() const exprtk_override 10501 | { 10502 | return expression_node<T>::e_strgenrange; 10503 | } 10504 | 10505 | inline bool valid() const exprtk_override 10506 | { 10507 | return initialised_ && branch_.first; 10508 | } 10509 | 10510 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 10511 | { 10512 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 10513 | } 10514 | 10515 | std::size_t node_depth() const exprtk_override 10516 | { 10517 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 10518 | } 10519 | 10520 | private: 10521 | 10522 | bool initialised_; 10523 | branch_t branch_; 10524 | str_base_ptr str_base_ptr_; 10525 | irange_ptr str_range_ptr_; 10526 | mutable range_t base_range_; 10527 | mutable range_t range_; 10528 | mutable std::string value_; 10529 | }; 10530 | 10531 | template <typename T> 10532 | class string_concat_node exprtk_final 10533 | : public binary_node <T> 10534 | , public string_base_node<T> 10535 | , public range_interface <T> 10536 | { 10537 | public: 10538 | 10539 | typedef typename range_interface<T>::range_t range_t; 10540 | typedef range_interface<T> irange_t; 10541 | typedef irange_t* irange_ptr; 10542 | typedef range_t* range_ptr; 10543 | typedef expression_node <T>* expression_ptr; 10544 | typedef string_base_node<T>* str_base_ptr; 10545 | 10546 | using binary_node<T>::branch; 10547 | 10548 | string_concat_node(const operator_type& opr, 10549 | expression_ptr branch0, 10550 | expression_ptr branch1) 10551 | : binary_node<T>(opr, branch0, branch1) 10552 | , initialised_(false) 10553 | , str0_base_ptr_ (0) 10554 | , str1_base_ptr_ (0) 10555 | , str0_range_ptr_(0) 10556 | , str1_range_ptr_(0) 10557 | { 10558 | range_.n0_c = std::make_pair<bool,std::size_t>(true,0); 10559 | range_.n1_c = std::make_pair<bool,std::size_t>(true,0); 10560 | 10561 | range_.cache.first = range_.n0_c.second; 10562 | range_.cache.second = range_.n1_c.second; 10563 | 10564 | if (is_generally_string_node(branch(0))) 10565 | { 10566 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 10567 | 10568 | if (0 == str0_base_ptr_) 10569 | return; 10570 | 10571 | str0_range_ptr_ = dynamic_cast<irange_ptr>(branch(0)); 10572 | 10573 | if (0 == str0_range_ptr_) 10574 | return; 10575 | } 10576 | 10577 | if (is_generally_string_node(branch(1))) 10578 | { 10579 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(branch(1)); 10580 | 10581 | if (0 == str1_base_ptr_) 10582 | return; 10583 | 10584 | str1_range_ptr_ = dynamic_cast<irange_ptr>(branch(1)); 10585 | 10586 | if (0 == str1_range_ptr_) 10587 | return; 10588 | } 10589 | 10590 | initialised_ = str0_base_ptr_ && 10591 | str1_base_ptr_ && 10592 | str0_range_ptr_ && 10593 | str1_range_ptr_ ; 10594 | 10595 | assert(valid()); 10596 | } 10597 | 10598 | inline T value() const exprtk_override 10599 | { 10600 | branch(0)->value(); 10601 | branch(1)->value(); 10602 | 10603 | std::size_t str0_r0 = 0; 10604 | std::size_t str0_r1 = 0; 10605 | 10606 | std::size_t str1_r0 = 0; 10607 | std::size_t str1_r1 = 0; 10608 | 10609 | const range_t& range0 = str0_range_ptr_->range_ref(); 10610 | const range_t& range1 = str1_range_ptr_->range_ref(); 10611 | 10612 | if ( 10613 | range0(str0_r0, str0_r1, str0_base_ptr_->size()) && 10614 | range1(str1_r0, str1_r1, str1_base_ptr_->size()) 10615 | ) 10616 | { 10617 | const std::size_t size0 = (str0_r1 - str0_r0); 10618 | const std::size_t size1 = (str1_r1 - str1_r0); 10619 | 10620 | value_.assign(str0_base_ptr_->base() + str0_r0, size0); 10621 | value_.append(str1_base_ptr_->base() + str1_r0, size1); 10622 | 10623 | range_.n1_c.second = value_.size(); 10624 | range_.cache.second = range_.n1_c.second; 10625 | } 10626 | 10627 | return std::numeric_limits<T>::quiet_NaN(); 10628 | } 10629 | 10630 | std::string str() const exprtk_override 10631 | { 10632 | return value_; 10633 | } 10634 | 10635 | char_cptr base() const exprtk_override 10636 | { 10637 | return &value_[0]; 10638 | } 10639 | 10640 | std::size_t size() const exprtk_override 10641 | { 10642 | return value_.size(); 10643 | } 10644 | 10645 | range_t& range_ref() exprtk_override 10646 | { 10647 | return range_; 10648 | } 10649 | 10650 | const range_t& range_ref() const exprtk_override 10651 | { 10652 | return range_; 10653 | } 10654 | 10655 | inline typename expression_node<T>::node_type type() const exprtk_override 10656 | { 10657 | return expression_node<T>::e_strconcat; 10658 | } 10659 | 10660 | inline bool valid() const exprtk_override 10661 | { 10662 | return initialised_ && binary_node<T>::valid(); 10663 | } 10664 | 10665 | private: 10666 | 10667 | bool initialised_; 10668 | str_base_ptr str0_base_ptr_; 10669 | str_base_ptr str1_base_ptr_; 10670 | irange_ptr str0_range_ptr_; 10671 | irange_ptr str1_range_ptr_; 10672 | mutable range_t range_; 10673 | mutable std::string value_; 10674 | }; 10675 | 10676 | template <typename T> 10677 | class swap_string_node exprtk_final 10678 | : public binary_node <T> 10679 | , public string_base_node<T> 10680 | , public range_interface <T> 10681 | { 10682 | public: 10683 | 10684 | typedef typename range_interface<T>::range_t range_t; 10685 | typedef range_t* range_ptr; 10686 | typedef range_interface<T> irange_t; 10687 | typedef irange_t* irange_ptr; 10688 | typedef expression_node <T>* expression_ptr; 10689 | typedef stringvar_node <T>* strvar_node_ptr; 10690 | typedef string_base_node<T>* str_base_ptr; 10691 | 10692 | using binary_node<T>::branch; 10693 | 10694 | swap_string_node(expression_ptr branch0, expression_ptr branch1) 10695 | : binary_node<T>(details::e_swap, branch0, branch1) 10696 | , initialised_(false) 10697 | , str0_node_ptr_(0) 10698 | , str1_node_ptr_(0) 10699 | { 10700 | if (is_string_node(branch(0))) 10701 | { 10702 | str0_node_ptr_ = static_cast<strvar_node_ptr>(branch(0)); 10703 | } 10704 | 10705 | if (is_string_node(branch(1))) 10706 | { 10707 | str1_node_ptr_ = static_cast<strvar_node_ptr>(branch(1)); 10708 | } 10709 | 10710 | initialised_ = (str0_node_ptr_ && str1_node_ptr_); 10711 | assert(valid()); 10712 | } 10713 | 10714 | inline T value() const exprtk_override 10715 | { 10716 | branch(0)->value(); 10717 | branch(1)->value(); 10718 | 10719 | std::swap(str0_node_ptr_->ref(), str1_node_ptr_->ref()); 10720 | 10721 | return std::numeric_limits<T>::quiet_NaN(); 10722 | } 10723 | 10724 | std::string str() const exprtk_override 10725 | { 10726 | return str0_node_ptr_->str(); 10727 | } 10728 | 10729 | char_cptr base() const exprtk_override 10730 | { 10731 | return str0_node_ptr_->base(); 10732 | } 10733 | 10734 | std::size_t size() const exprtk_override 10735 | { 10736 | return str0_node_ptr_->size(); 10737 | } 10738 | 10739 | range_t& range_ref() exprtk_override 10740 | { 10741 | return str0_node_ptr_->range_ref(); 10742 | } 10743 | 10744 | const range_t& range_ref() const exprtk_override 10745 | { 10746 | return str0_node_ptr_->range_ref(); 10747 | } 10748 | 10749 | inline typename expression_node<T>::node_type type() const exprtk_override 10750 | { 10751 | return expression_node<T>::e_strswap; 10752 | } 10753 | 10754 | inline bool valid() const exprtk_override 10755 | { 10756 | return initialised_ && binary_node<T>::valid(); 10757 | } 10758 | 10759 | private: 10760 | 10761 | bool initialised_; 10762 | strvar_node_ptr str0_node_ptr_; 10763 | strvar_node_ptr str1_node_ptr_; 10764 | }; 10765 | 10766 | template <typename T> 10767 | class swap_genstrings_node exprtk_final : public binary_node<T> 10768 | { 10769 | public: 10770 | 10771 | typedef typename range_interface<T>::range_t range_t; 10772 | typedef range_t* range_ptr; 10773 | typedef range_interface<T> irange_t; 10774 | typedef irange_t* irange_ptr; 10775 | typedef expression_node <T>* expression_ptr; 10776 | typedef string_base_node<T>* str_base_ptr; 10777 | 10778 | using binary_node<T>::branch; 10779 | 10780 | swap_genstrings_node(expression_ptr branch0, 10781 | expression_ptr branch1) 10782 | : binary_node<T>(details::e_default, branch0, branch1) 10783 | , str0_base_ptr_ (0) 10784 | , str1_base_ptr_ (0) 10785 | , str0_range_ptr_(0) 10786 | , str1_range_ptr_(0) 10787 | , initialised_(false) 10788 | { 10789 | if (is_generally_string_node(branch(0))) 10790 | { 10791 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 10792 | 10793 | if (0 == str0_base_ptr_) 10794 | return; 10795 | 10796 | irange_ptr range = dynamic_cast<irange_ptr>(branch(0)); 10797 | 10798 | if (0 == range) 10799 | return; 10800 | 10801 | str0_range_ptr_ = &(range->range_ref()); 10802 | } 10803 | 10804 | if (is_generally_string_node(branch(1))) 10805 | { 10806 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(branch(1)); 10807 | 10808 | if (0 == str1_base_ptr_) 10809 | return; 10810 | 10811 | irange_ptr range = dynamic_cast<irange_ptr>(branch(1)); 10812 | 10813 | if (0 == range) 10814 | return; 10815 | 10816 | str1_range_ptr_ = &(range->range_ref()); 10817 | } 10818 | 10819 | initialised_ = str0_base_ptr_ && 10820 | str1_base_ptr_ && 10821 | str0_range_ptr_ && 10822 | str1_range_ptr_ ; 10823 | 10824 | assert(valid()); 10825 | } 10826 | 10827 | inline T value() const exprtk_override 10828 | { 10829 | branch(0)->value(); 10830 | branch(1)->value(); 10831 | 10832 | std::size_t str0_r0 = 0; 10833 | std::size_t str0_r1 = 0; 10834 | 10835 | std::size_t str1_r0 = 0; 10836 | std::size_t str1_r1 = 0; 10837 | 10838 | const range_t& range0 = (*str0_range_ptr_); 10839 | const range_t& range1 = (*str1_range_ptr_); 10840 | 10841 | if ( 10842 | range0(str0_r0, str0_r1, str0_base_ptr_->size()) && 10843 | range1(str1_r0, str1_r1, str1_base_ptr_->size()) 10844 | ) 10845 | { 10846 | const std::size_t size0 = range0.cache_size(); 10847 | const std::size_t size1 = range1.cache_size(); 10848 | const std::size_t max_size = std::min(size0,size1); 10849 | 10850 | char_ptr s0 = const_cast<char_ptr>(str0_base_ptr_->base() + str0_r0); 10851 | char_ptr s1 = const_cast<char_ptr>(str1_base_ptr_->base() + str1_r0); 10852 | 10853 | loop_unroll::details lud(max_size); 10854 | char_cptr upper_bound = s0 + lud.upper_bound; 10855 | 10856 | while (s0 < upper_bound) 10857 | { 10858 | #define exprtk_loop(N) \ 10859 | std::swap(s0[N], s1[N]); \ 10860 | 10861 | exprtk_loop( 0) exprtk_loop( 1) 10862 | exprtk_loop( 2) exprtk_loop( 3) 10863 | #ifndef exprtk_disable_superscalar_unroll 10864 | exprtk_loop( 4) exprtk_loop( 5) 10865 | exprtk_loop( 6) exprtk_loop( 7) 10866 | exprtk_loop( 8) exprtk_loop( 9) 10867 | exprtk_loop(10) exprtk_loop(11) 10868 | exprtk_loop(12) exprtk_loop(13) 10869 | exprtk_loop(14) exprtk_loop(15) 10870 | #endif 10871 | 10872 | s0 += lud.batch_size; 10873 | s1 += lud.batch_size; 10874 | } 10875 | 10876 | int i = 0; 10877 | 10878 | switch (lud.remainder) 10879 | { 10880 | #define case_stmt(N) \ 10881 | case N : { std::swap(s0[i], s1[i]); ++i; } \ 10882 | exprtk_fallthrough \ 10883 | 10884 | #ifndef exprtk_disable_superscalar_unroll 10885 | case_stmt(15) case_stmt(14) 10886 | case_stmt(13) case_stmt(12) 10887 | case_stmt(11) case_stmt(10) 10888 | case_stmt( 9) case_stmt( 8) 10889 | case_stmt( 7) case_stmt( 6) 10890 | case_stmt( 5) case_stmt( 4) 10891 | #endif 10892 | case_stmt( 3) case_stmt( 2) 10893 | case_stmt( 1) 10894 | default: break; 10895 | } 10896 | 10897 | #undef exprtk_loop 10898 | #undef case_stmt 10899 | } 10900 | 10901 | return std::numeric_limits<T>::quiet_NaN(); 10902 | } 10903 | 10904 | inline typename expression_node<T>::node_type type() const exprtk_override 10905 | { 10906 | return expression_node<T>::e_strswap; 10907 | } 10908 | 10909 | inline bool valid() const exprtk_override 10910 | { 10911 | return initialised_ && binary_node<T>::valid(); 10912 | } 10913 | 10914 | private: 10915 | 10916 | swap_genstrings_node(const swap_genstrings_node<T>&) exprtk_delete; 10917 | swap_genstrings_node<T>& operator=(const swap_genstrings_node<T>&) exprtk_delete; 10918 | 10919 | str_base_ptr str0_base_ptr_; 10920 | str_base_ptr str1_base_ptr_; 10921 | range_ptr str0_range_ptr_; 10922 | range_ptr str1_range_ptr_; 10923 | bool initialised_; 10924 | }; 10925 | 10926 | template <typename T> 10927 | class stringvar_size_node exprtk_final : public expression_node<T> 10928 | { 10929 | public: 10930 | 10931 | static const std::string null_value; 10932 | 10933 | explicit stringvar_size_node() 10934 | : value_(&null_value) 10935 | {} 10936 | 10937 | explicit stringvar_size_node(std::string& v) 10938 | : value_(&v) 10939 | {} 10940 | 10941 | inline T value() const exprtk_override 10942 | { 10943 | return T((*value_).size()); 10944 | } 10945 | 10946 | inline typename expression_node<T>::node_type type() const exprtk_override 10947 | { 10948 | return expression_node<T>::e_stringvarsize; 10949 | } 10950 | 10951 | private: 10952 | 10953 | const std::string* value_; 10954 | }; 10955 | 10956 | template <typename T> 10957 | const std::string stringvar_size_node<T>::null_value = std::string(""); 10958 | 10959 | template <typename T> 10960 | class string_size_node exprtk_final : public expression_node<T> 10961 | { 10962 | public: 10963 | 10964 | typedef expression_node <T>* expression_ptr; 10965 | typedef string_base_node<T>* str_base_ptr; 10966 | typedef std::pair<expression_ptr,bool> branch_t; 10967 | 10968 | explicit string_size_node(expression_ptr branch) 10969 | : str_base_ptr_(0) 10970 | { 10971 | construct_branch_pair(branch_, branch); 10972 | 10973 | if (is_generally_string_node(branch_.first)) 10974 | { 10975 | str_base_ptr_ = dynamic_cast<str_base_ptr>(branch_.first); 10976 | } 10977 | 10978 | assert(valid()); 10979 | } 10980 | 10981 | inline T value() const exprtk_override 10982 | { 10983 | branch_.first->value(); 10984 | return T(str_base_ptr_->size()); 10985 | } 10986 | 10987 | inline typename expression_node<T>::node_type type() const exprtk_override 10988 | { 10989 | return expression_node<T>::e_stringsize; 10990 | } 10991 | 10992 | inline bool valid() const exprtk_override 10993 | { 10994 | return str_base_ptr_; 10995 | } 10996 | 10997 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 10998 | { 10999 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 11000 | } 11001 | 11002 | std::size_t node_depth() const exprtk_override 11003 | { 11004 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 11005 | } 11006 | 11007 | private: 11008 | 11009 | branch_t branch_; 11010 | str_base_ptr str_base_ptr_; 11011 | }; 11012 | 11013 | struct asn_assignment 11014 | { 11015 | static inline void execute(std::string& s, char_cptr data, const std::size_t size) 11016 | { s.assign(data,size); } 11017 | }; 11018 | 11019 | struct asn_addassignment 11020 | { 11021 | static inline void execute(std::string& s, char_cptr data, const std::size_t size) 11022 | { s.append(data,size); } 11023 | }; 11024 | 11025 | template <typename T, typename AssignmentProcess = asn_assignment> 11026 | class assignment_string_node exprtk_final 11027 | : public binary_node <T> 11028 | , public string_base_node<T> 11029 | , public range_interface <T> 11030 | { 11031 | public: 11032 | 11033 | typedef typename range_interface<T>::range_t range_t; 11034 | typedef range_t* range_ptr; 11035 | typedef range_interface <T> irange_t; 11036 | typedef irange_t* irange_ptr; 11037 | typedef expression_node <T>* expression_ptr; 11038 | typedef stringvar_node <T>* strvar_node_ptr; 11039 | typedef string_base_node<T>* str_base_ptr; 11040 | 11041 | using binary_node<T>::branch; 11042 | 11043 | assignment_string_node(const operator_type& opr, 11044 | expression_ptr branch0, 11045 | expression_ptr branch1) 11046 | : binary_node<T>(opr, branch0, branch1) 11047 | , initialised_(false) 11048 | , str0_base_ptr_ (0) 11049 | , str1_base_ptr_ (0) 11050 | , str0_node_ptr_ (0) 11051 | , str1_range_ptr_(0) 11052 | { 11053 | if (is_string_node(branch(0))) 11054 | { 11055 | str0_node_ptr_ = static_cast<strvar_node_ptr>(branch(0)); 11056 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 11057 | } 11058 | 11059 | if (is_generally_string_node(branch(1))) 11060 | { 11061 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(branch(1)); 11062 | 11063 | if (0 == str1_base_ptr_) 11064 | return; 11065 | 11066 | irange_ptr range = dynamic_cast<irange_ptr>(branch(1)); 11067 | 11068 | if (0 == range) 11069 | return; 11070 | 11071 | str1_range_ptr_ = &(range->range_ref()); 11072 | } 11073 | 11074 | initialised_ = str0_base_ptr_ && 11075 | str1_base_ptr_ && 11076 | str0_node_ptr_ && 11077 | str1_range_ptr_ ; 11078 | 11079 | assert(valid()); 11080 | } 11081 | 11082 | inline T value() const exprtk_override 11083 | { 11084 | branch(1)->value(); 11085 | 11086 | std::size_t r0 = 0; 11087 | std::size_t r1 = 0; 11088 | 11089 | const range_t& range = (*str1_range_ptr_); 11090 | 11091 | if (range(r0, r1, str1_base_ptr_->size())) 11092 | { 11093 | AssignmentProcess::execute( 11094 | str0_node_ptr_->ref(), 11095 | str1_base_ptr_->base() + r0, (r1 - r0)); 11096 | 11097 | branch(0)->value(); 11098 | } 11099 | 11100 | return std::numeric_limits<T>::quiet_NaN(); 11101 | } 11102 | 11103 | std::string str() const exprtk_override 11104 | { 11105 | return str0_node_ptr_->str(); 11106 | } 11107 | 11108 | char_cptr base() const exprtk_override 11109 | { 11110 | return str0_node_ptr_->base(); 11111 | } 11112 | 11113 | std::size_t size() const exprtk_override 11114 | { 11115 | return str0_node_ptr_->size(); 11116 | } 11117 | 11118 | range_t& range_ref() exprtk_override 11119 | { 11120 | return str0_node_ptr_->range_ref(); 11121 | } 11122 | 11123 | const range_t& range_ref() const exprtk_override 11124 | { 11125 | return str0_node_ptr_->range_ref(); 11126 | } 11127 | 11128 | inline typename expression_node<T>::node_type type() const exprtk_override 11129 | { 11130 | return expression_node<T>::e_strass; 11131 | } 11132 | 11133 | inline bool valid() const exprtk_override 11134 | { 11135 | return initialised_ && binary_node<T>::valid(); 11136 | } 11137 | 11138 | private: 11139 | 11140 | bool initialised_; 11141 | str_base_ptr str0_base_ptr_; 11142 | str_base_ptr str1_base_ptr_; 11143 | strvar_node_ptr str0_node_ptr_; 11144 | range_ptr str1_range_ptr_; 11145 | }; 11146 | 11147 | template <typename T, typename AssignmentProcess = asn_assignment> 11148 | class assignment_string_range_node exprtk_final 11149 | : public binary_node <T> 11150 | , public string_base_node<T> 11151 | , public range_interface <T> 11152 | { 11153 | public: 11154 | 11155 | typedef typename range_interface<T>::range_t range_t; 11156 | typedef range_t* range_ptr; 11157 | typedef range_interface <T> irange_t; 11158 | typedef irange_t* irange_ptr; 11159 | typedef expression_node <T>* expression_ptr; 11160 | typedef stringvar_node <T>* strvar_node_ptr; 11161 | typedef string_range_node<T>* str_rng_node_ptr; 11162 | typedef string_base_node <T>* str_base_ptr; 11163 | 11164 | using binary_node<T>::branch; 11165 | 11166 | assignment_string_range_node(const operator_type& opr, 11167 | expression_ptr branch0, 11168 | expression_ptr branch1) 11169 | : binary_node<T>(opr, branch0, branch1) 11170 | , initialised_(false) 11171 | , str0_base_ptr_ (0) 11172 | , str1_base_ptr_ (0) 11173 | , str0_rng_node_ptr_(0) 11174 | , str0_range_ptr_ (0) 11175 | , str1_range_ptr_ (0) 11176 | { 11177 | if (is_string_range_node(branch(0))) 11178 | { 11179 | str0_rng_node_ptr_ = static_cast<str_rng_node_ptr>(branch(0)); 11180 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 11181 | irange_ptr range = dynamic_cast<irange_ptr>(branch(0)); 11182 | 11183 | if (0 == range) 11184 | return; 11185 | 11186 | str0_range_ptr_ = &(range->range_ref()); 11187 | } 11188 | 11189 | if (is_generally_string_node(branch(1))) 11190 | { 11191 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(branch(1)); 11192 | 11193 | if (0 == str1_base_ptr_) 11194 | return; 11195 | 11196 | irange_ptr range = dynamic_cast<irange_ptr>(branch(1)); 11197 | 11198 | if (0 == range) 11199 | return; 11200 | 11201 | str1_range_ptr_ = &(range->range_ref()); 11202 | } 11203 | 11204 | initialised_ = str0_base_ptr_ && 11205 | str1_base_ptr_ && 11206 | str0_rng_node_ptr_ && 11207 | str0_range_ptr_ && 11208 | str1_range_ptr_ ; 11209 | 11210 | assert(valid()); 11211 | } 11212 | 11213 | inline T value() const exprtk_override 11214 | { 11215 | branch(0)->value(); 11216 | branch(1)->value(); 11217 | 11218 | std::size_t s0_r0 = 0; 11219 | std::size_t s0_r1 = 0; 11220 | 11221 | std::size_t s1_r0 = 0; 11222 | std::size_t s1_r1 = 0; 11223 | 11224 | const range_t& range0 = (*str0_range_ptr_); 11225 | const range_t& range1 = (*str1_range_ptr_); 11226 | 11227 | if ( 11228 | range0(s0_r0, s0_r1, str0_base_ptr_->size()) && 11229 | range1(s1_r0, s1_r1, str1_base_ptr_->size()) 11230 | ) 11231 | { 11232 | const std::size_t size = std::min((s0_r1 - s0_r0), (s1_r1 - s1_r0)); 11233 | 11234 | std::copy( 11235 | str1_base_ptr_->base() + s1_r0, 11236 | str1_base_ptr_->base() + s1_r0 + size, 11237 | const_cast<char_ptr>(base() + s0_r0)); 11238 | } 11239 | 11240 | return std::numeric_limits<T>::quiet_NaN(); 11241 | } 11242 | 11243 | std::string str() const exprtk_override 11244 | { 11245 | return str0_base_ptr_->str(); 11246 | } 11247 | 11248 | char_cptr base() const exprtk_override 11249 | { 11250 | return str0_base_ptr_->base(); 11251 | } 11252 | 11253 | std::size_t size() const exprtk_override 11254 | { 11255 | return str0_base_ptr_->size(); 11256 | } 11257 | 11258 | range_t& range_ref() exprtk_override 11259 | { 11260 | return str0_rng_node_ptr_->range_ref(); 11261 | } 11262 | 11263 | const range_t& range_ref() const exprtk_override 11264 | { 11265 | return str0_rng_node_ptr_->range_ref(); 11266 | } 11267 | 11268 | inline typename expression_node<T>::node_type type() const exprtk_override 11269 | { 11270 | return expression_node<T>::e_strass; 11271 | } 11272 | 11273 | inline bool valid() const exprtk_override 11274 | { 11275 | return initialised_ && binary_node<T>::valid(); 11276 | } 11277 | 11278 | private: 11279 | 11280 | bool initialised_; 11281 | str_base_ptr str0_base_ptr_; 11282 | str_base_ptr str1_base_ptr_; 11283 | str_rng_node_ptr str0_rng_node_ptr_; 11284 | range_ptr str0_range_ptr_; 11285 | range_ptr str1_range_ptr_; 11286 | }; 11287 | 11288 | template <typename T> 11289 | class conditional_string_node exprtk_final 11290 | : public trinary_node <T> 11291 | , public string_base_node<T> 11292 | , public range_interface <T> 11293 | { 11294 | public: 11295 | 11296 | typedef typename range_interface<T>::range_t range_t; 11297 | typedef range_t* range_ptr; 11298 | typedef range_interface <T> irange_t; 11299 | typedef irange_t* irange_ptr; 11300 | typedef expression_node <T>* expression_ptr; 11301 | typedef string_base_node<T>* str_base_ptr; 11302 | 11303 | conditional_string_node(expression_ptr condition, 11304 | expression_ptr consequent, 11305 | expression_ptr alternative) 11306 | : trinary_node<T>(details::e_default, consequent, alternative, condition) 11307 | , initialised_(false) 11308 | , str0_base_ptr_ (0) 11309 | , str1_base_ptr_ (0) 11310 | , str0_range_ptr_(0) 11311 | , str1_range_ptr_(0) 11312 | , condition_ (condition ) 11313 | , consequent_ (consequent ) 11314 | , alternative_(alternative) 11315 | { 11316 | range_.n0_c = std::make_pair<bool,std::size_t>(true,0); 11317 | range_.n1_c = std::make_pair<bool,std::size_t>(true,0); 11318 | 11319 | range_.cache.first = range_.n0_c.second; 11320 | range_.cache.second = range_.n1_c.second; 11321 | 11322 | if (is_generally_string_node(trinary_node<T>::branch_[0].first)) 11323 | { 11324 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(trinary_node<T>::branch_[0].first); 11325 | 11326 | if (0 == str0_base_ptr_) 11327 | return; 11328 | 11329 | str0_range_ptr_ = dynamic_cast<irange_ptr>(trinary_node<T>::branch_[0].first); 11330 | 11331 | if (0 == str0_range_ptr_) 11332 | return; 11333 | } 11334 | 11335 | if (is_generally_string_node(trinary_node<T>::branch_[1].first)) 11336 | { 11337 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(trinary_node<T>::branch_[1].first); 11338 | 11339 | if (0 == str1_base_ptr_) 11340 | return; 11341 | 11342 | str1_range_ptr_ = dynamic_cast<irange_ptr>(trinary_node<T>::branch_[1].first); 11343 | 11344 | if (0 == str1_range_ptr_) 11345 | return; 11346 | } 11347 | 11348 | initialised_ = str0_base_ptr_ && 11349 | str1_base_ptr_ && 11350 | str0_range_ptr_ && 11351 | str1_range_ptr_ ; 11352 | 11353 | assert(valid()); 11354 | } 11355 | 11356 | inline T value() const exprtk_override 11357 | { 11358 | std::size_t r0 = 0; 11359 | std::size_t r1 = 0; 11360 | 11361 | if (is_true(condition_)) 11362 | { 11363 | consequent_->value(); 11364 | 11365 | const range_t& range = str0_range_ptr_->range_ref(); 11366 | 11367 | if (range(r0, r1, str0_base_ptr_->size())) 11368 | { 11369 | const std::size_t size = (r1 - r0); 11370 | 11371 | value_.assign(str0_base_ptr_->base() + r0, size); 11372 | 11373 | range_.n1_c.second = value_.size(); 11374 | range_.cache.second = range_.n1_c.second; 11375 | 11376 | return T(1); 11377 | } 11378 | } 11379 | else 11380 | { 11381 | alternative_->value(); 11382 | 11383 | const range_t& range = str1_range_ptr_->range_ref(); 11384 | 11385 | if (range(r0, r1, str1_base_ptr_->size())) 11386 | { 11387 | const std::size_t size = (r1 - r0); 11388 | 11389 | value_.assign(str1_base_ptr_->base() + r0, size); 11390 | 11391 | range_.n1_c.second = value_.size(); 11392 | range_.cache.second = range_.n1_c.second; 11393 | 11394 | return T(0); 11395 | } 11396 | } 11397 | 11398 | return std::numeric_limits<T>::quiet_NaN(); 11399 | } 11400 | 11401 | std::string str() const exprtk_override 11402 | { 11403 | return value_; 11404 | } 11405 | 11406 | char_cptr base() const exprtk_override 11407 | { 11408 | return &value_[0]; 11409 | } 11410 | 11411 | std::size_t size() const exprtk_override 11412 | { 11413 | return value_.size(); 11414 | } 11415 | 11416 | range_t& range_ref() exprtk_override 11417 | { 11418 | return range_; 11419 | } 11420 | 11421 | const range_t& range_ref() const exprtk_override 11422 | { 11423 | return range_; 11424 | } 11425 | 11426 | inline typename expression_node<T>::node_type type() const exprtk_override 11427 | { 11428 | return expression_node<T>::e_strcondition; 11429 | } 11430 | 11431 | inline bool valid() const exprtk_override 11432 | { 11433 | return 11434 | initialised_ && 11435 | condition_ && condition_ ->valid() && 11436 | consequent_ && consequent_ ->valid() && 11437 | alternative_&& alternative_->valid() ; 11438 | } 11439 | 11440 | private: 11441 | 11442 | bool initialised_; 11443 | str_base_ptr str0_base_ptr_; 11444 | str_base_ptr str1_base_ptr_; 11445 | irange_ptr str0_range_ptr_; 11446 | irange_ptr str1_range_ptr_; 11447 | mutable range_t range_; 11448 | mutable std::string value_; 11449 | 11450 | expression_ptr condition_; 11451 | expression_ptr consequent_; 11452 | expression_ptr alternative_; 11453 | }; 11454 | 11455 | template <typename T> 11456 | class cons_conditional_str_node exprtk_final 11457 | : public binary_node <T> 11458 | , public string_base_node<T> 11459 | , public range_interface <T> 11460 | { 11461 | public: 11462 | 11463 | typedef typename range_interface<T>::range_t range_t; 11464 | typedef range_t* range_ptr; 11465 | typedef range_interface <T> irange_t; 11466 | typedef irange_t* irange_ptr; 11467 | typedef expression_node <T>* expression_ptr; 11468 | typedef string_base_node<T>* str_base_ptr; 11469 | 11470 | using binary_node<T>::branch; 11471 | 11472 | cons_conditional_str_node(expression_ptr condition, 11473 | expression_ptr consequent) 11474 | : binary_node<T>(details::e_default, consequent, condition) 11475 | , initialised_(false) 11476 | , str0_base_ptr_ (0) 11477 | , str0_range_ptr_(0) 11478 | , condition_ (condition ) 11479 | , consequent_(consequent) 11480 | { 11481 | range_.n0_c = std::make_pair<bool,std::size_t>(true,0); 11482 | range_.n1_c = std::make_pair<bool,std::size_t>(true,0); 11483 | 11484 | range_.cache.first = range_.n0_c.second; 11485 | range_.cache.second = range_.n1_c.second; 11486 | 11487 | if (is_generally_string_node(branch(0))) 11488 | { 11489 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 11490 | 11491 | if (0 == str0_base_ptr_) 11492 | return; 11493 | 11494 | str0_range_ptr_ = dynamic_cast<irange_ptr>(branch(0)); 11495 | 11496 | if (0 == str0_range_ptr_) 11497 | return; 11498 | } 11499 | 11500 | initialised_ = str0_base_ptr_ && str0_range_ptr_ ; 11501 | assert(valid()); 11502 | } 11503 | 11504 | inline T value() const exprtk_override 11505 | { 11506 | if (is_true(condition_)) 11507 | { 11508 | consequent_->value(); 11509 | 11510 | const range_t& range = str0_range_ptr_->range_ref(); 11511 | 11512 | std::size_t r0 = 0; 11513 | std::size_t r1 = 0; 11514 | 11515 | if (range(r0, r1, str0_base_ptr_->size())) 11516 | { 11517 | const std::size_t size = (r1 - r0); 11518 | 11519 | value_.assign(str0_base_ptr_->base() + r0, size); 11520 | 11521 | range_.n1_c.second = value_.size(); 11522 | range_.cache.second = range_.n1_c.second; 11523 | 11524 | return T(1); 11525 | } 11526 | } 11527 | 11528 | return std::numeric_limits<T>::quiet_NaN(); 11529 | } 11530 | 11531 | std::string str() const 11532 | { 11533 | return value_; 11534 | } 11535 | 11536 | char_cptr base() const 11537 | { 11538 | return &value_[0]; 11539 | } 11540 | 11541 | std::size_t size() const 11542 | { 11543 | return value_.size(); 11544 | } 11545 | 11546 | range_t& range_ref() 11547 | { 11548 | return range_; 11549 | } 11550 | 11551 | const range_t& range_ref() const 11552 | { 11553 | return range_; 11554 | } 11555 | 11556 | inline typename expression_node<T>::node_type type() const exprtk_override 11557 | { 11558 | return expression_node<T>::e_strccondition; 11559 | } 11560 | 11561 | inline bool valid() const exprtk_override 11562 | { 11563 | return 11564 | initialised_ && 11565 | condition_ && condition_ ->valid() && 11566 | consequent_ && consequent_ ->valid() ; 11567 | } 11568 | 11569 | private: 11570 | 11571 | bool initialised_; 11572 | str_base_ptr str0_base_ptr_; 11573 | irange_ptr str0_range_ptr_; 11574 | mutable range_t range_; 11575 | mutable std::string value_; 11576 | 11577 | expression_ptr condition_; 11578 | expression_ptr consequent_; 11579 | }; 11580 | 11581 | template <typename T, typename VarArgFunction> 11582 | class str_vararg_node exprtk_final 11583 | : public expression_node <T> 11584 | , public string_base_node<T> 11585 | , public range_interface <T> 11586 | { 11587 | public: 11588 | 11589 | typedef typename range_interface<T>::range_t range_t; 11590 | typedef range_t* range_ptr; 11591 | typedef range_interface <T> irange_t; 11592 | typedef irange_t* irange_ptr; 11593 | typedef expression_node <T>* expression_ptr; 11594 | typedef string_base_node<T>* str_base_ptr; 11595 | typedef std::pair<expression_ptr,bool> branch_t; 11596 | 11597 | template <typename Allocator, 11598 | template <typename, typename> class Sequence> 11599 | explicit str_vararg_node(const Sequence<expression_ptr,Allocator>& arg_list) 11600 | : initialised_(false) 11601 | , str_base_ptr_ (0) 11602 | , str_range_ptr_(0) 11603 | { 11604 | construct_branch_pair(final_node_, const_cast<expression_ptr>(arg_list.back())); 11605 | 11606 | if (0 == final_node_.first) 11607 | return; 11608 | else if (!is_generally_string_node(final_node_.first)) 11609 | return; 11610 | 11611 | str_base_ptr_ = dynamic_cast<str_base_ptr>(final_node_.first); 11612 | 11613 | if (0 == str_base_ptr_) 11614 | return; 11615 | 11616 | str_range_ptr_ = dynamic_cast<irange_ptr>(final_node_.first); 11617 | 11618 | if (0 == str_range_ptr_) 11619 | return; 11620 | 11621 | if (arg_list.size() > 1) 11622 | { 11623 | const std::size_t arg_list_size = arg_list.size() - 1; 11624 | 11625 | arg_list_.resize(arg_list_size); 11626 | 11627 | for (std::size_t i = 0; i < arg_list_size; ++i) 11628 | { 11629 | if (arg_list[i] && arg_list[i]->valid()) 11630 | { 11631 | construct_branch_pair(arg_list_[i], arg_list[i]); 11632 | } 11633 | else 11634 | { 11635 | arg_list_.clear(); 11636 | return; 11637 | } 11638 | } 11639 | 11640 | initialised_ = true; 11641 | } 11642 | 11643 | initialised_ &= str_base_ptr_ && str_range_ptr_; 11644 | assert(valid()); 11645 | } 11646 | 11647 | inline T value() const exprtk_override 11648 | { 11649 | if (!arg_list_.empty()) 11650 | { 11651 | VarArgFunction::process(arg_list_); 11652 | } 11653 | 11654 | final_node_.first->value(); 11655 | 11656 | return std::numeric_limits<T>::quiet_NaN(); 11657 | } 11658 | 11659 | std::string str() const exprtk_override 11660 | { 11661 | return str_base_ptr_->str(); 11662 | } 11663 | 11664 | char_cptr base() const exprtk_override 11665 | { 11666 | return str_base_ptr_->base(); 11667 | } 11668 | 11669 | std::size_t size() const exprtk_override 11670 | { 11671 | return str_base_ptr_->size(); 11672 | } 11673 | 11674 | range_t& range_ref() exprtk_override 11675 | { 11676 | return str_range_ptr_->range_ref(); 11677 | } 11678 | 11679 | const range_t& range_ref() const exprtk_override 11680 | { 11681 | return str_range_ptr_->range_ref(); 11682 | } 11683 | 11684 | inline typename expression_node<T>::node_type type() const exprtk_override 11685 | { 11686 | return expression_node<T>::e_stringvararg; 11687 | } 11688 | 11689 | inline bool valid() const exprtk_override 11690 | { 11691 | return 11692 | initialised_ && 11693 | final_node_.first && final_node_.first->valid(); 11694 | } 11695 | 11696 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 11697 | { 11698 | expression_node<T>::ndb_t::collect(final_node_ , node_delete_list); 11699 | expression_node<T>::ndb_t::collect(arg_list_ , node_delete_list); 11700 | } 11701 | 11702 | std::size_t node_depth() const exprtk_override 11703 | { 11704 | return std::max( 11705 | expression_node<T>::ndb_t::compute_node_depth(final_node_), 11706 | expression_node<T>::ndb_t::compute_node_depth(arg_list_ )); 11707 | } 11708 | 11709 | private: 11710 | 11711 | bool initialised_; 11712 | branch_t final_node_; 11713 | str_base_ptr str_base_ptr_; 11714 | irange_ptr str_range_ptr_; 11715 | std::vector<branch_t> arg_list_; 11716 | }; 11717 | #endif 11718 | 11719 | template <typename T> 11720 | class assert_node exprtk_final : public expression_node<T> 11721 | { 11722 | public: 11723 | 11724 | typedef expression_node<T>* expression_ptr; 11725 | typedef std::pair<expression_ptr,bool> branch_t; 11726 | typedef string_base_node<T>* str_base_ptr; 11727 | typedef assert_check::assert_context assert_context_t; 11728 | 11729 | assert_node(expression_ptr assert_condition_node, 11730 | expression_ptr assert_message_node, 11731 | assert_check_ptr assert_check, 11732 | assert_context_t context) 11733 | : assert_message_str_base_(0) 11734 | , assert_check_(assert_check) 11735 | , context_(context) 11736 | { 11737 | construct_branch_pair(assert_condition_node_, assert_condition_node); 11738 | construct_branch_pair(assert_message_node_ , assert_message_node ); 11739 | 11740 | #ifndef exprtk_disable_string_capabilities 11741 | if ( 11742 | assert_message_node_.first && 11743 | details::is_generally_string_node(assert_message_node_.first) 11744 | ) 11745 | { 11746 | assert_message_str_base_ = dynamic_cast<str_base_ptr>(assert_message_node_.first); 11747 | } 11748 | #endif 11749 | 11750 | assert(valid()); 11751 | } 11752 | 11753 | inline T value() const exprtk_override 11754 | { 11755 | if (details::is_true(assert_condition_node_.first->value())) 11756 | { 11757 | return T(1); 11758 | } 11759 | 11760 | #ifndef exprtk_disable_string_capabilities 11761 | if (assert_message_node_.first) 11762 | { 11763 | assert_message_node_.first->value(); 11764 | assert(assert_message_str_base_); 11765 | context_.message = assert_message_str_base_->str(); 11766 | } 11767 | #endif 11768 | 11769 | assert_check_->handle_assert(context_); 11770 | return T(0); 11771 | } 11772 | 11773 | inline typename expression_node<T>::node_type type() const exprtk_override 11774 | { 11775 | return expression_node<T>::e_assert; 11776 | } 11777 | 11778 | inline bool valid() const exprtk_override 11779 | { 11780 | return ( 11781 | assert_check_ && 11782 | assert_condition_node_.first && 11783 | assert_condition_node_.first->valid() 11784 | ) && 11785 | ( 11786 | (0 == assert_message_node_.first) || 11787 | ( 11788 | assert_message_node_.first && 11789 | assert_message_str_base_ && 11790 | assert_message_node_.first->valid() && 11791 | details::is_generally_string_node(assert_message_node_.first) 11792 | ) 11793 | ); 11794 | } 11795 | 11796 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 11797 | { 11798 | expression_node<T>::ndb_t::collect(assert_condition_node_, node_delete_list); 11799 | expression_node<T>::ndb_t::collect(assert_message_node_ , node_delete_list); 11800 | } 11801 | 11802 | std::size_t node_depth() const exprtk_override 11803 | { 11804 | return expression_node<T>::ndb_t::compute_node_depth 11805 | (assert_condition_node_, assert_message_node_); 11806 | } 11807 | 11808 | private: 11809 | 11810 | branch_t assert_condition_node_; 11811 | branch_t assert_message_node_; 11812 | str_base_ptr assert_message_str_base_; 11813 | assert_check_ptr assert_check_; 11814 | mutable assert_context_t context_; 11815 | }; 11816 | 11817 | template <typename T, std::size_t N> 11818 | inline T axn(const T a, const T x) 11819 | { 11820 | // a*x^n 11821 | return a * exprtk::details::numeric::fast_exp<T,N>::result(x); 11822 | } 11823 | 11824 | template <typename T, std::size_t N> 11825 | inline T axnb(const T a, const T x, const T b) 11826 | { 11827 | // a*x^n+b 11828 | return a * exprtk::details::numeric::fast_exp<T,N>::result(x) + b; 11829 | } 11830 | 11831 | template <typename T> 11832 | struct sf_base 11833 | { 11834 | typedef typename details::functor_t<T>::Type Type; 11835 | typedef typename details::functor_t<T> functor_t; 11836 | typedef typename functor_t::qfunc_t quaternary_functor_t; 11837 | typedef typename functor_t::tfunc_t trinary_functor_t; 11838 | typedef typename functor_t::bfunc_t binary_functor_t; 11839 | typedef typename functor_t::ufunc_t unary_functor_t; 11840 | }; 11841 | 11842 | #define define_sfop3(NN, OP0, OP1) \ 11843 | template <typename T> \ 11844 | struct sf##NN##_op : public sf_base<T> \ 11845 | { \ 11846 | typedef typename sf_base<T>::Type const Type; \ 11847 | static inline T process(Type x, Type y, Type z) \ 11848 | { \ 11849 | return (OP0); \ 11850 | } \ 11851 | static inline std::string id() \ 11852 | { \ 11853 | return (OP1); \ 11854 | } \ 11855 | }; \ 11856 | 11857 | define_sfop3(00,(x + y) / z ,"(t+t)/t") 11858 | define_sfop3(01,(x + y) * z ,"(t+t)*t") 11859 | define_sfop3(02,(x + y) - z ,"(t+t)-t") 11860 | define_sfop3(03,(x + y) + z ,"(t+t)+t") 11861 | define_sfop3(04,(x - y) + z ,"(t-t)+t") 11862 | define_sfop3(05,(x - y) / z ,"(t-t)/t") 11863 | define_sfop3(06,(x - y) * z ,"(t-t)*t") 11864 | define_sfop3(07,(x * y) + z ,"(t*t)+t") 11865 | define_sfop3(08,(x * y) - z ,"(t*t)-t") 11866 | define_sfop3(09,(x * y) / z ,"(t*t)/t") 11867 | define_sfop3(10,(x * y) * z ,"(t*t)*t") 11868 | define_sfop3(11,(x / y) + z ,"(t/t)+t") 11869 | define_sfop3(12,(x / y) - z ,"(t/t)-t") 11870 | define_sfop3(13,(x / y) / z ,"(t/t)/t") 11871 | define_sfop3(14,(x / y) * z ,"(t/t)*t") 11872 | define_sfop3(15,x / (y + z) ,"t/(t+t)") 11873 | define_sfop3(16,x / (y - z) ,"t/(t-t)") 11874 | define_sfop3(17,x / (y * z) ,"t/(t*t)") 11875 | define_sfop3(18,x / (y / z) ,"t/(t/t)") 11876 | define_sfop3(19,x * (y + z) ,"t*(t+t)") 11877 | define_sfop3(20,x * (y - z) ,"t*(t-t)") 11878 | define_sfop3(21,x * (y * z) ,"t*(t*t)") 11879 | define_sfop3(22,x * (y / z) ,"t*(t/t)") 11880 | define_sfop3(23,x - (y + z) ,"t-(t+t)") 11881 | define_sfop3(24,x - (y - z) ,"t-(t-t)") 11882 | define_sfop3(25,x - (y / z) ,"t-(t/t)") 11883 | define_sfop3(26,x - (y * z) ,"t-(t*t)") 11884 | define_sfop3(27,x + (y * z) ,"t+(t*t)") 11885 | define_sfop3(28,x + (y / z) ,"t+(t/t)") 11886 | define_sfop3(29,x + (y + z) ,"t+(t+t)") 11887 | define_sfop3(30,x + (y - z) ,"t+(t-t)") 11888 | define_sfop3(31,(axnb<T,2>(x,y,z))," ") 11889 | define_sfop3(32,(axnb<T,3>(x,y,z))," ") 11890 | define_sfop3(33,(axnb<T,4>(x,y,z))," ") 11891 | define_sfop3(34,(axnb<T,5>(x,y,z))," ") 11892 | define_sfop3(35,(axnb<T,6>(x,y,z))," ") 11893 | define_sfop3(36,(axnb<T,7>(x,y,z))," ") 11894 | define_sfop3(37,(axnb<T,8>(x,y,z))," ") 11895 | define_sfop3(38,(axnb<T,9>(x,y,z))," ") 11896 | define_sfop3(39,x * numeric::log(y) + z,"") 11897 | define_sfop3(40,x * numeric::log(y) - z,"") 11898 | define_sfop3(41,x * numeric::log10(y) + z,"") 11899 | define_sfop3(42,x * numeric::log10(y) - z,"") 11900 | define_sfop3(43,x * numeric::sin(y) + z ,"") 11901 | define_sfop3(44,x * numeric::sin(y) - z ,"") 11902 | define_sfop3(45,x * numeric::cos(y) + z ,"") 11903 | define_sfop3(46,x * numeric::cos(y) - z ,"") 11904 | define_sfop3(47,details::is_true(x) ? y : z,"") 11905 | 11906 | #define define_sfop4(NN, OP0, OP1) \ 11907 | template <typename T> \ 11908 | struct sf##NN##_op : public sf_base<T> \ 11909 | { \ 11910 | typedef typename sf_base<T>::Type const Type; \ 11911 | static inline T process(Type x, Type y, Type z, Type w) \ 11912 | { \ 11913 | return (OP0); \ 11914 | } \ 11915 | static inline std::string id() \ 11916 | { \ 11917 | return (OP1); \ 11918 | } \ 11919 | }; \ 11920 | 11921 | define_sfop4(48,(x + ((y + z) / w)),"t+((t+t)/t)") 11922 | define_sfop4(49,(x + ((y + z) * w)),"t+((t+t)*t)") 11923 | define_sfop4(50,(x + ((y - z) / w)),"t+((t-t)/t)") 11924 | define_sfop4(51,(x + ((y - z) * w)),"t+((t-t)*t)") 11925 | define_sfop4(52,(x + ((y * z) / w)),"t+((t*t)/t)") 11926 | define_sfop4(53,(x + ((y * z) * w)),"t+((t*t)*t)") 11927 | define_sfop4(54,(x + ((y / z) + w)),"t+((t/t)+t)") 11928 | define_sfop4(55,(x + ((y / z) / w)),"t+((t/t)/t)") 11929 | define_sfop4(56,(x + ((y / z) * w)),"t+((t/t)*t)") 11930 | define_sfop4(57,(x - ((y + z) / w)),"t-((t+t)/t)") 11931 | define_sfop4(58,(x - ((y + z) * w)),"t-((t+t)*t)") 11932 | define_sfop4(59,(x - ((y - z) / w)),"t-((t-t)/t)") 11933 | define_sfop4(60,(x - ((y - z) * w)),"t-((t-t)*t)") 11934 | define_sfop4(61,(x - ((y * z) / w)),"t-((t*t)/t)") 11935 | define_sfop4(62,(x - ((y * z) * w)),"t-((t*t)*t)") 11936 | define_sfop4(63,(x - ((y / z) / w)),"t-((t/t)/t)") 11937 | define_sfop4(64,(x - ((y / z) * w)),"t-((t/t)*t)") 11938 | define_sfop4(65,(((x + y) * z) - w),"((t+t)*t)-t") 11939 | define_sfop4(66,(((x - y) * z) - w),"((t-t)*t)-t") 11940 | define_sfop4(67,(((x * y) * z) - w),"((t*t)*t)-t") 11941 | define_sfop4(68,(((x / y) * z) - w),"((t/t)*t)-t") 11942 | define_sfop4(69,(((x + y) / z) - w),"((t+t)/t)-t") 11943 | define_sfop4(70,(((x - y) / z) - w),"((t-t)/t)-t") 11944 | define_sfop4(71,(((x * y) / z) - w),"((t*t)/t)-t") 11945 | define_sfop4(72,(((x / y) / z) - w),"((t/t)/t)-t") 11946 | define_sfop4(73,((x * y) + (z * w)),"(t*t)+(t*t)") 11947 | define_sfop4(74,((x * y) - (z * w)),"(t*t)-(t*t)") 11948 | define_sfop4(75,((x * y) + (z / w)),"(t*t)+(t/t)") 11949 | define_sfop4(76,((x * y) - (z / w)),"(t*t)-(t/t)") 11950 | define_sfop4(77,((x / y) + (z / w)),"(t/t)+(t/t)") 11951 | define_sfop4(78,((x / y) - (z / w)),"(t/t)-(t/t)") 11952 | define_sfop4(79,((x / y) - (z * w)),"(t/t)-(t*t)") 11953 | define_sfop4(80,(x / (y + (z * w))),"t/(t+(t*t))") 11954 | define_sfop4(81,(x / (y - (z * w))),"t/(t-(t*t))") 11955 | define_sfop4(82,(x * (y + (z * w))),"t*(t+(t*t))") 11956 | define_sfop4(83,(x * (y - (z * w))),"t*(t-(t*t))") 11957 | 11958 | define_sfop4(84,(axn<T,2>(x,y) + axn<T,2>(z,w)),"") 11959 | define_sfop4(85,(axn<T,3>(x,y) + axn<T,3>(z,w)),"") 11960 | define_sfop4(86,(axn<T,4>(x,y) + axn<T,4>(z,w)),"") 11961 | define_sfop4(87,(axn<T,5>(x,y) + axn<T,5>(z,w)),"") 11962 | define_sfop4(88,(axn<T,6>(x,y) + axn<T,6>(z,w)),"") 11963 | define_sfop4(89,(axn<T,7>(x,y) + axn<T,7>(z,w)),"") 11964 | define_sfop4(90,(axn<T,8>(x,y) + axn<T,8>(z,w)),"") 11965 | define_sfop4(91,(axn<T,9>(x,y) + axn<T,9>(z,w)),"") 11966 | define_sfop4(92,((details::is_true(x) && details::is_true(y)) ? z : w),"") 11967 | define_sfop4(93,((details::is_true(x) || details::is_true(y)) ? z : w),"") 11968 | define_sfop4(94,((x < y) ? z : w),"") 11969 | define_sfop4(95,((x <= y) ? z : w),"") 11970 | define_sfop4(96,((x > y) ? z : w),"") 11971 | define_sfop4(97,((x >= y) ? z : w),"") 11972 | define_sfop4(98,(details::is_true(numeric::equal(x,y)) ? z : w),"") 11973 | define_sfop4(99,(x * numeric::sin(y) + z * numeric::cos(w)),"") 11974 | 11975 | define_sfop4(ext00,((x + y) - (z * w)),"(t+t)-(t*t)") 11976 | define_sfop4(ext01,((x + y) - (z / w)),"(t+t)-(t/t)") 11977 | define_sfop4(ext02,((x + y) + (z * w)),"(t+t)+(t*t)") 11978 | define_sfop4(ext03,((x + y) + (z / w)),"(t+t)+(t/t)") 11979 | define_sfop4(ext04,((x - y) + (z * w)),"(t-t)+(t*t)") 11980 | define_sfop4(ext05,((x - y) + (z / w)),"(t-t)+(t/t)") 11981 | define_sfop4(ext06,((x - y) - (z * w)),"(t-t)-(t*t)") 11982 | define_sfop4(ext07,((x - y) - (z / w)),"(t-t)-(t/t)") 11983 | define_sfop4(ext08,((x + y) - (z - w)),"(t+t)-(t-t)") 11984 | define_sfop4(ext09,((x + y) + (z - w)),"(t+t)+(t-t)") 11985 | define_sfop4(ext10,((x + y) + (z + w)),"(t+t)+(t+t)") 11986 | define_sfop4(ext11,((x + y) * (z - w)),"(t+t)*(t-t)") 11987 | define_sfop4(ext12,((x + y) / (z - w)),"(t+t)/(t-t)") 11988 | define_sfop4(ext13,((x - y) - (z + w)),"(t-t)-(t+t)") 11989 | define_sfop4(ext14,((x - y) + (z + w)),"(t-t)+(t+t)") 11990 | define_sfop4(ext15,((x - y) * (z + w)),"(t-t)*(t+t)") 11991 | define_sfop4(ext16,((x - y) / (z + w)),"(t-t)/(t+t)") 11992 | define_sfop4(ext17,((x * y) - (z + w)),"(t*t)-(t+t)") 11993 | define_sfop4(ext18,((x / y) - (z + w)),"(t/t)-(t+t)") 11994 | define_sfop4(ext19,((x * y) + (z + w)),"(t*t)+(t+t)") 11995 | define_sfop4(ext20,((x / y) + (z + w)),"(t/t)+(t+t)") 11996 | define_sfop4(ext21,((x * y) + (z - w)),"(t*t)+(t-t)") 11997 | define_sfop4(ext22,((x / y) + (z - w)),"(t/t)+(t-t)") 11998 | define_sfop4(ext23,((x * y) - (z - w)),"(t*t)-(t-t)") 11999 | define_sfop4(ext24,((x / y) - (z - w)),"(t/t)-(t-t)") 12000 | define_sfop4(ext25,((x + y) * (z * w)),"(t+t)*(t*t)") 12001 | define_sfop4(ext26,((x + y) * (z / w)),"(t+t)*(t/t)") 12002 | define_sfop4(ext27,((x + y) / (z * w)),"(t+t)/(t*t)") 12003 | define_sfop4(ext28,((x + y) / (z / w)),"(t+t)/(t/t)") 12004 | define_sfop4(ext29,((x - y) / (z * w)),"(t-t)/(t*t)") 12005 | define_sfop4(ext30,((x - y) / (z / w)),"(t-t)/(t/t)") 12006 | define_sfop4(ext31,((x - y) * (z * w)),"(t-t)*(t*t)") 12007 | define_sfop4(ext32,((x - y) * (z / w)),"(t-t)*(t/t)") 12008 | define_sfop4(ext33,((x * y) * (z + w)),"(t*t)*(t+t)") 12009 | define_sfop4(ext34,((x / y) * (z + w)),"(t/t)*(t+t)") 12010 | define_sfop4(ext35,((x * y) / (z + w)),"(t*t)/(t+t)") 12011 | define_sfop4(ext36,((x / y) / (z + w)),"(t/t)/(t+t)") 12012 | define_sfop4(ext37,((x * y) / (z - w)),"(t*t)/(t-t)") 12013 | define_sfop4(ext38,((x / y) / (z - w)),"(t/t)/(t-t)") 12014 | define_sfop4(ext39,((x * y) * (z - w)),"(t*t)*(t-t)") 12015 | define_sfop4(ext40,((x * y) / (z * w)),"(t*t)/(t*t)") 12016 | define_sfop4(ext41,((x / y) * (z / w)),"(t/t)*(t/t)") 12017 | define_sfop4(ext42,((x / y) * (z - w)),"(t/t)*(t-t)") 12018 | define_sfop4(ext43,((x * y) * (z * w)),"(t*t)*(t*t)") 12019 | define_sfop4(ext44,(x + (y * (z / w))),"t+(t*(t/t))") 12020 | define_sfop4(ext45,(x - (y * (z / w))),"t-(t*(t/t))") 12021 | define_sfop4(ext46,(x + (y / (z * w))),"t+(t/(t*t))") 12022 | define_sfop4(ext47,(x - (y / (z * w))),"t-(t/(t*t))") 12023 | define_sfop4(ext48,(((x - y) - z) * w),"((t-t)-t)*t") 12024 | define_sfop4(ext49,(((x - y) - z) / w),"((t-t)-t)/t") 12025 | define_sfop4(ext50,(((x - y) + z) * w),"((t-t)+t)*t") 12026 | define_sfop4(ext51,(((x - y) + z) / w),"((t-t)+t)/t") 12027 | define_sfop4(ext52,((x + (y - z)) * w),"(t+(t-t))*t") 12028 | define_sfop4(ext53,((x + (y - z)) / w),"(t+(t-t))/t") 12029 | define_sfop4(ext54,((x + y) / (z + w)),"(t+t)/(t+t)") 12030 | define_sfop4(ext55,((x - y) / (z - w)),"(t-t)/(t-t)") 12031 | define_sfop4(ext56,((x + y) * (z + w)),"(t+t)*(t+t)") 12032 | define_sfop4(ext57,((x - y) * (z - w)),"(t-t)*(t-t)") 12033 | define_sfop4(ext58,((x - y) + (z - w)),"(t-t)+(t-t)") 12034 | define_sfop4(ext59,((x - y) - (z - w)),"(t-t)-(t-t)") 12035 | define_sfop4(ext60,((x / y) + (z * w)),"(t/t)+(t*t)") 12036 | define_sfop4(ext61,(((x * y) * z) / w),"((t*t)*t)/t") 12037 | 12038 | #undef define_sfop3 12039 | #undef define_sfop4 12040 | 12041 | template <typename T, typename SpecialFunction> 12042 | class sf3_node exprtk_final : public trinary_node<T> 12043 | { 12044 | public: 12045 | 12046 | typedef expression_node<T>* expression_ptr; 12047 | 12048 | sf3_node(const operator_type& opr, 12049 | expression_ptr branch0, 12050 | expression_ptr branch1, 12051 | expression_ptr branch2) 12052 | : trinary_node<T>(opr, branch0, branch1, branch2) 12053 | {} 12054 | 12055 | inline T value() const exprtk_override 12056 | { 12057 | const T x = trinary_node<T>::branch_[0].first->value(); 12058 | const T y = trinary_node<T>::branch_[1].first->value(); 12059 | const T z = trinary_node<T>::branch_[2].first->value(); 12060 | 12061 | return SpecialFunction::process(x, y, z); 12062 | } 12063 | }; 12064 | 12065 | template <typename T, typename SpecialFunction> 12066 | class sf4_node exprtk_final : public quaternary_node<T> 12067 | { 12068 | public: 12069 | 12070 | typedef expression_node<T>* expression_ptr; 12071 | 12072 | sf4_node(const operator_type& opr, 12073 | expression_ptr branch0, 12074 | expression_ptr branch1, 12075 | expression_ptr branch2, 12076 | expression_ptr branch3) 12077 | : quaternary_node<T>(opr, branch0, branch1, branch2, branch3) 12078 | {} 12079 | 12080 | inline T value() const exprtk_override 12081 | { 12082 | const T x = quaternary_node<T>::branch_[0].first->value(); 12083 | const T y = quaternary_node<T>::branch_[1].first->value(); 12084 | const T z = quaternary_node<T>::branch_[2].first->value(); 12085 | const T w = quaternary_node<T>::branch_[3].first->value(); 12086 | 12087 | return SpecialFunction::process(x, y, z, w); 12088 | } 12089 | }; 12090 | 12091 | template <typename T, typename SpecialFunction> 12092 | class sf3_var_node exprtk_final : public expression_node<T> 12093 | { 12094 | public: 12095 | 12096 | typedef expression_node<T>* expression_ptr; 12097 | 12098 | sf3_var_node(const T& v0, const T& v1, const T& v2) 12099 | : v0_(v0) 12100 | , v1_(v1) 12101 | , v2_(v2) 12102 | {} 12103 | 12104 | inline T value() const exprtk_override 12105 | { 12106 | return SpecialFunction::process(v0_, v1_, v2_); 12107 | } 12108 | 12109 | inline typename expression_node<T>::node_type type() const exprtk_override 12110 | { 12111 | return expression_node<T>::e_trinary; 12112 | } 12113 | 12114 | private: 12115 | 12116 | sf3_var_node(const sf3_var_node<T,SpecialFunction>&) exprtk_delete; 12117 | sf3_var_node<T,SpecialFunction>& operator=(const sf3_var_node<T,SpecialFunction>&) exprtk_delete; 12118 | 12119 | const T& v0_; 12120 | const T& v1_; 12121 | const T& v2_; 12122 | }; 12123 | 12124 | template <typename T, typename SpecialFunction> 12125 | class sf4_var_node exprtk_final : public expression_node<T> 12126 | { 12127 | public: 12128 | 12129 | typedef expression_node<T>* expression_ptr; 12130 | 12131 | sf4_var_node(const T& v0, const T& v1, const T& v2, const T& v3) 12132 | : v0_(v0) 12133 | , v1_(v1) 12134 | , v2_(v2) 12135 | , v3_(v3) 12136 | {} 12137 | 12138 | inline T value() const exprtk_override 12139 | { 12140 | return SpecialFunction::process(v0_, v1_, v2_, v3_); 12141 | } 12142 | 12143 | inline typename expression_node<T>::node_type type() const exprtk_override 12144 | { 12145 | return expression_node<T>::e_trinary; 12146 | } 12147 | 12148 | private: 12149 | 12150 | sf4_var_node(const sf4_var_node<T,SpecialFunction>&) exprtk_delete; 12151 | sf4_var_node<T,SpecialFunction>& operator=(const sf4_var_node<T,SpecialFunction>&) exprtk_delete; 12152 | 12153 | const T& v0_; 12154 | const T& v1_; 12155 | const T& v2_; 12156 | const T& v3_; 12157 | }; 12158 | 12159 | template <typename T, typename VarArgFunction> 12160 | class vararg_node exprtk_final : public expression_node<T> 12161 | { 12162 | public: 12163 | 12164 | typedef expression_node<T>* expression_ptr; 12165 | typedef std::pair<expression_ptr,bool> branch_t; 12166 | 12167 | template <typename Allocator, 12168 | template <typename, typename> class Sequence> 12169 | explicit vararg_node(const Sequence<expression_ptr,Allocator>& arg_list) 12170 | : initialised_(false) 12171 | { 12172 | arg_list_.resize(arg_list.size()); 12173 | 12174 | for (std::size_t i = 0; i < arg_list.size(); ++i) 12175 | { 12176 | if (arg_list[i] && arg_list[i]->valid()) 12177 | { 12178 | construct_branch_pair(arg_list_[i],arg_list[i]); 12179 | } 12180 | else 12181 | { 12182 | arg_list_.clear(); 12183 | return; 12184 | } 12185 | } 12186 | 12187 | initialised_ = (arg_list_.size() == arg_list.size()); 12188 | assert(valid()); 12189 | } 12190 | 12191 | inline T value() const exprtk_override 12192 | { 12193 | return VarArgFunction::process(arg_list_); 12194 | } 12195 | 12196 | inline typename expression_node<T>::node_type type() const exprtk_override 12197 | { 12198 | return expression_node<T>::e_vararg; 12199 | } 12200 | 12201 | inline bool valid() const exprtk_override 12202 | { 12203 | return initialised_; 12204 | } 12205 | 12206 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 12207 | { 12208 | expression_node<T>::ndb_t::collect(arg_list_, node_delete_list); 12209 | } 12210 | 12211 | std::size_t node_depth() const exprtk_override 12212 | { 12213 | return expression_node<T>::ndb_t::compute_node_depth(arg_list_); 12214 | } 12215 | 12216 | std::size_t size() const 12217 | { 12218 | return arg_list_.size(); 12219 | } 12220 | 12221 | expression_ptr operator[](const std::size_t& index) const 12222 | { 12223 | return arg_list_[index].first; 12224 | } 12225 | 12226 | private: 12227 | 12228 | std::vector<branch_t> arg_list_; 12229 | bool initialised_; 12230 | }; 12231 | 12232 | template <typename T, typename VarArgFunction> 12233 | class vararg_varnode exprtk_final : public expression_node<T> 12234 | { 12235 | public: 12236 | 12237 | typedef expression_node<T>* expression_ptr; 12238 | 12239 | template <typename Allocator, 12240 | template <typename, typename> class Sequence> 12241 | explicit vararg_varnode(const Sequence<expression_ptr,Allocator>& arg_list) 12242 | : initialised_(false) 12243 | { 12244 | arg_list_.resize(arg_list.size()); 12245 | 12246 | for (std::size_t i = 0; i < arg_list.size(); ++i) 12247 | { 12248 | if (arg_list[i] && arg_list[i]->valid() && is_variable_node(arg_list[i])) 12249 | { 12250 | variable_node<T>* var_node_ptr = static_cast<variable_node<T>*>(arg_list[i]); 12251 | arg_list_[i] = (&var_node_ptr->ref()); 12252 | } 12253 | else 12254 | { 12255 | arg_list_.clear(); 12256 | return; 12257 | } 12258 | } 12259 | 12260 | initialised_ = (arg_list.size() == arg_list_.size()); 12261 | assert(valid()); 12262 | } 12263 | 12264 | inline T value() const exprtk_override 12265 | { 12266 | return VarArgFunction::process(arg_list_); 12267 | } 12268 | 12269 | inline typename expression_node<T>::node_type type() const exprtk_override 12270 | { 12271 | return expression_node<T>::e_vararg; 12272 | } 12273 | 12274 | inline bool valid() const exprtk_override 12275 | { 12276 | return initialised_; 12277 | } 12278 | 12279 | private: 12280 | 12281 | std::vector<const T*> arg_list_; 12282 | bool initialised_; 12283 | }; 12284 | 12285 | template <typename T, typename VecFunction> 12286 | class vectorize_node exprtk_final : public expression_node<T> 12287 | { 12288 | public: 12289 | 12290 | typedef expression_node<T>* expression_ptr; 12291 | typedef std::pair<expression_ptr,bool> branch_t; 12292 | 12293 | explicit vectorize_node(const expression_ptr v) 12294 | : ivec_ptr_(0) 12295 | { 12296 | construct_branch_pair(v_, v); 12297 | 12298 | if (is_ivector_node(v_.first)) 12299 | { 12300 | ivec_ptr_ = dynamic_cast<vector_interface<T>*>(v_.first); 12301 | } 12302 | } 12303 | 12304 | inline T value() const exprtk_override 12305 | { 12306 | v_.first->value(); 12307 | return VecFunction::process(ivec_ptr_); 12308 | } 12309 | 12310 | inline typename expression_node<T>::node_type type() const exprtk_override 12311 | { 12312 | return expression_node<T>::e_vecfunc; 12313 | } 12314 | 12315 | inline bool valid() const exprtk_override 12316 | { 12317 | return ivec_ptr_ && v_.first && v_.first->valid(); 12318 | } 12319 | 12320 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 12321 | { 12322 | expression_node<T>::ndb_t::collect(v_, node_delete_list); 12323 | } 12324 | 12325 | std::size_t node_depth() const exprtk_override 12326 | { 12327 | return expression_node<T>::ndb_t::compute_node_depth(v_); 12328 | } 12329 | 12330 | private: 12331 | 12332 | vector_interface<T>* ivec_ptr_; 12333 | branch_t v_; 12334 | }; 12335 | 12336 | template <typename T> 12337 | class assignment_node exprtk_final : public binary_node<T> 12338 | { 12339 | public: 12340 | 12341 | typedef expression_node<T>* expression_ptr; 12342 | using binary_node<T>::branch; 12343 | 12344 | assignment_node(const operator_type& opr, 12345 | expression_ptr branch0, 12346 | expression_ptr branch1) 12347 | : binary_node<T>(opr, branch0, branch1) 12348 | , var_node_ptr_(0) 12349 | { 12350 | if (is_variable_node(branch(0))) 12351 | { 12352 | var_node_ptr_ = static_cast<variable_node<T>*>(branch(0)); 12353 | } 12354 | } 12355 | 12356 | inline T value() const exprtk_override 12357 | { 12358 | T& result = var_node_ptr_->ref(); 12359 | result = branch(1)->value(); 12360 | 12361 | return result; 12362 | } 12363 | 12364 | inline bool valid() const exprtk_override 12365 | { 12366 | return var_node_ptr_ && binary_node<T>::valid(); 12367 | } 12368 | 12369 | private: 12370 | 12371 | variable_node<T>* var_node_ptr_; 12372 | }; 12373 | 12374 | template <typename T> 12375 | class assignment_vec_elem_node exprtk_final : public binary_node<T> 12376 | { 12377 | public: 12378 | 12379 | typedef expression_node<T>* expression_ptr; 12380 | using binary_node<T>::branch; 12381 | 12382 | assignment_vec_elem_node(const operator_type& opr, 12383 | expression_ptr branch0, 12384 | expression_ptr branch1) 12385 | : binary_node<T>(opr, branch0, branch1) 12386 | , vec_node_ptr_(0) 12387 | { 12388 | if (is_vector_elem_node(branch(0))) 12389 | { 12390 | vec_node_ptr_ = static_cast<vector_elem_node<T>*>(branch(0)); 12391 | } 12392 | 12393 | assert(valid()); 12394 | } 12395 | 12396 | inline T value() const exprtk_override 12397 | { 12398 | T& result = vec_node_ptr_->ref(); 12399 | result = branch(1)->value(); 12400 | 12401 | return result; 12402 | } 12403 | 12404 | inline bool valid() const exprtk_override 12405 | { 12406 | return vec_node_ptr_ && binary_node<T>::valid(); 12407 | } 12408 | 12409 | private: 12410 | 12411 | vector_elem_node<T>* vec_node_ptr_; 12412 | }; 12413 | 12414 | template <typename T> 12415 | class assignment_vec_elem_rtc_node exprtk_final : public binary_node<T> 12416 | { 12417 | public: 12418 | 12419 | typedef expression_node<T>* expression_ptr; 12420 | using binary_node<T>::branch; 12421 | 12422 | assignment_vec_elem_rtc_node(const operator_type& opr, 12423 | expression_ptr branch0, 12424 | expression_ptr branch1) 12425 | : binary_node<T>(opr, branch0, branch1) 12426 | , vec_node_ptr_(0) 12427 | { 12428 | if (is_vector_elem_rtc_node(branch(0))) 12429 | { 12430 | vec_node_ptr_ = static_cast<vector_elem_rtc_node<T>*>(branch(0)); 12431 | } 12432 | 12433 | assert(valid()); 12434 | } 12435 | 12436 | inline T value() const exprtk_override 12437 | { 12438 | T& result = vec_node_ptr_->ref(); 12439 | result = branch(1)->value(); 12440 | 12441 | return result; 12442 | } 12443 | 12444 | inline bool valid() const exprtk_override 12445 | { 12446 | return vec_node_ptr_ && binary_node<T>::valid(); 12447 | } 12448 | 12449 | private: 12450 | 12451 | vector_elem_rtc_node<T>* vec_node_ptr_; 12452 | }; 12453 | 12454 | template <typename T> 12455 | class assignment_rebasevec_elem_node exprtk_final : public binary_node<T> 12456 | { 12457 | public: 12458 | 12459 | typedef expression_node<T>* expression_ptr; 12460 | using expression_node<T>::branch; 12461 | 12462 | assignment_rebasevec_elem_node(const operator_type& opr, 12463 | expression_ptr branch0, 12464 | expression_ptr branch1) 12465 | : binary_node<T>(opr, branch0, branch1) 12466 | , rbvec_node_ptr_(0) 12467 | { 12468 | if (is_rebasevector_elem_node(branch(0))) 12469 | { 12470 | rbvec_node_ptr_ = static_cast<rebasevector_elem_node<T>*>(branch(0)); 12471 | } 12472 | 12473 | assert(valid()); 12474 | } 12475 | 12476 | inline T value() const exprtk_override 12477 | { 12478 | T& result = rbvec_node_ptr_->ref(); 12479 | result = branch(1)->value(); 12480 | 12481 | return result; 12482 | } 12483 | 12484 | inline bool valid() const exprtk_override 12485 | { 12486 | return rbvec_node_ptr_ && binary_node<T>::valid(); 12487 | } 12488 | 12489 | private: 12490 | 12491 | rebasevector_elem_node<T>* rbvec_node_ptr_; 12492 | }; 12493 | 12494 | template <typename T> 12495 | class assignment_rebasevec_elem_rtc_node exprtk_final : public binary_node<T> 12496 | { 12497 | public: 12498 | 12499 | typedef expression_node<T>* expression_ptr; 12500 | using expression_node<T>::branch; 12501 | 12502 | assignment_rebasevec_elem_rtc_node(const operator_type& opr, 12503 | expression_ptr branch0, 12504 | expression_ptr branch1) 12505 | : binary_node<T>(opr, branch0, branch1) 12506 | , rbvec_node_ptr_(0) 12507 | { 12508 | if (is_rebasevector_elem_rtc_node(branch(0))) 12509 | { 12510 | rbvec_node_ptr_ = static_cast<rebasevector_elem_rtc_node<T>*>(branch(0)); 12511 | } 12512 | 12513 | assert(valid()); 12514 | } 12515 | 12516 | inline T value() const exprtk_override 12517 | { 12518 | T& result = rbvec_node_ptr_->ref(); 12519 | result = branch(1)->value(); 12520 | 12521 | return result; 12522 | } 12523 | 12524 | inline bool valid() const exprtk_override 12525 | { 12526 | return rbvec_node_ptr_ && binary_node<T>::valid(); 12527 | } 12528 | 12529 | private: 12530 | 12531 | rebasevector_elem_rtc_node<T>* rbvec_node_ptr_; 12532 | }; 12533 | 12534 | template <typename T> 12535 | class assignment_rebasevec_celem_node exprtk_final : public binary_node<T> 12536 | { 12537 | public: 12538 | 12539 | typedef expression_node<T>* expression_ptr; 12540 | using binary_node<T>::branch; 12541 | 12542 | assignment_rebasevec_celem_node(const operator_type& opr, 12543 | expression_ptr branch0, 12544 | expression_ptr branch1) 12545 | : binary_node<T>(opr, branch0, branch1) 12546 | , rbvec_node_ptr_(0) 12547 | { 12548 | if (is_rebasevector_celem_node(branch(0))) 12549 | { 12550 | rbvec_node_ptr_ = static_cast<rebasevector_celem_node<T>*>(branch(0)); 12551 | } 12552 | 12553 | assert(valid()); 12554 | } 12555 | 12556 | inline T value() const exprtk_override 12557 | { 12558 | T& result = rbvec_node_ptr_->ref(); 12559 | result = branch(1)->value(); 12560 | 12561 | return result; 12562 | } 12563 | 12564 | inline bool valid() const exprtk_override 12565 | { 12566 | return rbvec_node_ptr_ && binary_node<T>::valid(); 12567 | } 12568 | 12569 | private: 12570 | 12571 | rebasevector_celem_node<T>* rbvec_node_ptr_; 12572 | }; 12573 | 12574 | template <typename T> 12575 | class assignment_vec_node exprtk_final 12576 | : public binary_node <T> 12577 | , public vector_interface<T> 12578 | { 12579 | public: 12580 | 12581 | typedef expression_node<T>* expression_ptr; 12582 | typedef vector_node<T>* vector_node_ptr; 12583 | typedef vec_data_store<T> vds_t; 12584 | 12585 | using binary_node<T>::branch; 12586 | 12587 | assignment_vec_node(const operator_type& opr, 12588 | expression_ptr branch0, 12589 | expression_ptr branch1) 12590 | : binary_node<T>(opr, branch0, branch1) 12591 | , vec_node_ptr_(0) 12592 | { 12593 | if (is_vector_node(branch(0))) 12594 | { 12595 | vec_node_ptr_ = static_cast<vector_node<T>*>(branch(0)); 12596 | vds() = vec_node_ptr_->vds(); 12597 | } 12598 | 12599 | assert(valid()); 12600 | } 12601 | 12602 | inline T value() const exprtk_override 12603 | { 12604 | const T v = branch(1)->value(); 12605 | 12606 | T* vec = vds().data(); 12607 | 12608 | loop_unroll::details lud(size()); 12609 | const T* upper_bound = vec + lud.upper_bound; 12610 | 12611 | while (vec < upper_bound) 12612 | { 12613 | #define exprtk_loop(N) \ 12614 | vec[N] = v; \ 12615 | 12616 | exprtk_loop( 0) exprtk_loop( 1) 12617 | exprtk_loop( 2) exprtk_loop( 3) 12618 | #ifndef exprtk_disable_superscalar_unroll 12619 | exprtk_loop( 4) exprtk_loop( 5) 12620 | exprtk_loop( 6) exprtk_loop( 7) 12621 | exprtk_loop( 8) exprtk_loop( 9) 12622 | exprtk_loop(10) exprtk_loop(11) 12623 | exprtk_loop(12) exprtk_loop(13) 12624 | exprtk_loop(14) exprtk_loop(15) 12625 | #endif 12626 | 12627 | vec += lud.batch_size; 12628 | } 12629 | 12630 | switch (lud.remainder) 12631 | { 12632 | #define case_stmt(N) \ 12633 | case N : *vec++ = v; \ 12634 | exprtk_fallthrough \ 12635 | 12636 | #ifndef exprtk_disable_superscalar_unroll 12637 | case_stmt(15) case_stmt(14) 12638 | case_stmt(13) case_stmt(12) 12639 | case_stmt(11) case_stmt(10) 12640 | case_stmt( 9) case_stmt( 8) 12641 | case_stmt( 7) case_stmt( 6) 12642 | case_stmt( 5) case_stmt( 4) 12643 | #endif 12644 | case_stmt( 3) case_stmt( 2) 12645 | case 1 : *vec++ = v; 12646 | } 12647 | 12648 | #undef exprtk_loop 12649 | #undef case_stmt 12650 | 12651 | return vec_node_ptr_->value(); 12652 | } 12653 | 12654 | vector_node_ptr vec() const exprtk_override 12655 | { 12656 | return vec_node_ptr_; 12657 | } 12658 | 12659 | vector_node_ptr vec() exprtk_override 12660 | { 12661 | return vec_node_ptr_; 12662 | } 12663 | 12664 | inline typename expression_node<T>::node_type type() const exprtk_override 12665 | { 12666 | return expression_node<T>::e_vecvalass; 12667 | } 12668 | 12669 | inline bool valid() const exprtk_override 12670 | { 12671 | return 12672 | vec_node_ptr_ && 12673 | (vds().size() <= vec_node_ptr_->vec_holder().base_size()) && 12674 | binary_node<T>::valid(); 12675 | } 12676 | 12677 | std::size_t size() const exprtk_override 12678 | { 12679 | return vec_node_ptr_->vec_holder().size(); 12680 | } 12681 | 12682 | std::size_t base_size() const exprtk_override 12683 | { 12684 | return vec_node_ptr_->vec_holder().base_size(); 12685 | } 12686 | 12687 | vds_t& vds() exprtk_override 12688 | { 12689 | return vds_; 12690 | } 12691 | 12692 | const vds_t& vds() const exprtk_override 12693 | { 12694 | return vds_; 12695 | } 12696 | 12697 | private: 12698 | 12699 | vector_node<T>* vec_node_ptr_; 12700 | vds_t vds_; 12701 | }; 12702 | 12703 | template <typename T> 12704 | class assignment_vecvec_node exprtk_final 12705 | : public binary_node <T> 12706 | , public vector_interface<T> 12707 | { 12708 | public: 12709 | 12710 | typedef expression_node<T>* expression_ptr; 12711 | typedef vector_node<T>* vector_node_ptr; 12712 | typedef vec_data_store<T> vds_t; 12713 | 12714 | using binary_node<T>::branch; 12715 | 12716 | assignment_vecvec_node(const operator_type& opr, 12717 | expression_ptr branch0, 12718 | expression_ptr branch1) 12719 | : binary_node<T>(opr, branch0, branch1) 12720 | , vec0_node_ptr_(0) 12721 | , vec1_node_ptr_(0) 12722 | , initialised_(false) 12723 | , src_is_ivec_(false) 12724 | { 12725 | if (is_vector_node(branch(0))) 12726 | { 12727 | vec0_node_ptr_ = static_cast<vector_node<T>*>(branch(0)); 12728 | vds() = vec0_node_ptr_->vds(); 12729 | } 12730 | 12731 | if (is_vector_node(branch(1))) 12732 | { 12733 | vec1_node_ptr_ = static_cast<vector_node<T>*>(branch(1)); 12734 | vds_t::match_sizes(vds(),vec1_node_ptr_->vds()); 12735 | } 12736 | else if (is_ivector_node(branch(1))) 12737 | { 12738 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 12739 | 12740 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(1)))) 12741 | { 12742 | vec1_node_ptr_ = vi->vec(); 12743 | 12744 | if (!vi->side_effect()) 12745 | { 12746 | vi->vds() = vds(); 12747 | src_is_ivec_ = true; 12748 | } 12749 | else 12750 | vds_t::match_sizes(vds(),vi->vds()); 12751 | } 12752 | } 12753 | 12754 | initialised_ = 12755 | vec0_node_ptr_ && 12756 | vec1_node_ptr_ && 12757 | (size() <= base_size()) && 12758 | (vds_.size() <= base_size()) && 12759 | binary_node<T>::valid(); 12760 | 12761 | assert(valid()); 12762 | } 12763 | 12764 | inline T value() const exprtk_override 12765 | { 12766 | branch(1)->value(); 12767 | 12768 | if (src_is_ivec_) 12769 | return vec0_node_ptr_->value(); 12770 | 12771 | T* vec0 = vec0_node_ptr_->vds().data(); 12772 | T* vec1 = vec1_node_ptr_->vds().data(); 12773 | 12774 | loop_unroll::details lud(size()); 12775 | const T* upper_bound = vec0 + lud.upper_bound; 12776 | 12777 | while (vec0 < upper_bound) 12778 | { 12779 | #define exprtk_loop(N) \ 12780 | vec0[N] = vec1[N]; \ 12781 | 12782 | exprtk_loop( 0) exprtk_loop( 1) 12783 | exprtk_loop( 2) exprtk_loop( 3) 12784 | #ifndef exprtk_disable_superscalar_unroll 12785 | exprtk_loop( 4) exprtk_loop( 5) 12786 | exprtk_loop( 6) exprtk_loop( 7) 12787 | exprtk_loop( 8) exprtk_loop( 9) 12788 | exprtk_loop(10) exprtk_loop(11) 12789 | exprtk_loop(12) exprtk_loop(13) 12790 | exprtk_loop(14) exprtk_loop(15) 12791 | #endif 12792 | 12793 | vec0 += lud.batch_size; 12794 | vec1 += lud.batch_size; 12795 | } 12796 | 12797 | switch (lud.remainder) 12798 | { 12799 | #define case_stmt(N,fall_through) \ 12800 | case N : *vec0++ = *vec1++; \ 12801 | fall_through \ 12802 | 12803 | #ifndef exprtk_disable_superscalar_unroll 12804 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 12805 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 12806 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 12807 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 12808 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 12809 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 12810 | #endif 12811 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 12812 | case_stmt( 1, (void)0;) 12813 | } 12814 | 12815 | #undef exprtk_loop 12816 | #undef case_stmt 12817 | 12818 | return vec0_node_ptr_->value(); 12819 | } 12820 | 12821 | vector_node_ptr vec() exprtk_override 12822 | { 12823 | return vec0_node_ptr_; 12824 | } 12825 | 12826 | vector_node_ptr vec() const exprtk_override 12827 | { 12828 | return vec0_node_ptr_; 12829 | } 12830 | 12831 | inline typename expression_node<T>::node_type type() const exprtk_override 12832 | { 12833 | return expression_node<T>::e_vecvecass; 12834 | } 12835 | 12836 | inline bool valid() const exprtk_override 12837 | { 12838 | return initialised_; 12839 | } 12840 | 12841 | std::size_t size() const exprtk_override 12842 | { 12843 | return std::min( 12844 | vec0_node_ptr_->vec_holder().size(), 12845 | vec1_node_ptr_->vec_holder().size()); 12846 | } 12847 | 12848 | std::size_t base_size() const exprtk_override 12849 | { 12850 | return std::min( 12851 | vec0_node_ptr_->vec_holder().base_size(), 12852 | vec1_node_ptr_->vec_holder().base_size()); 12853 | } 12854 | 12855 | vds_t& vds() exprtk_override 12856 | { 12857 | return vds_; 12858 | } 12859 | 12860 | const vds_t& vds() const exprtk_override 12861 | { 12862 | return vds_; 12863 | } 12864 | 12865 | private: 12866 | 12867 | vector_node<T>* vec0_node_ptr_; 12868 | vector_node<T>* vec1_node_ptr_; 12869 | bool initialised_; 12870 | bool src_is_ivec_; 12871 | vds_t vds_; 12872 | }; 12873 | 12874 | template <typename T, typename Operation> 12875 | class assignment_op_node exprtk_final : public binary_node<T> 12876 | { 12877 | public: 12878 | 12879 | typedef expression_node<T>* expression_ptr; 12880 | using binary_node<T>::branch; 12881 | 12882 | assignment_op_node(const operator_type& opr, 12883 | expression_ptr branch0, 12884 | expression_ptr branch1) 12885 | : binary_node<T>(opr, branch0, branch1) 12886 | , var_node_ptr_(0) 12887 | { 12888 | if (is_variable_node(branch(0))) 12889 | { 12890 | var_node_ptr_ = static_cast<variable_node<T>*>(branch(0)); 12891 | } 12892 | 12893 | assert(valid()); 12894 | } 12895 | 12896 | inline T value() const exprtk_override 12897 | { 12898 | T& v = var_node_ptr_->ref(); 12899 | v = Operation::process(v,branch(1)->value()); 12900 | 12901 | return v; 12902 | } 12903 | 12904 | inline bool valid() const exprtk_override 12905 | { 12906 | return var_node_ptr_ && binary_node<T>::valid(); 12907 | } 12908 | 12909 | private: 12910 | 12911 | variable_node<T>* var_node_ptr_; 12912 | }; 12913 | 12914 | template <typename T, typename Operation> 12915 | class assignment_vec_elem_op_node exprtk_final : public binary_node<T> 12916 | { 12917 | public: 12918 | 12919 | typedef expression_node<T>* expression_ptr; 12920 | using binary_node<T>::branch; 12921 | 12922 | assignment_vec_elem_op_node(const operator_type& opr, 12923 | expression_ptr branch0, 12924 | expression_ptr branch1) 12925 | : binary_node<T>(opr, branch0, branch1) 12926 | , vec_node_ptr_(0) 12927 | { 12928 | if (is_vector_elem_node(branch(0))) 12929 | { 12930 | vec_node_ptr_ = static_cast<vector_elem_node<T>*>(branch(0)); 12931 | } 12932 | 12933 | assert(valid()); 12934 | } 12935 | 12936 | inline T value() const exprtk_override 12937 | { 12938 | T& v = vec_node_ptr_->ref(); 12939 | v = Operation::process(v,branch(1)->value()); 12940 | 12941 | return v; 12942 | } 12943 | 12944 | inline bool valid() const exprtk_override 12945 | { 12946 | return vec_node_ptr_ && binary_node<T>::valid(); 12947 | } 12948 | 12949 | private: 12950 | 12951 | vector_elem_node<T>* vec_node_ptr_; 12952 | }; 12953 | 12954 | template <typename T, typename Operation> 12955 | class assignment_vec_elem_op_rtc_node exprtk_final : public binary_node<T> 12956 | { 12957 | public: 12958 | 12959 | typedef expression_node<T>* expression_ptr; 12960 | using binary_node<T>::branch; 12961 | 12962 | assignment_vec_elem_op_rtc_node(const operator_type& opr, 12963 | expression_ptr branch0, 12964 | expression_ptr branch1) 12965 | : binary_node<T>(opr, branch0, branch1) 12966 | , vec_node_ptr_(0) 12967 | { 12968 | if (is_vector_elem_rtc_node(branch(0))) 12969 | { 12970 | vec_node_ptr_ = static_cast<vector_elem_rtc_node<T>*>(branch(0)); 12971 | } 12972 | 12973 | assert(valid()); 12974 | } 12975 | 12976 | inline T value() const exprtk_override 12977 | { 12978 | T& v = vec_node_ptr_->ref(); 12979 | v = Operation::process(v,branch(1)->value()); 12980 | 12981 | return v; 12982 | } 12983 | 12984 | inline bool valid() const exprtk_override 12985 | { 12986 | return vec_node_ptr_ && binary_node<T>::valid(); 12987 | } 12988 | 12989 | private: 12990 | 12991 | vector_elem_rtc_node<T>* vec_node_ptr_; 12992 | }; 12993 | 12994 | template <typename T, typename Operation> 12995 | class assignment_vec_celem_op_rtc_node exprtk_final : public binary_node<T> 12996 | { 12997 | public: 12998 | 12999 | typedef expression_node<T>* expression_ptr; 13000 | using binary_node<T>::branch; 13001 | 13002 | assignment_vec_celem_op_rtc_node(const operator_type& opr, 13003 | expression_ptr branch0, 13004 | expression_ptr branch1) 13005 | : binary_node<T>(opr, branch0, branch1) 13006 | , vec_node_ptr_(0) 13007 | { 13008 | if (is_vector_celem_rtc_node(branch(0))) 13009 | { 13010 | vec_node_ptr_ = static_cast<vector_celem_rtc_node<T>*>(branch(0)); 13011 | } 13012 | 13013 | assert(valid()); 13014 | } 13015 | 13016 | inline T value() const exprtk_override 13017 | { 13018 | T& v = vec_node_ptr_->ref(); 13019 | v = Operation::process(v,branch(1)->value()); 13020 | 13021 | return v; 13022 | } 13023 | 13024 | inline bool valid() const exprtk_override 13025 | { 13026 | return vec_node_ptr_ && binary_node<T>::valid(); 13027 | } 13028 | 13029 | private: 13030 | 13031 | vector_celem_rtc_node<T>* vec_node_ptr_; 13032 | }; 13033 | 13034 | template <typename T, typename Operation> 13035 | class assignment_rebasevec_elem_op_node exprtk_final : public binary_node<T> 13036 | { 13037 | public: 13038 | 13039 | typedef expression_node<T>* expression_ptr; 13040 | using binary_node<T>::branch; 13041 | 13042 | assignment_rebasevec_elem_op_node(const operator_type& opr, 13043 | expression_ptr branch0, 13044 | expression_ptr branch1) 13045 | : binary_node<T>(opr, branch0, branch1) 13046 | , rbvec_node_ptr_(0) 13047 | { 13048 | if (is_rebasevector_elem_node(branch(0))) 13049 | { 13050 | rbvec_node_ptr_ = static_cast<rebasevector_elem_node<T>*>(branch(0)); 13051 | } 13052 | 13053 | assert(valid()); 13054 | } 13055 | 13056 | inline T value() const exprtk_override 13057 | { 13058 | T& v = rbvec_node_ptr_->ref(); 13059 | v = Operation::process(v,branch(1)->value()); 13060 | 13061 | return v; 13062 | } 13063 | 13064 | inline bool valid() const exprtk_override 13065 | { 13066 | return rbvec_node_ptr_ && binary_node<T>::valid(); 13067 | } 13068 | 13069 | private: 13070 | 13071 | rebasevector_elem_node<T>* rbvec_node_ptr_; 13072 | }; 13073 | 13074 | template <typename T, typename Operation> 13075 | class assignment_rebasevec_celem_op_node exprtk_final : public binary_node<T> 13076 | { 13077 | public: 13078 | 13079 | typedef expression_node<T>* expression_ptr; 13080 | using binary_node<T>::branch; 13081 | 13082 | assignment_rebasevec_celem_op_node(const operator_type& opr, 13083 | expression_ptr branch0, 13084 | expression_ptr branch1) 13085 | : binary_node<T>(opr, branch0, branch1) 13086 | , rbvec_node_ptr_(0) 13087 | { 13088 | if (is_rebasevector_celem_node(branch(0))) 13089 | { 13090 | rbvec_node_ptr_ = static_cast<rebasevector_celem_node<T>*>(branch(0)); 13091 | } 13092 | 13093 | assert(valid()); 13094 | } 13095 | 13096 | inline T value() const exprtk_override 13097 | { 13098 | T& v = rbvec_node_ptr_->ref(); 13099 | v = Operation::process(v,branch(1)->value()); 13100 | 13101 | return v; 13102 | } 13103 | 13104 | inline bool valid() const exprtk_override 13105 | { 13106 | return rbvec_node_ptr_ && binary_node<T>::valid(); 13107 | } 13108 | 13109 | private: 13110 | 13111 | rebasevector_celem_node<T>* rbvec_node_ptr_; 13112 | }; 13113 | 13114 | template <typename T, typename Operation> 13115 | class assignment_rebasevec_elem_op_rtc_node exprtk_final : public binary_node<T> 13116 | { 13117 | public: 13118 | 13119 | typedef expression_node<T>* expression_ptr; 13120 | using binary_node<T>::branch; 13121 | 13122 | assignment_rebasevec_elem_op_rtc_node(const operator_type& opr, 13123 | expression_ptr branch0, 13124 | expression_ptr branch1) 13125 | : binary_node<T>(opr, branch0, branch1) 13126 | , rbvec_node_ptr_(0) 13127 | { 13128 | if (is_rebasevector_elem_rtc_node(branch(0))) 13129 | { 13130 | rbvec_node_ptr_ = static_cast<rebasevector_elem_rtc_node<T>*>(branch(0)); 13131 | } 13132 | 13133 | assert(valid()); 13134 | } 13135 | 13136 | inline T value() const exprtk_override 13137 | { 13138 | T& v = rbvec_node_ptr_->ref(); 13139 | v = Operation::process(v,branch(1)->value()); 13140 | 13141 | return v; 13142 | } 13143 | 13144 | inline bool valid() const exprtk_override 13145 | { 13146 | return rbvec_node_ptr_ && binary_node<T>::valid(); 13147 | } 13148 | 13149 | private: 13150 | 13151 | rebasevector_elem_rtc_node<T>* rbvec_node_ptr_; 13152 | }; 13153 | 13154 | template <typename T, typename Operation> 13155 | class assignment_rebasevec_celem_op_rtc_node exprtk_final : public binary_node<T> 13156 | { 13157 | public: 13158 | 13159 | typedef expression_node<T>* expression_ptr; 13160 | using binary_node<T>::branch; 13161 | 13162 | assignment_rebasevec_celem_op_rtc_node(const operator_type& opr, 13163 | expression_ptr branch0, 13164 | expression_ptr branch1) 13165 | : binary_node<T>(opr, branch0, branch1) 13166 | , rbvec_node_ptr_(0) 13167 | { 13168 | if (is_rebasevector_celem_rtc_node(branch(0))) 13169 | { 13170 | rbvec_node_ptr_ = static_cast<rebasevector_celem_rtc_node<T>*>(branch(0)); 13171 | } 13172 | 13173 | assert(valid()); 13174 | } 13175 | 13176 | inline T value() const exprtk_override 13177 | { 13178 | T& v = rbvec_node_ptr_->ref(); 13179 | v = Operation::process(v,branch(1)->value()); 13180 | 13181 | return v; 13182 | } 13183 | 13184 | inline bool valid() const exprtk_override 13185 | { 13186 | return rbvec_node_ptr_ && binary_node<T>::valid(); 13187 | } 13188 | 13189 | private: 13190 | 13191 | rebasevector_celem_rtc_node<T>* rbvec_node_ptr_; 13192 | }; 13193 | 13194 | template <typename T, typename Operation> 13195 | class assignment_vec_op_node exprtk_final 13196 | : public binary_node <T> 13197 | , public vector_interface<T> 13198 | { 13199 | public: 13200 | 13201 | typedef expression_node<T>* expression_ptr; 13202 | typedef vector_node<T>* vector_node_ptr; 13203 | typedef vec_data_store<T> vds_t; 13204 | 13205 | using binary_node<T>::branch; 13206 | 13207 | assignment_vec_op_node(const operator_type& opr, 13208 | expression_ptr branch0, 13209 | expression_ptr branch1) 13210 | : binary_node<T>(opr, branch0, branch1) 13211 | , vec_node_ptr_(0) 13212 | { 13213 | if (is_vector_node(branch(0))) 13214 | { 13215 | vec_node_ptr_ = static_cast<vector_node<T>*>(branch(0)); 13216 | vds() = vec_node_ptr_->vds(); 13217 | } 13218 | 13219 | assert(valid()); 13220 | } 13221 | 13222 | inline T value() const exprtk_override 13223 | { 13224 | const T v = branch(1)->value(); 13225 | 13226 | T* vec = vds().data(); 13227 | 13228 | loop_unroll::details lud(size()); 13229 | const T* upper_bound = vec + lud.upper_bound; 13230 | 13231 | while (vec < upper_bound) 13232 | { 13233 | #define exprtk_loop(N) \ 13234 | Operation::assign(vec[N],v); \ 13235 | 13236 | exprtk_loop( 0) exprtk_loop( 1) 13237 | exprtk_loop( 2) exprtk_loop( 3) 13238 | #ifndef exprtk_disable_superscalar_unroll 13239 | exprtk_loop( 4) exprtk_loop( 5) 13240 | exprtk_loop( 6) exprtk_loop( 7) 13241 | exprtk_loop( 8) exprtk_loop( 9) 13242 | exprtk_loop(10) exprtk_loop(11) 13243 | exprtk_loop(12) exprtk_loop(13) 13244 | exprtk_loop(14) exprtk_loop(15) 13245 | #endif 13246 | 13247 | vec += lud.batch_size; 13248 | } 13249 | 13250 | switch (lud.remainder) 13251 | { 13252 | #define case_stmt(N,fall_through) \ 13253 | case N : Operation::assign(*vec++,v); \ 13254 | fall_through \ 13255 | 13256 | #ifndef exprtk_disable_superscalar_unroll 13257 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 13258 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 13259 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 13260 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 13261 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 13262 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 13263 | #endif 13264 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 13265 | case_stmt( 1, (void)0;) 13266 | } 13267 | 13268 | #undef exprtk_loop 13269 | #undef case_stmt 13270 | 13271 | return vec_node_ptr_->value(); 13272 | } 13273 | 13274 | vector_node_ptr vec() const exprtk_override 13275 | { 13276 | return vec_node_ptr_; 13277 | } 13278 | 13279 | vector_node_ptr vec() exprtk_override 13280 | { 13281 | return vec_node_ptr_; 13282 | } 13283 | 13284 | inline typename expression_node<T>::node_type type() const exprtk_override 13285 | { 13286 | return expression_node<T>::e_vecopvalass; 13287 | } 13288 | 13289 | inline bool valid() const exprtk_override 13290 | { 13291 | return 13292 | vec_node_ptr_ && 13293 | (size() <= base_size()) && 13294 | binary_node<T>::valid() ; 13295 | } 13296 | 13297 | std::size_t size() const exprtk_override 13298 | { 13299 | return vec_node_ptr_->vec_holder().size(); 13300 | } 13301 | 13302 | std::size_t base_size() const exprtk_override 13303 | { 13304 | return vec_node_ptr_->vec_holder().base_size(); 13305 | } 13306 | 13307 | vds_t& vds() exprtk_override 13308 | { 13309 | return vds_; 13310 | } 13311 | 13312 | const vds_t& vds() const exprtk_override 13313 | { 13314 | return vds_; 13315 | } 13316 | 13317 | bool side_effect() const exprtk_override 13318 | { 13319 | return true; 13320 | } 13321 | 13322 | private: 13323 | 13324 | vector_node<T>* vec_node_ptr_; 13325 | vds_t vds_; 13326 | }; 13327 | 13328 | template <typename T, typename Operation> 13329 | class assignment_vecvec_op_node exprtk_final 13330 | : public binary_node <T> 13331 | , public vector_interface<T> 13332 | { 13333 | public: 13334 | 13335 | typedef expression_node<T>* expression_ptr; 13336 | typedef vector_node<T>* vector_node_ptr; 13337 | typedef vec_data_store<T> vds_t; 13338 | 13339 | using binary_node<T>::branch; 13340 | 13341 | assignment_vecvec_op_node(const operator_type& opr, 13342 | expression_ptr branch0, 13343 | expression_ptr branch1) 13344 | : binary_node<T>(opr, branch0, branch1) 13345 | , vec0_node_ptr_(0) 13346 | , vec1_node_ptr_(0) 13347 | , initialised_(false) 13348 | { 13349 | if (is_vector_node(branch(0))) 13350 | { 13351 | vec0_node_ptr_ = static_cast<vector_node<T>*>(branch(0)); 13352 | vds() = vec0_node_ptr_->vds(); 13353 | } 13354 | 13355 | if (is_vector_node(branch(1))) 13356 | { 13357 | vec1_node_ptr_ = static_cast<vector_node<T>*>(branch(1)); 13358 | vec1_node_ptr_->vds() = vds(); 13359 | } 13360 | else if (is_ivector_node(branch(1))) 13361 | { 13362 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 13363 | 13364 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(1)))) 13365 | { 13366 | vec1_node_ptr_ = vi->vec(); 13367 | vec1_node_ptr_->vds() = vi->vds(); 13368 | } 13369 | else 13370 | vds_t::match_sizes(vds(),vec1_node_ptr_->vds()); 13371 | } 13372 | 13373 | initialised_ = 13374 | vec0_node_ptr_ && 13375 | vec1_node_ptr_ && 13376 | (size() <= base_size()) && 13377 | binary_node<T>::valid(); 13378 | 13379 | assert(valid()); 13380 | } 13381 | 13382 | inline T value() const exprtk_override 13383 | { 13384 | branch(0)->value(); 13385 | branch(1)->value(); 13386 | 13387 | T* vec0 = vec0_node_ptr_->vds().data(); 13388 | const T* vec1 = vec1_node_ptr_->vds().data(); 13389 | 13390 | loop_unroll::details lud(size()); 13391 | const T* upper_bound = vec0 + lud.upper_bound; 13392 | 13393 | while (vec0 < upper_bound) 13394 | { 13395 | #define exprtk_loop(N) \ 13396 | vec0[N] = Operation::process(vec0[N], vec1[N]); \ 13397 | 13398 | exprtk_loop( 0) exprtk_loop( 1) 13399 | exprtk_loop( 2) exprtk_loop( 3) 13400 | #ifndef exprtk_disable_superscalar_unroll 13401 | exprtk_loop( 4) exprtk_loop( 5) 13402 | exprtk_loop( 6) exprtk_loop( 7) 13403 | exprtk_loop( 8) exprtk_loop( 9) 13404 | exprtk_loop(10) exprtk_loop(11) 13405 | exprtk_loop(12) exprtk_loop(13) 13406 | exprtk_loop(14) exprtk_loop(15) 13407 | #endif 13408 | 13409 | vec0 += lud.batch_size; 13410 | vec1 += lud.batch_size; 13411 | } 13412 | 13413 | int i = 0; 13414 | 13415 | switch (lud.remainder) 13416 | { 13417 | #define case_stmt(N,fall_through) \ 13418 | case N : { vec0[i] = Operation::process(vec0[i], vec1[i]); ++i; } \ 13419 | fall_through \ 13420 | 13421 | #ifndef exprtk_disable_superscalar_unroll 13422 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 13423 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 13424 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 13425 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 13426 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 13427 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 13428 | #endif 13429 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 13430 | case_stmt( 1, (void)0;) 13431 | } 13432 | 13433 | #undef exprtk_loop 13434 | #undef case_stmt 13435 | 13436 | return vec0_node_ptr_->value(); 13437 | } 13438 | 13439 | vector_node_ptr vec() const exprtk_override 13440 | { 13441 | return vec0_node_ptr_; 13442 | } 13443 | 13444 | vector_node_ptr vec() exprtk_override 13445 | { 13446 | return vec0_node_ptr_; 13447 | } 13448 | 13449 | inline typename expression_node<T>::node_type type() const exprtk_override 13450 | { 13451 | return expression_node<T>::e_vecopvecass; 13452 | } 13453 | 13454 | inline bool valid() const exprtk_override 13455 | { 13456 | return initialised_; 13457 | } 13458 | 13459 | std::size_t size() const exprtk_override 13460 | { 13461 | return std::min( 13462 | vec0_node_ptr_->vec_holder().size(), 13463 | vec1_node_ptr_->vec_holder().size()); 13464 | } 13465 | 13466 | std::size_t base_size() const exprtk_override 13467 | { 13468 | return std::min( 13469 | vec0_node_ptr_->vec_holder().base_size(), 13470 | vec1_node_ptr_->vec_holder().base_size()); 13471 | } 13472 | 13473 | vds_t& vds() exprtk_override 13474 | { 13475 | return vds_; 13476 | } 13477 | 13478 | const vds_t& vds() const exprtk_override 13479 | { 13480 | return vds_; 13481 | } 13482 | 13483 | bool side_effect() const exprtk_override 13484 | { 13485 | return true; 13486 | } 13487 | 13488 | private: 13489 | 13490 | vector_node<T>* vec0_node_ptr_; 13491 | vector_node<T>* vec1_node_ptr_; 13492 | bool initialised_; 13493 | vds_t vds_; 13494 | }; 13495 | 13496 | template <typename T> 13497 | struct memory_context_t 13498 | { 13499 | typedef vector_node<T>* vector_node_ptr; 13500 | typedef vector_holder<T> vector_holder_t; 13501 | typedef vector_holder_t* vector_holder_ptr; 13502 | 13503 | memory_context_t() 13504 | : temp_(0) 13505 | , temp_vec_node_(0) 13506 | {} 13507 | 13508 | void clear() 13509 | { 13510 | delete temp_vec_node_; 13511 | delete temp_; 13512 | } 13513 | 13514 | vector_holder_ptr temp_; 13515 | vector_node_ptr temp_vec_node_; 13516 | }; 13517 | 13518 | template <typename T> 13519 | inline memory_context_t<T> make_memory_context(vector_holder<T>& vec_holder, 13520 | vec_data_store<T>& vds) 13521 | { 13522 | memory_context_t<T> result_ctxt; 13523 | 13524 | result_ctxt.temp_ = (vec_holder.rebaseable()) ? 13525 | new vector_holder<T>(vec_holder,vds) : 13526 | new vector_holder<T>(vds) ; 13527 | 13528 | result_ctxt.temp_vec_node_ = new vector_node<T>(vds,result_ctxt.temp_); 13529 | 13530 | return result_ctxt; 13531 | } 13532 | 13533 | template <typename T> 13534 | inline memory_context_t<T> make_memory_context(vector_holder<T>& vec_holder0, 13535 | vector_holder<T>& vec_holder1, 13536 | vec_data_store<T>& vds) 13537 | { 13538 | memory_context_t<T> result_ctxt; 13539 | 13540 | if (!vec_holder0.rebaseable() && !vec_holder1.rebaseable()) 13541 | result_ctxt.temp_ = new vector_holder<T>(vds); 13542 | else if (vec_holder0.rebaseable() && !vec_holder1.rebaseable()) 13543 | result_ctxt.temp_ = new vector_holder<T>(vec_holder0,vds); 13544 | else if (!vec_holder0.rebaseable() && vec_holder1.rebaseable()) 13545 | result_ctxt.temp_ = new vector_holder<T>(vec_holder1,vds); 13546 | else 13547 | { 13548 | result_ctxt.temp_ = (vec_holder0.base_size() >= vec_holder1.base_size()) ? 13549 | new vector_holder<T>(vec_holder0, vds) : 13550 | new vector_holder<T>(vec_holder1, vds) ; 13551 | } 13552 | 13553 | result_ctxt.temp_vec_node_ = new vector_node<T>(vds,result_ctxt.temp_); 13554 | 13555 | return result_ctxt; 13556 | } 13557 | 13558 | template <typename T, typename Operation> 13559 | class vec_binop_vecvec_node exprtk_final 13560 | : public binary_node <T> 13561 | , public vector_interface<T> 13562 | { 13563 | public: 13564 | 13565 | typedef expression_node<T>* expression_ptr; 13566 | typedef vector_node<T>* vector_node_ptr; 13567 | typedef vector_holder<T> vector_holder_t; 13568 | typedef vector_holder_t* vector_holder_ptr; 13569 | typedef vec_data_store<T> vds_t; 13570 | typedef memory_context_t<T> memory_context; 13571 | 13572 | using binary_node<T>::branch; 13573 | 13574 | vec_binop_vecvec_node(const operator_type& opr, 13575 | expression_ptr branch0, 13576 | expression_ptr branch1) 13577 | : binary_node<T>(opr, branch0, branch1) 13578 | , vec0_node_ptr_(0) 13579 | , vec1_node_ptr_(0) 13580 | , initialised_(false) 13581 | { 13582 | bool v0_is_ivec = false; 13583 | bool v1_is_ivec = false; 13584 | 13585 | if (is_vector_node(branch(0))) 13586 | { 13587 | vec0_node_ptr_ = static_cast<vector_node_ptr>(branch(0)); 13588 | } 13589 | else if (is_ivector_node(branch(0))) 13590 | { 13591 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 13592 | 13593 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(0)))) 13594 | { 13595 | vec0_node_ptr_ = vi->vec(); 13596 | v0_is_ivec = true; 13597 | } 13598 | } 13599 | 13600 | if (is_vector_node(branch(1))) 13601 | { 13602 | vec1_node_ptr_ = static_cast<vector_node_ptr>(branch(1)); 13603 | } 13604 | else if (is_ivector_node(branch(1))) 13605 | { 13606 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 13607 | 13608 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(1)))) 13609 | { 13610 | vec1_node_ptr_ = vi->vec(); 13611 | v1_is_ivec = true; 13612 | } 13613 | } 13614 | 13615 | if (vec0_node_ptr_ && vec1_node_ptr_) 13616 | { 13617 | vector_holder<T>& vec0 = vec0_node_ptr_->vec_holder(); 13618 | vector_holder<T>& vec1 = vec1_node_ptr_->vec_holder(); 13619 | 13620 | if (v0_is_ivec && (vec0.base_size() <= vec1.base_size())) 13621 | { 13622 | vds_ = vds_t(vec0_node_ptr_->vds()); 13623 | } 13624 | else if (v1_is_ivec && (vec1.base_size() <= vec0.base_size())) 13625 | { 13626 | vds_ = vds_t(vec1_node_ptr_->vds()); 13627 | } 13628 | else 13629 | { 13630 | vds_ = vds_t(std::min(vec0.base_size(),vec1.base_size())); 13631 | } 13632 | 13633 | memory_context_ = make_memory_context(vec0, vec1, vds()); 13634 | 13635 | initialised_ = 13636 | (size() <= base_size()) && 13637 | binary_node<T>::valid(); 13638 | } 13639 | 13640 | assert(valid()); 13641 | } 13642 | 13643 | ~vec_binop_vecvec_node() 13644 | { 13645 | memory_context_.clear(); 13646 | } 13647 | 13648 | inline T value() const exprtk_override 13649 | { 13650 | branch(0)->value(); 13651 | branch(1)->value(); 13652 | 13653 | const T* vec0 = vec0_node_ptr_->vds().data(); 13654 | const T* vec1 = vec1_node_ptr_->vds().data(); 13655 | T* vec2 = vds().data(); 13656 | 13657 | loop_unroll::details lud(size()); 13658 | const T* upper_bound = vec2 + lud.upper_bound; 13659 | 13660 | while (vec2 < upper_bound) 13661 | { 13662 | #define exprtk_loop(N) \ 13663 | vec2[N] = Operation::process(vec0[N], vec1[N]); \ 13664 | 13665 | exprtk_loop( 0) exprtk_loop( 1) 13666 | exprtk_loop( 2) exprtk_loop( 3) 13667 | #ifndef exprtk_disable_superscalar_unroll 13668 | exprtk_loop( 4) exprtk_loop( 5) 13669 | exprtk_loop( 6) exprtk_loop( 7) 13670 | exprtk_loop( 8) exprtk_loop( 9) 13671 | exprtk_loop(10) exprtk_loop(11) 13672 | exprtk_loop(12) exprtk_loop(13) 13673 | exprtk_loop(14) exprtk_loop(15) 13674 | #endif 13675 | 13676 | vec0 += lud.batch_size; 13677 | vec1 += lud.batch_size; 13678 | vec2 += lud.batch_size; 13679 | } 13680 | 13681 | int i = 0; 13682 | 13683 | switch (lud.remainder) 13684 | { 13685 | #define case_stmt(N) \ 13686 | case N : { vec2[i] = Operation::process(vec0[i], vec1[i]); ++i; } \ 13687 | exprtk_fallthrough \ 13688 | 13689 | #ifndef exprtk_disable_superscalar_unroll 13690 | case_stmt(15) case_stmt(14) 13691 | case_stmt(13) case_stmt(12) 13692 | case_stmt(11) case_stmt(10) 13693 | case_stmt( 9) case_stmt( 8) 13694 | case_stmt( 7) case_stmt( 6) 13695 | case_stmt( 5) case_stmt( 4) 13696 | #endif 13697 | case_stmt( 3) case_stmt( 2) 13698 | case_stmt( 1) 13699 | default: break; 13700 | } 13701 | 13702 | #undef exprtk_loop 13703 | #undef case_stmt 13704 | 13705 | return (vds().data())[0]; 13706 | } 13707 | 13708 | vector_node_ptr vec() const exprtk_override 13709 | { 13710 | return memory_context_.temp_vec_node_; 13711 | } 13712 | 13713 | vector_node_ptr vec() exprtk_override 13714 | { 13715 | return memory_context_.temp_vec_node_; 13716 | } 13717 | 13718 | inline typename expression_node<T>::node_type type() const exprtk_override 13719 | { 13720 | return expression_node<T>::e_vecvecarith; 13721 | } 13722 | 13723 | inline bool valid() const exprtk_override 13724 | { 13725 | return initialised_; 13726 | } 13727 | 13728 | std::size_t size() const exprtk_override 13729 | { 13730 | return std::min( 13731 | vec0_node_ptr_->vec_holder().size(), 13732 | vec1_node_ptr_->vec_holder().size()); 13733 | } 13734 | 13735 | std::size_t base_size() const exprtk_override 13736 | { 13737 | return std::min( 13738 | vec0_node_ptr_->vec_holder().base_size(), 13739 | vec1_node_ptr_->vec_holder().base_size()); 13740 | } 13741 | 13742 | vds_t& vds() exprtk_override 13743 | { 13744 | return vds_; 13745 | } 13746 | 13747 | const vds_t& vds() const exprtk_override 13748 | { 13749 | return vds_; 13750 | } 13751 | 13752 | private: 13753 | 13754 | vector_node_ptr vec0_node_ptr_; 13755 | vector_node_ptr vec1_node_ptr_; 13756 | bool initialised_; 13757 | vds_t vds_; 13758 | memory_context memory_context_; 13759 | }; 13760 | 13761 | template <typename T, typename Operation> 13762 | class vec_binop_vecval_node exprtk_final 13763 | : public binary_node <T> 13764 | , public vector_interface<T> 13765 | { 13766 | public: 13767 | 13768 | typedef expression_node<T>* expression_ptr; 13769 | typedef vector_node<T>* vector_node_ptr; 13770 | typedef vector_holder<T> vector_holder_t; 13771 | typedef vector_holder_t* vector_holder_ptr; 13772 | typedef vec_data_store<T> vds_t; 13773 | typedef memory_context_t<T> memory_context; 13774 | 13775 | using binary_node<T>::branch; 13776 | 13777 | vec_binop_vecval_node(const operator_type& opr, 13778 | expression_ptr branch0, 13779 | expression_ptr branch1) 13780 | : binary_node<T>(opr, branch0, branch1) 13781 | , vec0_node_ptr_(0) 13782 | { 13783 | bool v0_is_ivec = false; 13784 | 13785 | if (is_vector_node(branch(0))) 13786 | { 13787 | vec0_node_ptr_ = static_cast<vector_node_ptr>(branch(0)); 13788 | } 13789 | else if (is_ivector_node(branch(0))) 13790 | { 13791 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 13792 | 13793 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(0)))) 13794 | { 13795 | vec0_node_ptr_ = vi->vec(); 13796 | v0_is_ivec = true; 13797 | } 13798 | } 13799 | 13800 | if (vec0_node_ptr_) 13801 | { 13802 | if (v0_is_ivec) 13803 | vds() = vec0_node_ptr_->vds(); 13804 | else 13805 | vds() = vds_t(vec0_node_ptr_->base_size()); 13806 | 13807 | memory_context_ = make_memory_context(vec0_node_ptr_->vec_holder(), vds()); 13808 | } 13809 | 13810 | assert(valid()); 13811 | } 13812 | 13813 | ~vec_binop_vecval_node() 13814 | { 13815 | memory_context_.clear(); 13816 | } 13817 | 13818 | inline T value() const exprtk_override 13819 | { 13820 | branch(0)->value(); 13821 | const T v = branch(1)->value(); 13822 | 13823 | const T* vec0 = vec0_node_ptr_->vds().data(); 13824 | T* vec1 = vds().data(); 13825 | 13826 | loop_unroll::details lud(size()); 13827 | const T* upper_bound = vec0 + lud.upper_bound; 13828 | 13829 | while (vec0 < upper_bound) 13830 | { 13831 | #define exprtk_loop(N) \ 13832 | vec1[N] = Operation::process(vec0[N], v); \ 13833 | 13834 | exprtk_loop( 0) exprtk_loop( 1) 13835 | exprtk_loop( 2) exprtk_loop( 3) 13836 | #ifndef exprtk_disable_superscalar_unroll 13837 | exprtk_loop( 4) exprtk_loop( 5) 13838 | exprtk_loop( 6) exprtk_loop( 7) 13839 | exprtk_loop( 8) exprtk_loop( 9) 13840 | exprtk_loop(10) exprtk_loop(11) 13841 | exprtk_loop(12) exprtk_loop(13) 13842 | exprtk_loop(14) exprtk_loop(15) 13843 | #endif 13844 | 13845 | vec0 += lud.batch_size; 13846 | vec1 += lud.batch_size; 13847 | } 13848 | 13849 | int i = 0; 13850 | 13851 | switch (lud.remainder) 13852 | { 13853 | #define case_stmt(N,fall_through) \ 13854 | case N : { vec1[i] = Operation::process(vec0[i], v); ++i; } \ 13855 | fall_through \ 13856 | 13857 | #ifndef exprtk_disable_superscalar_unroll 13858 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 13859 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 13860 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 13861 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 13862 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 13863 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 13864 | #endif 13865 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 13866 | case_stmt( 1, (void)0;) 13867 | } 13868 | 13869 | #undef exprtk_loop 13870 | #undef case_stmt 13871 | 13872 | return (vds().data())[0]; 13873 | } 13874 | 13875 | vector_node_ptr vec() const exprtk_override 13876 | { 13877 | return memory_context_.temp_vec_node_; 13878 | } 13879 | 13880 | vector_node_ptr vec() exprtk_override 13881 | { 13882 | return memory_context_.temp_vec_node_; 13883 | } 13884 | 13885 | inline typename expression_node<T>::node_type type() const exprtk_override 13886 | { 13887 | return expression_node<T>::e_vecvalarith; 13888 | } 13889 | 13890 | inline bool valid() const exprtk_override 13891 | { 13892 | return 13893 | vec0_node_ptr_ && 13894 | (size() <= base_size()) && 13895 | binary_node<T>::valid(); 13896 | } 13897 | 13898 | std::size_t size() const exprtk_override 13899 | { 13900 | return vec0_node_ptr_->size(); 13901 | } 13902 | 13903 | std::size_t base_size() const exprtk_override 13904 | { 13905 | return vec0_node_ptr_->vec_holder().base_size(); 13906 | } 13907 | 13908 | vds_t& vds() exprtk_override 13909 | { 13910 | return vds_; 13911 | } 13912 | 13913 | const vds_t& vds() const exprtk_override 13914 | { 13915 | return vds_; 13916 | } 13917 | 13918 | private: 13919 | 13920 | vector_node_ptr vec0_node_ptr_; 13921 | vds_t vds_; 13922 | memory_context memory_context_; 13923 | }; 13924 | 13925 | template <typename T, typename Operation> 13926 | class vec_binop_valvec_node exprtk_final 13927 | : public binary_node <T> 13928 | , public vector_interface<T> 13929 | { 13930 | public: 13931 | 13932 | typedef expression_node<T>* expression_ptr; 13933 | typedef vector_node<T>* vector_node_ptr; 13934 | typedef vector_holder<T> vector_holder_t; 13935 | typedef vector_holder_t* vector_holder_ptr; 13936 | typedef vec_data_store<T> vds_t; 13937 | typedef memory_context_t<T> memory_context; 13938 | 13939 | using binary_node<T>::branch; 13940 | 13941 | vec_binop_valvec_node(const operator_type& opr, 13942 | expression_ptr branch0, 13943 | expression_ptr branch1) 13944 | : binary_node<T>(opr, branch0, branch1) 13945 | , vec1_node_ptr_(0) 13946 | { 13947 | bool v1_is_ivec = false; 13948 | 13949 | if (is_vector_node(branch(1))) 13950 | { 13951 | vec1_node_ptr_ = static_cast<vector_node_ptr>(branch(1)); 13952 | } 13953 | else if (is_ivector_node(branch(1))) 13954 | { 13955 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 13956 | 13957 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(1)))) 13958 | { 13959 | vec1_node_ptr_ = vi->vec(); 13960 | v1_is_ivec = true; 13961 | } 13962 | } 13963 | 13964 | if (vec1_node_ptr_) 13965 | { 13966 | if (v1_is_ivec) 13967 | vds() = vec1_node_ptr_->vds(); 13968 | else 13969 | vds() = vds_t(vec1_node_ptr_->base_size()); 13970 | 13971 | memory_context_ = make_memory_context(vec1_node_ptr_->vec_holder(), vds()); 13972 | } 13973 | 13974 | assert(valid()); 13975 | } 13976 | 13977 | ~vec_binop_valvec_node() 13978 | { 13979 | memory_context_.clear(); 13980 | } 13981 | 13982 | inline T value() const exprtk_override 13983 | { 13984 | const T v = branch(0)->value(); 13985 | branch(1)->value(); 13986 | 13987 | T* vec0 = vds().data(); 13988 | const T* vec1 = vec1_node_ptr_->vds().data(); 13989 | 13990 | loop_unroll::details lud(size()); 13991 | const T* upper_bound = vec0 + lud.upper_bound; 13992 | 13993 | while (vec0 < upper_bound) 13994 | { 13995 | #define exprtk_loop(N) \ 13996 | vec0[N] = Operation::process(v, vec1[N]); \ 13997 | 13998 | exprtk_loop( 0) exprtk_loop( 1) 13999 | exprtk_loop( 2) exprtk_loop( 3) 14000 | #ifndef exprtk_disable_superscalar_unroll 14001 | exprtk_loop( 4) exprtk_loop( 5) 14002 | exprtk_loop( 6) exprtk_loop( 7) 14003 | exprtk_loop( 8) exprtk_loop( 9) 14004 | exprtk_loop(10) exprtk_loop(11) 14005 | exprtk_loop(12) exprtk_loop(13) 14006 | exprtk_loop(14) exprtk_loop(15) 14007 | #endif 14008 | 14009 | vec0 += lud.batch_size; 14010 | vec1 += lud.batch_size; 14011 | } 14012 | 14013 | int i = 0; 14014 | 14015 | switch (lud.remainder) 14016 | { 14017 | #define case_stmt(N,fall_through) \ 14018 | case N : { vec0[i] = Operation::process(v, vec1[i]); ++i; } \ 14019 | fall_through \ 14020 | 14021 | #ifndef exprtk_disable_superscalar_unroll 14022 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 14023 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 14024 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 14025 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 14026 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 14027 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 14028 | #endif 14029 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 14030 | case_stmt( 1, (void)0;) 14031 | } 14032 | 14033 | #undef exprtk_loop 14034 | #undef case_stmt 14035 | 14036 | return (vds().data())[0]; 14037 | } 14038 | 14039 | vector_node_ptr vec() const exprtk_override 14040 | { 14041 | return memory_context_.temp_vec_node_; 14042 | } 14043 | 14044 | vector_node_ptr vec() exprtk_override 14045 | { 14046 | return memory_context_.temp_vec_node_; 14047 | } 14048 | 14049 | inline typename expression_node<T>::node_type type() const exprtk_override 14050 | { 14051 | return expression_node<T>::e_vecvalarith; 14052 | } 14053 | 14054 | inline bool valid() const exprtk_override 14055 | { 14056 | return 14057 | vec1_node_ptr_ && 14058 | (size() <= base_size()) && 14059 | (vds_.size() <= base_size()) && 14060 | binary_node<T>::valid(); 14061 | } 14062 | 14063 | std::size_t size() const exprtk_override 14064 | { 14065 | return vec1_node_ptr_->vec_holder().size(); 14066 | } 14067 | 14068 | std::size_t base_size() const exprtk_override 14069 | { 14070 | return vec1_node_ptr_->vec_holder().base_size(); 14071 | } 14072 | 14073 | vds_t& vds() exprtk_override 14074 | { 14075 | return vds_; 14076 | } 14077 | 14078 | const vds_t& vds() const exprtk_override 14079 | { 14080 | return vds_; 14081 | } 14082 | 14083 | private: 14084 | 14085 | vector_node_ptr vec1_node_ptr_; 14086 | vds_t vds_; 14087 | memory_context memory_context_; 14088 | }; 14089 | 14090 | template <typename T, typename Operation> 14091 | class unary_vector_node exprtk_final 14092 | : public unary_node <T> 14093 | , public vector_interface<T> 14094 | { 14095 | public: 14096 | 14097 | typedef expression_node<T>* expression_ptr; 14098 | typedef vector_node<T>* vector_node_ptr; 14099 | typedef vector_holder<T> vector_holder_t; 14100 | typedef vector_holder_t* vector_holder_ptr; 14101 | typedef vec_data_store<T> vds_t; 14102 | typedef memory_context_t<T> memory_context; 14103 | 14104 | using expression_node<T>::branch; 14105 | 14106 | unary_vector_node(const operator_type& opr, expression_ptr branch0) 14107 | : unary_node<T>(opr, branch0) 14108 | , vec0_node_ptr_(0) 14109 | { 14110 | bool vec0_is_ivec = false; 14111 | 14112 | if (is_vector_node(branch(0))) 14113 | { 14114 | vec0_node_ptr_ = static_cast<vector_node_ptr>(branch(0)); 14115 | } 14116 | else if (is_ivector_node(branch(0))) 14117 | { 14118 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 14119 | 14120 | if (0 != (vi = dynamic_cast<vector_interface<T>*>(branch(0)))) 14121 | { 14122 | vec0_node_ptr_ = vi->vec(); 14123 | vec0_is_ivec = true; 14124 | } 14125 | } 14126 | 14127 | if (vec0_node_ptr_) 14128 | { 14129 | if (vec0_is_ivec) 14130 | vds_ = vec0_node_ptr_->vds(); 14131 | else 14132 | vds_ = vds_t(vec0_node_ptr_->base_size()); 14133 | 14134 | memory_context_ = make_memory_context(vec0_node_ptr_->vec_holder(), vds()); 14135 | } 14136 | 14137 | assert(valid()); 14138 | } 14139 | 14140 | ~unary_vector_node() 14141 | { 14142 | memory_context_.clear(); 14143 | } 14144 | 14145 | inline T value() const exprtk_override 14146 | { 14147 | branch()->value(); 14148 | 14149 | const T* vec0 = vec0_node_ptr_->vds().data(); 14150 | T* vec1 = vds().data(); 14151 | 14152 | loop_unroll::details lud(size()); 14153 | const T* upper_bound = vec0 + lud.upper_bound; 14154 | 14155 | while (vec0 < upper_bound) 14156 | { 14157 | #define exprtk_loop(N) \ 14158 | vec1[N] = Operation::process(vec0[N]); \ 14159 | 14160 | exprtk_loop( 0) exprtk_loop( 1) 14161 | exprtk_loop( 2) exprtk_loop( 3) 14162 | #ifndef exprtk_disable_superscalar_unroll 14163 | exprtk_loop( 4) exprtk_loop( 5) 14164 | exprtk_loop( 6) exprtk_loop( 7) 14165 | exprtk_loop( 8) exprtk_loop( 9) 14166 | exprtk_loop(10) exprtk_loop(11) 14167 | exprtk_loop(12) exprtk_loop(13) 14168 | exprtk_loop(14) exprtk_loop(15) 14169 | #endif 14170 | 14171 | vec0 += lud.batch_size; 14172 | vec1 += lud.batch_size; 14173 | } 14174 | 14175 | int i = 0; 14176 | 14177 | switch (lud.remainder) 14178 | { 14179 | #define case_stmt(N) \ 14180 | case N : { vec1[i] = Operation::process(vec0[i]); ++i; } \ 14181 | exprtk_fallthrough \ 14182 | 14183 | #ifndef exprtk_disable_superscalar_unroll 14184 | case_stmt(15) case_stmt(14) 14185 | case_stmt(13) case_stmt(12) 14186 | case_stmt(11) case_stmt(10) 14187 | case_stmt( 9) case_stmt( 8) 14188 | case_stmt( 7) case_stmt( 6) 14189 | case_stmt( 5) case_stmt( 4) 14190 | #endif 14191 | case_stmt( 3) case_stmt( 2) 14192 | case_stmt( 1) 14193 | default: break; 14194 | } 14195 | 14196 | #undef exprtk_loop 14197 | #undef case_stmt 14198 | 14199 | return (vds().data())[0]; 14200 | } 14201 | 14202 | vector_node_ptr vec() const exprtk_override 14203 | { 14204 | return memory_context_.temp_vec_node_; 14205 | } 14206 | 14207 | vector_node_ptr vec() exprtk_override 14208 | { 14209 | return memory_context_.temp_vec_node_; 14210 | } 14211 | 14212 | inline typename expression_node<T>::node_type type() const exprtk_override 14213 | { 14214 | return expression_node<T>::e_vecunaryop; 14215 | } 14216 | 14217 | inline bool valid() const exprtk_override 14218 | { 14219 | return vec0_node_ptr_ && unary_node<T>::valid(); 14220 | } 14221 | 14222 | std::size_t size() const exprtk_override 14223 | { 14224 | return vec0_node_ptr_->vec_holder().size(); 14225 | } 14226 | 14227 | std::size_t base_size() const exprtk_override 14228 | { 14229 | return vec0_node_ptr_->vec_holder().base_size(); 14230 | } 14231 | 14232 | vds_t& vds() exprtk_override 14233 | { 14234 | return vds_; 14235 | } 14236 | 14237 | const vds_t& vds() const exprtk_override 14238 | { 14239 | return vds_; 14240 | } 14241 | 14242 | private: 14243 | 14244 | vector_node_ptr vec0_node_ptr_; 14245 | vds_t vds_; 14246 | memory_context memory_context_; 14247 | }; 14248 | 14249 | template <typename T> 14250 | class conditional_vector_node exprtk_final 14251 | : public expression_node <T> 14252 | , public vector_interface<T> 14253 | { 14254 | public: 14255 | 14256 | typedef expression_node <T>* expression_ptr; 14257 | typedef vector_interface<T>* vec_interface_ptr; 14258 | typedef vector_node <T>* vector_node_ptr; 14259 | typedef vector_holder <T> vector_holder_t; 14260 | typedef vector_holder_t* vector_holder_ptr; 14261 | typedef vec_data_store <T> vds_t; 14262 | typedef memory_context_t<T> memory_context; 14263 | typedef std::pair<expression_ptr,bool> branch_t; 14264 | 14265 | conditional_vector_node(expression_ptr condition, 14266 | expression_ptr consequent, 14267 | expression_ptr alternative) 14268 | : consequent_node_ptr_ (0) 14269 | , alternative_node_ptr_(0) 14270 | , temp_vec_node_ (0) 14271 | , temp_ (0) 14272 | , result_vec_size_ (0) 14273 | , initialised_ (false) 14274 | { 14275 | construct_branch_pair(condition_ , condition ); 14276 | construct_branch_pair(consequent_ , consequent ); 14277 | construct_branch_pair(alternative_, alternative); 14278 | 14279 | if (details::is_ivector_node(consequent_.first)) 14280 | { 14281 | vec_interface_ptr ivec_ptr = dynamic_cast<vec_interface_ptr>(consequent_.first); 14282 | 14283 | if (0 != ivec_ptr) 14284 | { 14285 | consequent_node_ptr_ = ivec_ptr->vec(); 14286 | } 14287 | } 14288 | 14289 | if (details::is_ivector_node(alternative_.first)) 14290 | { 14291 | vec_interface_ptr ivec_ptr = dynamic_cast<vec_interface_ptr>(alternative_.first); 14292 | 14293 | if (0 != ivec_ptr) 14294 | { 14295 | alternative_node_ptr_ = ivec_ptr->vec(); 14296 | } 14297 | } 14298 | 14299 | if (consequent_node_ptr_ && alternative_node_ptr_) 14300 | { 14301 | const std::size_t vec_size = 14302 | std::max(consequent_node_ptr_ ->vec_holder().base_size(), 14303 | alternative_node_ptr_->vec_holder().base_size()); 14304 | 14305 | vds_ = vds_t(vec_size); 14306 | memory_context_ = make_memory_context( 14307 | consequent_node_ptr_ ->vec_holder(), 14308 | alternative_node_ptr_->vec_holder(), 14309 | vds()); 14310 | 14311 | initialised_ = (vec_size > 0); 14312 | } 14313 | 14314 | assert(initialised_); 14315 | } 14316 | 14317 | ~conditional_vector_node() 14318 | { 14319 | memory_context_.clear(); 14320 | } 14321 | 14322 | inline T value() const exprtk_override 14323 | { 14324 | T result = T(0); 14325 | T* source_vector = 0; 14326 | T* result_vector = vds().data(); 14327 | 14328 | if (is_true(condition_)) 14329 | { 14330 | result = consequent_.first->value(); 14331 | source_vector = consequent_node_ptr_->vds().data(); 14332 | result_vec_size_ = consequent_node_ptr_->size(); 14333 | } 14334 | else 14335 | { 14336 | result = alternative_.first->value(); 14337 | source_vector = alternative_node_ptr_->vds().data(); 14338 | result_vec_size_ = alternative_node_ptr_->size(); 14339 | } 14340 | 14341 | for (std::size_t i = 0; i < result_vec_size_; ++i) 14342 | { 14343 | result_vector[i] = source_vector[i]; 14344 | } 14345 | 14346 | return result; 14347 | } 14348 | 14349 | vector_node_ptr vec() const exprtk_override 14350 | { 14351 | return memory_context_.temp_vec_node_; 14352 | } 14353 | 14354 | vector_node_ptr vec() exprtk_override 14355 | { 14356 | return memory_context_.temp_vec_node_; 14357 | } 14358 | 14359 | inline typename expression_node<T>::node_type type() const exprtk_override 14360 | { 14361 | return expression_node<T>::e_vecondition; 14362 | } 14363 | 14364 | inline bool valid() const exprtk_override 14365 | { 14366 | return 14367 | initialised_ && 14368 | condition_ .first && condition_ .first->valid() && 14369 | consequent_ .first && consequent_ .first->valid() && 14370 | alternative_.first && alternative_.first->valid() && 14371 | size() <= base_size(); 14372 | } 14373 | 14374 | std::size_t size() const exprtk_override 14375 | { 14376 | return result_vec_size_; 14377 | } 14378 | 14379 | std::size_t base_size() const exprtk_override 14380 | { 14381 | return std::min( 14382 | consequent_node_ptr_ ->vec_holder().base_size(), 14383 | alternative_node_ptr_->vec_holder().base_size()); 14384 | } 14385 | 14386 | vds_t& vds() exprtk_override 14387 | { 14388 | return vds_; 14389 | } 14390 | 14391 | const vds_t& vds() const exprtk_override 14392 | { 14393 | return vds_; 14394 | } 14395 | 14396 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 14397 | { 14398 | expression_node<T>::ndb_t::collect(condition_ , node_delete_list); 14399 | expression_node<T>::ndb_t::collect(consequent_ , node_delete_list); 14400 | expression_node<T>::ndb_t::collect(alternative_ , node_delete_list); 14401 | } 14402 | 14403 | std::size_t node_depth() const exprtk_override 14404 | { 14405 | return expression_node<T>::ndb_t::compute_node_depth 14406 | (condition_, consequent_, alternative_); 14407 | } 14408 | 14409 | private: 14410 | 14411 | branch_t condition_; 14412 | branch_t consequent_; 14413 | branch_t alternative_; 14414 | vector_node_ptr consequent_node_ptr_; 14415 | vector_node_ptr alternative_node_ptr_; 14416 | vector_node_ptr temp_vec_node_; 14417 | vector_holder_ptr temp_; 14418 | vds_t vds_; 14419 | mutable std::size_t result_vec_size_; 14420 | bool initialised_; 14421 | memory_context memory_context_; 14422 | }; 14423 | 14424 | template <typename T> 14425 | class scand_node exprtk_final : public binary_node<T> 14426 | { 14427 | public: 14428 | 14429 | typedef expression_node<T>* expression_ptr; 14430 | using binary_node<T>::branch; 14431 | 14432 | scand_node(const operator_type& opr, 14433 | expression_ptr branch0, 14434 | expression_ptr branch1) 14435 | : binary_node<T>(opr, branch0, branch1) 14436 | { 14437 | assert(binary_node<T>::valid()); 14438 | } 14439 | 14440 | inline T value() const exprtk_override 14441 | { 14442 | return ( 14443 | std::not_equal_to<T>() 14444 | (T(0),branch(0)->value()) && 14445 | std::not_equal_to<T>() 14446 | (T(0),branch(1)->value()) 14447 | ) ? T(1) : T(0); 14448 | } 14449 | }; 14450 | 14451 | template <typename T> 14452 | class scor_node exprtk_final : public binary_node<T> 14453 | { 14454 | public: 14455 | 14456 | typedef expression_node<T>* expression_ptr; 14457 | using binary_node<T>::branch; 14458 | 14459 | scor_node(const operator_type& opr, 14460 | expression_ptr branch0, 14461 | expression_ptr branch1) 14462 | : binary_node<T>(opr, branch0, branch1) 14463 | { 14464 | assert(binary_node<T>::valid()); 14465 | } 14466 | 14467 | inline T value() const exprtk_override 14468 | { 14469 | return ( 14470 | std::not_equal_to<T>() 14471 | (T(0),branch(0)->value()) || 14472 | std::not_equal_to<T>() 14473 | (T(0),branch(1)->value()) 14474 | ) ? T(1) : T(0); 14475 | } 14476 | }; 14477 | 14478 | template <typename T, typename IFunction, std::size_t N> 14479 | class function_N_node exprtk_final : public expression_node<T> 14480 | { 14481 | public: 14482 | 14483 | // Function of N parameters. 14484 | typedef expression_node<T>* expression_ptr; 14485 | typedef std::pair<expression_ptr,bool> branch_t; 14486 | typedef IFunction ifunction; 14487 | 14488 | explicit function_N_node(ifunction* func) 14489 | : function_((N == func->param_count) ? func : reinterpret_cast<ifunction*>(0)) 14490 | , parameter_count_(func->param_count) 14491 | , initialised_(false) 14492 | {} 14493 | 14494 | template <std::size_t NumBranches> 14495 | bool init_branches(expression_ptr (&b)[NumBranches]) 14496 | { 14497 | // Needed for incompetent and broken msvc compiler versions 14498 | #ifdef _MSC_VER 14499 | #pragma warning(push) 14500 | #pragma warning(disable: 4127) 14501 | #endif 14502 | 14503 | if (N != NumBranches) 14504 | { 14505 | return false; 14506 | } 14507 | 14508 | for (std::size_t i = 0; i < NumBranches; ++i) 14509 | { 14510 | if (b[i] && b[i]->valid()) 14511 | branch_[i] = std::make_pair(b[i],branch_deletable(b[i])); 14512 | else 14513 | return false; 14514 | } 14515 | 14516 | initialised_ = function_; 14517 | assert(valid()); 14518 | return initialised_; 14519 | 14520 | #ifdef _MSC_VER 14521 | #pragma warning(pop) 14522 | #endif 14523 | } 14524 | 14525 | inline bool operator <(const function_N_node<T,IFunction,N>& fn) const 14526 | { 14527 | return this < (&fn); 14528 | } 14529 | 14530 | inline T value() const exprtk_override 14531 | { 14532 | // Needed for incompetent and broken msvc compiler versions 14533 | #ifdef _MSC_VER 14534 | #pragma warning(push) 14535 | #pragma warning(disable: 4127) 14536 | #endif 14537 | 14538 | T v[N]; 14539 | evaluate_branches<T,N>::execute(v,branch_); 14540 | return invoke<T,N>::execute(*function_,v); 14541 | 14542 | #ifdef _MSC_VER 14543 | #pragma warning(pop) 14544 | #endif 14545 | } 14546 | 14547 | inline typename expression_node<T>::node_type type() const exprtk_override 14548 | { 14549 | return expression_node<T>::e_function; 14550 | } 14551 | 14552 | inline bool valid() const exprtk_override 14553 | { 14554 | return initialised_; 14555 | } 14556 | 14557 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 14558 | { 14559 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 14560 | } 14561 | 14562 | std::size_t node_depth() const exprtk_override 14563 | { 14564 | return expression_node<T>::ndb_t::template compute_node_depth<N>(branch_); 14565 | } 14566 | 14567 | template <typename T_, std::size_t BranchCount> 14568 | struct evaluate_branches 14569 | { 14570 | static inline void execute(T_ (&v)[BranchCount], const branch_t (&b)[BranchCount]) 14571 | { 14572 | for (std::size_t i = 0; i < BranchCount; ++i) 14573 | { 14574 | v[i] = b[i].first->value(); 14575 | } 14576 | } 14577 | }; 14578 | 14579 | template <typename T_> 14580 | struct evaluate_branches <T_,6> 14581 | { 14582 | static inline void execute(T_ (&v)[6], const branch_t (&b)[6]) 14583 | { 14584 | v[0] = b[0].first->value(); 14585 | v[1] = b[1].first->value(); 14586 | v[2] = b[2].first->value(); 14587 | v[3] = b[3].first->value(); 14588 | v[4] = b[4].first->value(); 14589 | v[5] = b[5].first->value(); 14590 | } 14591 | }; 14592 | 14593 | template <typename T_> 14594 | struct evaluate_branches <T_,5> 14595 | { 14596 | static inline void execute(T_ (&v)[5], const branch_t (&b)[5]) 14597 | { 14598 | v[0] = b[0].first->value(); 14599 | v[1] = b[1].first->value(); 14600 | v[2] = b[2].first->value(); 14601 | v[3] = b[3].first->value(); 14602 | v[4] = b[4].first->value(); 14603 | } 14604 | }; 14605 | 14606 | template <typename T_> 14607 | struct evaluate_branches <T_,4> 14608 | { 14609 | static inline void execute(T_ (&v)[4], const branch_t (&b)[4]) 14610 | { 14611 | v[0] = b[0].first->value(); 14612 | v[1] = b[1].first->value(); 14613 | v[2] = b[2].first->value(); 14614 | v[3] = b[3].first->value(); 14615 | } 14616 | }; 14617 | 14618 | template <typename T_> 14619 | struct evaluate_branches <T_,3> 14620 | { 14621 | static inline void execute(T_ (&v)[3], const branch_t (&b)[3]) 14622 | { 14623 | v[0] = b[0].first->value(); 14624 | v[1] = b[1].first->value(); 14625 | v[2] = b[2].first->value(); 14626 | } 14627 | }; 14628 | 14629 | template <typename T_> 14630 | struct evaluate_branches <T_,2> 14631 | { 14632 | static inline void execute(T_ (&v)[2], const branch_t (&b)[2]) 14633 | { 14634 | v[0] = b[0].first->value(); 14635 | v[1] = b[1].first->value(); 14636 | } 14637 | }; 14638 | 14639 | template <typename T_> 14640 | struct evaluate_branches <T_,1> 14641 | { 14642 | static inline void execute(T_ (&v)[1], const branch_t (&b)[1]) 14643 | { 14644 | v[0] = b[0].first->value(); 14645 | } 14646 | }; 14647 | 14648 | template <typename T_, std::size_t ParamCount> 14649 | struct invoke { static inline T execute(ifunction&, branch_t (&)[ParamCount]) { return std::numeric_limits<T_>::quiet_NaN(); } }; 14650 | 14651 | template <typename T_> 14652 | struct invoke<T_,20> 14653 | { 14654 | static inline T_ execute(ifunction& f, T_ (&v)[20]) 14655 | { return f(v[0],v[1],v[2],v[3],v[4],v[5],v[6],v[7],v[8],v[9],v[10],v[11],v[12],v[13],v[14],v[15],v[16],v[17],v[18],v[19]); } 14656 | }; 14657 | 14658 | template <typename T_> 14659 | struct invoke<T_,19> 14660 | { 14661 | static inline T_ execute(ifunction& f, T_ (&v)[19]) 14662 | { return f(v[0],v[1],v[2],v[3],v[4],v[5],v[6],v[7],v[8],v[9],v[10],v[11],v[12],v[13],v[14],v[15],v[16],v[17],v[18]); } 14663 | }; 14664 | 14665 | template <typename T_> 14666 | struct invoke<T_,18> 14667 | { 14668 | static inline T_ execute(ifunction& f, T_ (&v)[18]) 14669 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12], v[13], v[14], v[15], v[16], v[17]); } 14670 | }; 14671 | 14672 | template <typename T_> 14673 | struct invoke<T_,17> 14674 | { 14675 | static inline T_ execute(ifunction& f, T_ (&v)[17]) 14676 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12], v[13], v[14], v[15], v[16]); } 14677 | }; 14678 | 14679 | template <typename T_> 14680 | struct invoke<T_,16> 14681 | { 14682 | static inline T_ execute(ifunction& f, T_ (&v)[16]) 14683 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12], v[13], v[14], v[15]); } 14684 | }; 14685 | 14686 | template <typename T_> 14687 | struct invoke<T_,15> 14688 | { 14689 | static inline T_ execute(ifunction& f, T_ (&v)[15]) 14690 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12], v[13], v[14]); } 14691 | }; 14692 | 14693 | template <typename T_> 14694 | struct invoke<T_,14> 14695 | { 14696 | static inline T_ execute(ifunction& f, T_ (&v)[14]) 14697 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12], v[13]); } 14698 | }; 14699 | 14700 | template <typename T_> 14701 | struct invoke<T_,13> 14702 | { 14703 | static inline T_ execute(ifunction& f, T_ (&v)[13]) 14704 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11], v[12]); } 14705 | }; 14706 | 14707 | template <typename T_> 14708 | struct invoke<T_,12> 14709 | { 14710 | static inline T_ execute(ifunction& f, T_ (&v)[12]) 14711 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10], v[11]); } 14712 | }; 14713 | 14714 | template <typename T_> 14715 | struct invoke<T_,11> 14716 | { 14717 | static inline T_ execute(ifunction& f, T_ (&v)[11]) 14718 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9], v[10]); } 14719 | }; 14720 | 14721 | template <typename T_> 14722 | struct invoke<T_,10> 14723 | { 14724 | static inline T_ execute(ifunction& f, T_ (&v)[10]) 14725 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8], v[9]); } 14726 | }; 14727 | 14728 | template <typename T_> 14729 | struct invoke<T_,9> 14730 | { 14731 | static inline T_ execute(ifunction& f, T_ (&v)[9]) 14732 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7], v[8]); } 14733 | }; 14734 | 14735 | template <typename T_> 14736 | struct invoke<T_,8> 14737 | { 14738 | static inline T_ execute(ifunction& f, T_ (&v)[8]) 14739 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6], v[7]); } 14740 | }; 14741 | 14742 | template <typename T_> 14743 | struct invoke<T_,7> 14744 | { 14745 | static inline T_ execute(ifunction& f, T_ (&v)[7]) 14746 | { return f(v[0], v[1], v[2], v[3], v[4], v[5], v[6]); } 14747 | }; 14748 | 14749 | template <typename T_> 14750 | struct invoke<T_,6> 14751 | { 14752 | static inline T_ execute(ifunction& f, T_ (&v)[6]) 14753 | { return f(v[0], v[1], v[2], v[3], v[4], v[5]); } 14754 | }; 14755 | 14756 | template <typename T_> 14757 | struct invoke<T_,5> 14758 | { 14759 | static inline T_ execute(ifunction& f, T_ (&v)[5]) 14760 | { return f(v[0], v[1], v[2], v[3], v[4]); } 14761 | }; 14762 | 14763 | template <typename T_> 14764 | struct invoke<T_,4> 14765 | { 14766 | static inline T_ execute(ifunction& f, T_ (&v)[4]) 14767 | { return f(v[0], v[1], v[2], v[3]); } 14768 | }; 14769 | 14770 | template <typename T_> 14771 | struct invoke<T_,3> 14772 | { 14773 | static inline T_ execute(ifunction& f, T_ (&v)[3]) 14774 | { return f(v[0], v[1], v[2]); } 14775 | }; 14776 | 14777 | template <typename T_> 14778 | struct invoke<T_,2> 14779 | { 14780 | static inline T_ execute(ifunction& f, T_ (&v)[2]) 14781 | { return f(v[0], v[1]); } 14782 | }; 14783 | 14784 | template <typename T_> 14785 | struct invoke<T_,1> 14786 | { 14787 | static inline T_ execute(ifunction& f, T_ (&v)[1]) 14788 | { return f(v[0]); } 14789 | }; 14790 | 14791 | private: 14792 | 14793 | ifunction* function_; 14794 | std::size_t parameter_count_; 14795 | branch_t branch_[N]; 14796 | bool initialised_; 14797 | }; 14798 | 14799 | template <typename T, typename IFunction> 14800 | class function_N_node<T,IFunction,0> exprtk_final : public expression_node<T> 14801 | { 14802 | public: 14803 | 14804 | typedef expression_node<T>* expression_ptr; 14805 | typedef IFunction ifunction; 14806 | 14807 | explicit function_N_node(ifunction* func) 14808 | : function_((0 == func->param_count) ? func : reinterpret_cast<ifunction*>(0)) 14809 | { 14810 | assert(valid()); 14811 | } 14812 | 14813 | inline bool operator <(const function_N_node<T,IFunction,0>& fn) const 14814 | { 14815 | return this < (&fn); 14816 | } 14817 | 14818 | inline T value() const exprtk_override 14819 | { 14820 | return (*function_)(); 14821 | } 14822 | 14823 | inline typename expression_node<T>::node_type type() const exprtk_override 14824 | { 14825 | return expression_node<T>::e_function; 14826 | } 14827 | 14828 | inline bool valid() const exprtk_override 14829 | { 14830 | return function_; 14831 | } 14832 | 14833 | private: 14834 | 14835 | ifunction* function_; 14836 | }; 14837 | 14838 | template <typename T, typename VarArgFunction> 14839 | class vararg_function_node exprtk_final : public expression_node<T> 14840 | { 14841 | public: 14842 | 14843 | typedef expression_node<T>* expression_ptr; 14844 | 14845 | vararg_function_node(VarArgFunction* func, 14846 | const std::vector<expression_ptr>& arg_list) 14847 | : function_(func) 14848 | , arg_list_(arg_list) 14849 | { 14850 | value_list_.resize(arg_list.size(),std::numeric_limits<T>::quiet_NaN()); 14851 | assert(valid()); 14852 | } 14853 | 14854 | inline bool operator <(const vararg_function_node<T,VarArgFunction>& fn) const 14855 | { 14856 | return this < (&fn); 14857 | } 14858 | 14859 | inline T value() const exprtk_override 14860 | { 14861 | populate_value_list(); 14862 | return (*function_)(value_list_); 14863 | } 14864 | 14865 | inline typename expression_node<T>::node_type type() const exprtk_override 14866 | { 14867 | return expression_node<T>::e_vafunction; 14868 | } 14869 | 14870 | inline bool valid() const exprtk_override 14871 | { 14872 | return function_; 14873 | } 14874 | 14875 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 14876 | { 14877 | for (std::size_t i = 0; i < arg_list_.size(); ++i) 14878 | { 14879 | if (arg_list_[i] && !details::is_variable_node(arg_list_[i])) 14880 | { 14881 | node_delete_list.push_back(&arg_list_[i]); 14882 | } 14883 | } 14884 | } 14885 | 14886 | std::size_t node_depth() const exprtk_override 14887 | { 14888 | return expression_node<T>::ndb_t::compute_node_depth(arg_list_); 14889 | } 14890 | 14891 | private: 14892 | 14893 | inline void populate_value_list() const 14894 | { 14895 | for (std::size_t i = 0; i < arg_list_.size(); ++i) 14896 | { 14897 | value_list_[i] = arg_list_[i]->value(); 14898 | } 14899 | } 14900 | 14901 | VarArgFunction* function_; 14902 | std::vector<expression_ptr> arg_list_; 14903 | mutable std::vector<T> value_list_; 14904 | }; 14905 | 14906 | template <typename T, typename GenericFunction> 14907 | class generic_function_node : public expression_node<T> 14908 | { 14909 | public: 14910 | 14911 | typedef type_store<T> type_store_t; 14912 | typedef expression_node<T>* expression_ptr; 14913 | typedef variable_node<T> variable_node_t; 14914 | typedef vector_node<T> vector_node_t; 14915 | typedef variable_node_t* variable_node_ptr_t; 14916 | typedef vector_node_t* vector_node_ptr_t; 14917 | typedef range_interface<T> range_interface_t; 14918 | typedef range_data_type<T> range_data_type_t; 14919 | typedef typename range_interface<T>::range_t range_t; 14920 | 14921 | typedef std::pair<expression_ptr,bool> branch_t; 14922 | typedef vector_holder<T>* vh_t; 14923 | typedef vector_view<T>* vecview_t; 14924 | 14925 | typedef std::vector<T> tmp_vs_t; 14926 | typedef std::vector<type_store_t> typestore_list_t; 14927 | typedef std::vector<range_data_type_t> range_list_t; 14928 | 14929 | explicit generic_function_node(const std::vector<expression_ptr>& arg_list, 14930 | GenericFunction* func = reinterpret_cast<GenericFunction*>(0)) 14931 | : function_(func) 14932 | , arg_list_(arg_list) 14933 | {} 14934 | 14935 | virtual ~generic_function_node() 14936 | { 14937 | for (std::size_t i = 0; i < vv_list_.size(); ++i) 14938 | { 14939 | vecview_t& vv = vv_list_[i]; 14940 | if (vv && typestore_list_[i].vec_data) 14941 | { 14942 | vv->remove_ref(&typestore_list_[i].vec_data); 14943 | typestore_list_[i].vec_data = 0; 14944 | } 14945 | } 14946 | } 14947 | 14948 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 14949 | { 14950 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 14951 | } 14952 | 14953 | std::size_t node_depth() const exprtk_override exprtk_final 14954 | { 14955 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 14956 | } 14957 | 14958 | virtual bool init_branches() 14959 | { 14960 | expr_as_vec1_store_.resize(arg_list_.size(), T(0) ); 14961 | typestore_list_ .resize(arg_list_.size(), type_store_t() ); 14962 | range_list_ .resize(arg_list_.size(), range_data_type_t()); 14963 | branch_ .resize(arg_list_.size(), branch_t(reinterpret_cast<expression_ptr>(0),false)); 14964 | vv_list_ .resize(arg_list_.size(), vecview_t(0)); 14965 | 14966 | for (std::size_t i = 0; i < arg_list_.size(); ++i) 14967 | { 14968 | type_store_t& ts = typestore_list_[i]; 14969 | 14970 | if (0 == arg_list_[i]) 14971 | return false; 14972 | else if (is_ivector_node(arg_list_[i])) 14973 | { 14974 | vector_interface<T>* vi = reinterpret_cast<vector_interface<T>*>(0); 14975 | 14976 | if (0 == (vi = dynamic_cast<vector_interface<T>*>(arg_list_[i]))) 14977 | return false; 14978 | 14979 | ts.size = vi->size(); 14980 | ts.data = vi->vds().data(); 14981 | ts.type = type_store_t::e_vector; 14982 | 14983 | if ( 14984 | vi->vec()->vec_holder().rebaseable() && 14985 | vi->vec()->vec_holder().rebaseable_instance() 14986 | ) 14987 | { 14988 | vv_list_[i] = vi->vec()->vec_holder().rebaseable_instance(); 14989 | vv_list_[i]->set_ref(&ts.vec_data); 14990 | } 14991 | } 14992 | #ifndef exprtk_disable_string_capabilities 14993 | else if (is_generally_string_node(arg_list_[i])) 14994 | { 14995 | string_base_node<T>* sbn = reinterpret_cast<string_base_node<T>*>(0); 14996 | 14997 | if (0 == (sbn = dynamic_cast<string_base_node<T>*>(arg_list_[i]))) 14998 | return false; 14999 | 15000 | ts.size = sbn->size(); 15001 | ts.data = reinterpret_cast<void*>(const_cast<char_ptr>(sbn->base())); 15002 | ts.type = type_store_t::e_string; 15003 | 15004 | range_list_[i].data = ts.data; 15005 | range_list_[i].size = ts.size; 15006 | range_list_[i].type_size = sizeof(char); 15007 | range_list_[i].str_node = sbn; 15008 | 15009 | range_interface_t* ri = reinterpret_cast<range_interface_t*>(0); 15010 | 15011 | if (0 == (ri = dynamic_cast<range_interface_t*>(arg_list_[i]))) 15012 | return false; 15013 | 15014 | const range_t& rp = ri->range_ref(); 15015 | 15016 | if ( 15017 | rp.const_range() && 15018 | is_const_string_range_node(arg_list_[i]) 15019 | ) 15020 | { 15021 | ts.size = rp.const_size(); 15022 | ts.data = static_cast<char_ptr>(ts.data) + rp.n0_c.second; 15023 | range_list_[i].range = reinterpret_cast<range_t*>(0); 15024 | } 15025 | else 15026 | { 15027 | range_list_[i].range = &(ri->range_ref()); 15028 | range_param_list_.push_back(i); 15029 | } 15030 | } 15031 | #endif 15032 | else if (is_variable_node(arg_list_[i])) 15033 | { 15034 | variable_node_ptr_t var = variable_node_ptr_t(0); 15035 | 15036 | if (0 == (var = dynamic_cast<variable_node_ptr_t>(arg_list_[i]))) 15037 | return false; 15038 | 15039 | ts.size = 1; 15040 | ts.data = &var->ref(); 15041 | ts.type = type_store_t::e_scalar; 15042 | } 15043 | else 15044 | { 15045 | ts.size = 1; 15046 | ts.data = reinterpret_cast<void*>(&expr_as_vec1_store_[i]); 15047 | ts.type = type_store_t::e_scalar; 15048 | } 15049 | 15050 | branch_[i] = std::make_pair(arg_list_[i],branch_deletable(arg_list_[i])); 15051 | } 15052 | 15053 | return true; 15054 | } 15055 | 15056 | inline bool operator <(const generic_function_node<T,GenericFunction>& fn) const 15057 | { 15058 | return this < (&fn); 15059 | } 15060 | 15061 | inline T value() const exprtk_override 15062 | { 15063 | if (populate_value_list()) 15064 | { 15065 | typedef typename GenericFunction::parameter_list_t parameter_list_t; 15066 | 15067 | return (*function_)(parameter_list_t(typestore_list_)); 15068 | } 15069 | 15070 | return std::numeric_limits<T>::quiet_NaN(); 15071 | } 15072 | 15073 | inline typename expression_node<T>::node_type type() const exprtk_override 15074 | { 15075 | return expression_node<T>::e_genfunction; 15076 | } 15077 | 15078 | inline bool valid() const exprtk_override 15079 | { 15080 | return function_; 15081 | } 15082 | 15083 | protected: 15084 | 15085 | inline virtual bool populate_value_list() const 15086 | { 15087 | for (std::size_t i = 0; i < branch_.size(); ++i) 15088 | { 15089 | expr_as_vec1_store_[i] = branch_[i].first->value(); 15090 | } 15091 | 15092 | if (!range_param_list_.empty()) 15093 | { 15094 | assert(range_param_list_.size() <= branch_.size()); 15095 | 15096 | for (std::size_t i = 0; i < range_param_list_.size(); ++i) 15097 | { 15098 | const std::size_t index = range_param_list_[i]; 15099 | range_data_type_t& rdt = range_list_[index]; 15100 | 15101 | const range_t& rp = (*rdt.range); 15102 | std::size_t r0 = 0; 15103 | std::size_t r1 = 0; 15104 | 15105 | const std::size_t data_size = 15106 | #ifndef exprtk_disable_string_capabilities 15107 | rdt.str_node ? rdt.str_node->size() : rdt.size; 15108 | #else 15109 | rdt.size; 15110 | #endif 15111 | 15112 | if (!rp(r0, r1, data_size)) 15113 | { 15114 | return false; 15115 | } 15116 | 15117 | type_store_t& ts = typestore_list_[index]; 15118 | 15119 | ts.size = rp.cache_size(); 15120 | #ifndef exprtk_disable_string_capabilities 15121 | if (ts.type == type_store_t::e_string) 15122 | ts.data = const_cast<char_ptr>(rdt.str_node->base()) + rp.cache.first; 15123 | else 15124 | #endif 15125 | ts.data = static_cast<char_ptr>(rdt.data) + (rp.cache.first * rdt.type_size); 15126 | } 15127 | } 15128 | 15129 | return true; 15130 | } 15131 | 15132 | GenericFunction* function_; 15133 | mutable typestore_list_t typestore_list_; 15134 | 15135 | private: 15136 | 15137 | std::vector<expression_ptr> arg_list_; 15138 | std::vector<branch_t> branch_; 15139 | std::vector<vecview_t> vv_list_; 15140 | mutable tmp_vs_t expr_as_vec1_store_; 15141 | mutable range_list_t range_list_; 15142 | std::vector<std::size_t> range_param_list_; 15143 | }; 15144 | 15145 | #ifndef exprtk_disable_string_capabilities 15146 | template <typename T, typename StringFunction> 15147 | class string_function_node : public generic_function_node<T,StringFunction> 15148 | , public string_base_node<T> 15149 | , public range_interface <T> 15150 | { 15151 | public: 15152 | 15153 | typedef generic_function_node<T,StringFunction> gen_function_t; 15154 | typedef typename range_interface<T>::range_t range_t; 15155 | 15156 | string_function_node(StringFunction* func, 15157 | const std::vector<typename gen_function_t::expression_ptr>& arg_list) 15158 | : gen_function_t(arg_list,func) 15159 | { 15160 | range_.n0_c = std::make_pair<bool,std::size_t>(true,0); 15161 | range_.n1_c = std::make_pair<bool,std::size_t>(true,0); 15162 | range_.cache.first = range_.n0_c.second; 15163 | range_.cache.second = range_.n1_c.second; 15164 | assert(valid()); 15165 | } 15166 | 15167 | inline bool operator <(const string_function_node<T,StringFunction>& fn) const 15168 | { 15169 | return this < (&fn); 15170 | } 15171 | 15172 | inline T value() const exprtk_override 15173 | { 15174 | if (gen_function_t::populate_value_list()) 15175 | { 15176 | typedef typename StringFunction::parameter_list_t parameter_list_t; 15177 | 15178 | const T result = 15179 | (*gen_function_t::function_) 15180 | ( 15181 | ret_string_, 15182 | parameter_list_t(gen_function_t::typestore_list_) 15183 | ); 15184 | 15185 | range_.n1_c.second = ret_string_.size(); 15186 | range_.cache.second = range_.n1_c.second; 15187 | 15188 | return result; 15189 | } 15190 | 15191 | return std::numeric_limits<T>::quiet_NaN(); 15192 | } 15193 | 15194 | inline typename expression_node<T>::node_type type() const exprtk_override 15195 | { 15196 | return expression_node<T>::e_strfunction; 15197 | } 15198 | 15199 | inline bool valid() const exprtk_override 15200 | { 15201 | return gen_function_t::function_; 15202 | } 15203 | 15204 | std::string str() const exprtk_override 15205 | { 15206 | return ret_string_; 15207 | } 15208 | 15209 | char_cptr base() const exprtk_override 15210 | { 15211 | return &ret_string_[0]; 15212 | } 15213 | 15214 | std::size_t size() const exprtk_override 15215 | { 15216 | return ret_string_.size(); 15217 | } 15218 | 15219 | range_t& range_ref() exprtk_override 15220 | { 15221 | return range_; 15222 | } 15223 | 15224 | const range_t& range_ref() const exprtk_override 15225 | { 15226 | return range_; 15227 | } 15228 | 15229 | protected: 15230 | 15231 | mutable range_t range_; 15232 | mutable std::string ret_string_; 15233 | }; 15234 | #endif 15235 | 15236 | template <typename T, typename GenericFunction> 15237 | class multimode_genfunction_node : public generic_function_node<T,GenericFunction> 15238 | { 15239 | public: 15240 | 15241 | typedef generic_function_node<T,GenericFunction> gen_function_t; 15242 | typedef typename gen_function_t::range_t range_t; 15243 | 15244 | multimode_genfunction_node(GenericFunction* func, 15245 | const std::size_t& param_seq_index, 15246 | const std::vector<typename gen_function_t::expression_ptr>& arg_list) 15247 | : gen_function_t(arg_list,func) 15248 | , param_seq_index_(param_seq_index) 15249 | {} 15250 | 15251 | inline T value() const exprtk_override 15252 | { 15253 | assert(gen_function_t::valid()); 15254 | 15255 | if (gen_function_t::populate_value_list()) 15256 | { 15257 | typedef typename GenericFunction::parameter_list_t parameter_list_t; 15258 | 15259 | return 15260 | (*gen_function_t::function_) 15261 | ( 15262 | param_seq_index_, 15263 | parameter_list_t(gen_function_t::typestore_list_) 15264 | ); 15265 | } 15266 | 15267 | return std::numeric_limits<T>::quiet_NaN(); 15268 | } 15269 | 15270 | inline typename expression_node<T>::node_type type() const exprtk_override exprtk_final 15271 | { 15272 | return expression_node<T>::e_genfunction; 15273 | } 15274 | 15275 | private: 15276 | 15277 | std::size_t param_seq_index_; 15278 | }; 15279 | 15280 | #ifndef exprtk_disable_string_capabilities 15281 | template <typename T, typename StringFunction> 15282 | class multimode_strfunction_node exprtk_final : public string_function_node<T,StringFunction> 15283 | { 15284 | public: 15285 | 15286 | typedef string_function_node<T,StringFunction> str_function_t; 15287 | typedef typename str_function_t::range_t range_t; 15288 | 15289 | multimode_strfunction_node(StringFunction* func, 15290 | const std::size_t& param_seq_index, 15291 | const std::vector<typename str_function_t::expression_ptr>& arg_list) 15292 | : str_function_t(func,arg_list) 15293 | , param_seq_index_(param_seq_index) 15294 | {} 15295 | 15296 | inline T value() const exprtk_override 15297 | { 15298 | if (str_function_t::populate_value_list()) 15299 | { 15300 | typedef typename StringFunction::parameter_list_t parameter_list_t; 15301 | 15302 | const T result = 15303 | (*str_function_t::function_) 15304 | ( 15305 | param_seq_index_, 15306 | str_function_t::ret_string_, 15307 | parameter_list_t(str_function_t::typestore_list_) 15308 | ); 15309 | 15310 | str_function_t::range_.n1_c.second = str_function_t::ret_string_.size(); 15311 | str_function_t::range_.cache.second = str_function_t::range_.n1_c.second; 15312 | 15313 | return result; 15314 | } 15315 | 15316 | return std::numeric_limits<T>::quiet_NaN(); 15317 | } 15318 | 15319 | inline typename expression_node<T>::node_type type() const exprtk_override 15320 | { 15321 | return expression_node<T>::e_strfunction; 15322 | } 15323 | 15324 | private: 15325 | 15326 | const std::size_t param_seq_index_; 15327 | }; 15328 | #endif 15329 | 15330 | class return_exception {}; 15331 | 15332 | template <typename T> 15333 | class null_igenfunc 15334 | { 15335 | public: 15336 | 15337 | virtual ~null_igenfunc() 15338 | {} 15339 | 15340 | typedef type_store<T> generic_type; 15341 | typedef typename generic_type::parameter_list parameter_list_t; 15342 | 15343 | inline virtual T operator() (parameter_list_t) 15344 | { 15345 | return std::numeric_limits<T>::quiet_NaN(); 15346 | } 15347 | }; 15348 | 15349 | #ifndef exprtk_disable_return_statement 15350 | template <typename T> 15351 | class return_node exprtk_final : public generic_function_node<T,null_igenfunc<T> > 15352 | { 15353 | public: 15354 | 15355 | typedef results_context<T> results_context_t; 15356 | typedef null_igenfunc<T> igeneric_function_t; 15357 | typedef igeneric_function_t* igeneric_function_ptr; 15358 | typedef generic_function_node<T,igeneric_function_t> gen_function_t; 15359 | 15360 | return_node(const std::vector<typename gen_function_t::expression_ptr>& arg_list, 15361 | results_context_t& rc) 15362 | : gen_function_t (arg_list) 15363 | , results_context_(&rc) 15364 | { 15365 | assert(valid()); 15366 | } 15367 | 15368 | inline T value() const exprtk_override 15369 | { 15370 | if (gen_function_t::populate_value_list()) 15371 | { 15372 | typedef typename type_store<T>::parameter_list parameter_list_t; 15373 | 15374 | results_context_-> 15375 | assign(parameter_list_t(gen_function_t::typestore_list_)); 15376 | 15377 | throw return_exception(); 15378 | } 15379 | 15380 | return std::numeric_limits<T>::quiet_NaN(); 15381 | } 15382 | 15383 | inline typename expression_node<T>::node_type type() const exprtk_override 15384 | { 15385 | return expression_node<T>::e_return; 15386 | } 15387 | 15388 | inline bool valid() const exprtk_override 15389 | { 15390 | return results_context_; 15391 | } 15392 | 15393 | private: 15394 | 15395 | results_context_t* results_context_; 15396 | }; 15397 | 15398 | template <typename T> 15399 | class return_envelope_node exprtk_final : public expression_node<T> 15400 | { 15401 | public: 15402 | 15403 | typedef expression_node<T>* expression_ptr; 15404 | typedef results_context<T> results_context_t; 15405 | typedef std::pair<expression_ptr,bool> branch_t; 15406 | 15407 | return_envelope_node(expression_ptr body, results_context_t& rc) 15408 | : results_context_(&rc ) 15409 | , return_invoked_ (false) 15410 | { 15411 | construct_branch_pair(body_, body); 15412 | assert(valid()); 15413 | } 15414 | 15415 | inline T value() const exprtk_override 15416 | { 15417 | try 15418 | { 15419 | return_invoked_ = false; 15420 | results_context_->clear(); 15421 | 15422 | return body_.first->value(); 15423 | } 15424 | catch(const return_exception&) 15425 | { 15426 | return_invoked_ = true; 15427 | 15428 | return std::numeric_limits<T>::quiet_NaN(); 15429 | } 15430 | } 15431 | 15432 | inline typename expression_node<T>::node_type type() const exprtk_override 15433 | { 15434 | return expression_node<T>::e_retenv; 15435 | } 15436 | 15437 | inline bool valid() const exprtk_override 15438 | { 15439 | return results_context_ && body_.first; 15440 | } 15441 | 15442 | inline bool* retinvk_ptr() 15443 | { 15444 | return &return_invoked_; 15445 | } 15446 | 15447 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 15448 | { 15449 | expression_node<T>::ndb_t::collect(body_, node_delete_list); 15450 | } 15451 | 15452 | std::size_t node_depth() const exprtk_override 15453 | { 15454 | return expression_node<T>::ndb_t::compute_node_depth(body_); 15455 | } 15456 | 15457 | private: 15458 | 15459 | results_context_t* results_context_; 15460 | mutable bool return_invoked_; 15461 | branch_t body_; 15462 | }; 15463 | #endif 15464 | 15465 | #define exprtk_define_unary_op(OpName) \ 15466 | template <typename T> \ 15467 | struct OpName##_op \ 15468 | { \ 15469 | typedef typename functor_t<T>::Type Type; \ 15470 | typedef typename expression_node<T>::node_type node_t; \ 15471 | \ 15472 | static inline T process(Type v) \ 15473 | { \ 15474 | return numeric:: OpName (v); \ 15475 | } \ 15476 | \ 15477 | static inline node_t type() \ 15478 | { \ 15479 | return expression_node<T>::e_##OpName; \ 15480 | } \ 15481 | \ 15482 | static inline details::operator_type operation() \ 15483 | { \ 15484 | return details::e_##OpName; \ 15485 | } \ 15486 | }; \ 15487 | 15488 | exprtk_define_unary_op(abs ) 15489 | exprtk_define_unary_op(acos ) 15490 | exprtk_define_unary_op(acosh) 15491 | exprtk_define_unary_op(asin ) 15492 | exprtk_define_unary_op(asinh) 15493 | exprtk_define_unary_op(atan ) 15494 | exprtk_define_unary_op(atanh) 15495 | exprtk_define_unary_op(ceil ) 15496 | exprtk_define_unary_op(cos ) 15497 | exprtk_define_unary_op(cosh ) 15498 | exprtk_define_unary_op(cot ) 15499 | exprtk_define_unary_op(csc ) 15500 | exprtk_define_unary_op(d2g ) 15501 | exprtk_define_unary_op(d2r ) 15502 | exprtk_define_unary_op(erf ) 15503 | exprtk_define_unary_op(erfc ) 15504 | exprtk_define_unary_op(exp ) 15505 | exprtk_define_unary_op(expm1) 15506 | exprtk_define_unary_op(floor) 15507 | exprtk_define_unary_op(frac ) 15508 | exprtk_define_unary_op(g2d ) 15509 | exprtk_define_unary_op(log ) 15510 | exprtk_define_unary_op(log10) 15511 | exprtk_define_unary_op(log2 ) 15512 | exprtk_define_unary_op(log1p) 15513 | exprtk_define_unary_op(ncdf ) 15514 | exprtk_define_unary_op(neg ) 15515 | exprtk_define_unary_op(notl ) 15516 | exprtk_define_unary_op(pos ) 15517 | exprtk_define_unary_op(r2d ) 15518 | exprtk_define_unary_op(round) 15519 | exprtk_define_unary_op(sec ) 15520 | exprtk_define_unary_op(sgn ) 15521 | exprtk_define_unary_op(sin ) 15522 | exprtk_define_unary_op(sinc ) 15523 | exprtk_define_unary_op(sinh ) 15524 | exprtk_define_unary_op(sqrt ) 15525 | exprtk_define_unary_op(tan ) 15526 | exprtk_define_unary_op(tanh ) 15527 | exprtk_define_unary_op(trunc) 15528 | #undef exprtk_define_unary_op 15529 | 15530 | template <typename T> 15531 | struct opr_base 15532 | { 15533 | typedef typename details::functor_t<T>::Type Type; 15534 | typedef typename details::functor_t<T>::RefType RefType; 15535 | typedef typename details::functor_t<T> functor_t; 15536 | typedef typename functor_t::qfunc_t quaternary_functor_t; 15537 | typedef typename functor_t::tfunc_t trinary_functor_t; 15538 | typedef typename functor_t::bfunc_t binary_functor_t; 15539 | typedef typename functor_t::ufunc_t unary_functor_t; 15540 | }; 15541 | 15542 | template <typename T> 15543 | struct add_op : public opr_base<T> 15544 | { 15545 | typedef typename opr_base<T>::Type Type; 15546 | typedef typename opr_base<T>::RefType RefType; 15547 | 15548 | static inline T process(Type t1, Type t2) { return t1 + t2; } 15549 | static inline T process(Type t1, Type t2, Type t3) { return t1 + t2 + t3; } 15550 | static inline void assign(RefType t1, Type t2) { t1 += t2; } 15551 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_add; } 15552 | static inline details::operator_type operation() { return details::e_add; } 15553 | }; 15554 | 15555 | template <typename T> 15556 | struct mul_op : public opr_base<T> 15557 | { 15558 | typedef typename opr_base<T>::Type Type; 15559 | typedef typename opr_base<T>::RefType RefType; 15560 | 15561 | static inline T process(Type t1, Type t2) { return t1 * t2; } 15562 | static inline T process(Type t1, Type t2, Type t3) { return t1 * t2 * t3; } 15563 | static inline void assign(RefType t1, Type t2) { t1 *= t2; } 15564 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_mul; } 15565 | static inline details::operator_type operation() { return details::e_mul; } 15566 | }; 15567 | 15568 | template <typename T> 15569 | struct sub_op : public opr_base<T> 15570 | { 15571 | typedef typename opr_base<T>::Type Type; 15572 | typedef typename opr_base<T>::RefType RefType; 15573 | 15574 | static inline T process(Type t1, Type t2) { return t1 - t2; } 15575 | static inline T process(Type t1, Type t2, Type t3) { return t1 - t2 - t3; } 15576 | static inline void assign(RefType t1, Type t2) { t1 -= t2; } 15577 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_sub; } 15578 | static inline details::operator_type operation() { return details::e_sub; } 15579 | }; 15580 | 15581 | template <typename T> 15582 | struct div_op : public opr_base<T> 15583 | { 15584 | typedef typename opr_base<T>::Type Type; 15585 | typedef typename opr_base<T>::RefType RefType; 15586 | 15587 | static inline T process(Type t1, Type t2) { return t1 / t2; } 15588 | static inline T process(Type t1, Type t2, Type t3) { return t1 / t2 / t3; } 15589 | static inline void assign(RefType t1, Type t2) { t1 /= t2; } 15590 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_div; } 15591 | static inline details::operator_type operation() { return details::e_div; } 15592 | }; 15593 | 15594 | template <typename T> 15595 | struct mod_op : public opr_base<T> 15596 | { 15597 | typedef typename opr_base<T>::Type Type; 15598 | typedef typename opr_base<T>::RefType RefType; 15599 | 15600 | static inline T process(Type t1, Type t2) { return numeric::modulus<T>(t1,t2); } 15601 | static inline void assign(RefType t1, Type t2) { t1 = numeric::modulus<T>(t1,t2); } 15602 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_mod; } 15603 | static inline details::operator_type operation() { return details::e_mod; } 15604 | }; 15605 | 15606 | template <typename T> 15607 | struct pow_op : public opr_base<T> 15608 | { 15609 | typedef typename opr_base<T>::Type Type; 15610 | typedef typename opr_base<T>::RefType RefType; 15611 | 15612 | static inline T process(Type t1, Type t2) { return numeric::pow<T>(t1,t2); } 15613 | static inline void assign(RefType t1, Type t2) { t1 = numeric::pow<T>(t1,t2); } 15614 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_pow; } 15615 | static inline details::operator_type operation() { return details::e_pow; } 15616 | }; 15617 | 15618 | template <typename T> 15619 | struct lt_op : public opr_base<T> 15620 | { 15621 | typedef typename opr_base<T>::Type Type; 15622 | 15623 | static inline T process(Type t1, Type t2) { return ((t1 < t2) ? T(1) : T(0)); } 15624 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 < t2) ? T(1) : T(0)); } 15625 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_lt; } 15626 | static inline details::operator_type operation() { return details::e_lt; } 15627 | }; 15628 | 15629 | template <typename T> 15630 | struct lte_op : public opr_base<T> 15631 | { 15632 | typedef typename opr_base<T>::Type Type; 15633 | 15634 | static inline T process(Type t1, Type t2) { return ((t1 <= t2) ? T(1) : T(0)); } 15635 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 <= t2) ? T(1) : T(0)); } 15636 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_lte; } 15637 | static inline details::operator_type operation() { return details::e_lte; } 15638 | }; 15639 | 15640 | template <typename T> 15641 | struct gt_op : public opr_base<T> 15642 | { 15643 | typedef typename opr_base<T>::Type Type; 15644 | 15645 | static inline T process(Type t1, Type t2) { return ((t1 > t2) ? T(1) : T(0)); } 15646 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 > t2) ? T(1) : T(0)); } 15647 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_gt; } 15648 | static inline details::operator_type operation() { return details::e_gt; } 15649 | }; 15650 | 15651 | template <typename T> 15652 | struct gte_op : public opr_base<T> 15653 | { 15654 | typedef typename opr_base<T>::Type Type; 15655 | 15656 | static inline T process(Type t1, Type t2) { return ((t1 >= t2) ? T(1) : T(0)); } 15657 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 >= t2) ? T(1) : T(0)); } 15658 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_gte; } 15659 | static inline details::operator_type operation() { return details::e_gte; } 15660 | }; 15661 | 15662 | template <typename T> 15663 | struct eq_op : public opr_base<T> 15664 | { 15665 | typedef typename opr_base<T>::Type Type; 15666 | static inline T process(Type t1, Type t2) { return (std::equal_to<T>()(t1,t2) ? T(1) : T(0)); } 15667 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 == t2) ? T(1) : T(0)); } 15668 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_eq; } 15669 | static inline details::operator_type operation() { return details::e_eq; } 15670 | }; 15671 | 15672 | template <typename T> 15673 | struct equal_op : public opr_base<T> 15674 | { 15675 | typedef typename opr_base<T>::Type Type; 15676 | 15677 | static inline T process(Type t1, Type t2) { return numeric::equal(t1,t2); } 15678 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 == t2) ? T(1) : T(0)); } 15679 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_eq; } 15680 | static inline details::operator_type operation() { return details::e_equal; } 15681 | }; 15682 | 15683 | template <typename T> 15684 | struct ne_op : public opr_base<T> 15685 | { 15686 | typedef typename opr_base<T>::Type Type; 15687 | 15688 | static inline T process(Type t1, Type t2) { return (std::not_equal_to<T>()(t1,t2) ? T(1) : T(0)); } 15689 | static inline T process(const std::string& t1, const std::string& t2) { return ((t1 != t2) ? T(1) : T(0)); } 15690 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_ne; } 15691 | static inline details::operator_type operation() { return details::e_ne; } 15692 | }; 15693 | 15694 | template <typename T> 15695 | struct and_op : public opr_base<T> 15696 | { 15697 | typedef typename opr_base<T>::Type Type; 15698 | 15699 | static inline T process(Type t1, Type t2) { return (details::is_true(t1) && details::is_true(t2)) ? T(1) : T(0); } 15700 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_and; } 15701 | static inline details::operator_type operation() { return details::e_and; } 15702 | }; 15703 | 15704 | template <typename T> 15705 | struct nand_op : public opr_base<T> 15706 | { 15707 | typedef typename opr_base<T>::Type Type; 15708 | 15709 | static inline T process(Type t1, Type t2) { return (details::is_true(t1) && details::is_true(t2)) ? T(0) : T(1); } 15710 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_nand; } 15711 | static inline details::operator_type operation() { return details::e_nand; } 15712 | }; 15713 | 15714 | template <typename T> 15715 | struct or_op : public opr_base<T> 15716 | { 15717 | typedef typename opr_base<T>::Type Type; 15718 | 15719 | static inline T process(Type t1, Type t2) { return (details::is_true(t1) || details::is_true(t2)) ? T(1) : T(0); } 15720 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_or; } 15721 | static inline details::operator_type operation() { return details::e_or; } 15722 | }; 15723 | 15724 | template <typename T> 15725 | struct nor_op : public opr_base<T> 15726 | { 15727 | typedef typename opr_base<T>::Type Type; 15728 | 15729 | static inline T process(Type t1, Type t2) { return (details::is_true(t1) || details::is_true(t2)) ? T(0) : T(1); } 15730 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_nor; } 15731 | static inline details::operator_type operation() { return details::e_nor; } 15732 | }; 15733 | 15734 | template <typename T> 15735 | struct xor_op : public opr_base<T> 15736 | { 15737 | typedef typename opr_base<T>::Type Type; 15738 | 15739 | static inline T process(Type t1, Type t2) { return numeric::xor_opr<T>(t1,t2); } 15740 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_nor; } 15741 | static inline details::operator_type operation() { return details::e_xor; } 15742 | }; 15743 | 15744 | template <typename T> 15745 | struct xnor_op : public opr_base<T> 15746 | { 15747 | typedef typename opr_base<T>::Type Type; 15748 | 15749 | static inline T process(Type t1, Type t2) { return numeric::xnor_opr<T>(t1,t2); } 15750 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_nor; } 15751 | static inline details::operator_type operation() { return details::e_xnor; } 15752 | }; 15753 | 15754 | template <typename T> 15755 | struct in_op : public opr_base<T> 15756 | { 15757 | typedef typename opr_base<T>::Type Type; 15758 | 15759 | static inline T process(const T&, const T&) { return std::numeric_limits<T>::quiet_NaN(); } 15760 | static inline T process(const std::string& t1, const std::string& t2) { return ((std::string::npos != t2.find(t1)) ? T(1) : T(0)); } 15761 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_in; } 15762 | static inline details::operator_type operation() { return details::e_in; } 15763 | }; 15764 | 15765 | template <typename T> 15766 | struct like_op : public opr_base<T> 15767 | { 15768 | typedef typename opr_base<T>::Type Type; 15769 | 15770 | static inline T process(const T&, const T&) { return std::numeric_limits<T>::quiet_NaN(); } 15771 | static inline T process(const std::string& t1, const std::string& t2) { return (details::wc_match(t2,t1) ? T(1) : T(0)); } 15772 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_like; } 15773 | static inline details::operator_type operation() { return details::e_like; } 15774 | }; 15775 | 15776 | template <typename T> 15777 | struct ilike_op : public opr_base<T> 15778 | { 15779 | typedef typename opr_base<T>::Type Type; 15780 | 15781 | static inline T process(const T&, const T&) { return std::numeric_limits<T>::quiet_NaN(); } 15782 | static inline T process(const std::string& t1, const std::string& t2) { return (details::wc_imatch(t2,t1) ? T(1) : T(0)); } 15783 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_ilike; } 15784 | static inline details::operator_type operation() { return details::e_ilike; } 15785 | }; 15786 | 15787 | template <typename T> 15788 | struct inrange_op : public opr_base<T> 15789 | { 15790 | typedef typename opr_base<T>::Type Type; 15791 | 15792 | static inline T process(const T& t0, const T& t1, const T& t2) { return ((t0 <= t1) && (t1 <= t2)) ? T(1) : T(0); } 15793 | static inline T process(const std::string& t0, const std::string& t1, const std::string& t2) 15794 | { 15795 | return ((t0 <= t1) && (t1 <= t2)) ? T(1) : T(0); 15796 | } 15797 | static inline typename expression_node<T>::node_type type() { return expression_node<T>::e_inranges; } 15798 | static inline details::operator_type operation() { return details::e_inrange; } 15799 | }; 15800 | 15801 | template <typename T> 15802 | inline T value(details::expression_node<T>* n) 15803 | { 15804 | return n->value(); 15805 | } 15806 | 15807 | template <typename T> 15808 | inline T value(std::pair<details::expression_node<T>*,bool> n) 15809 | { 15810 | return n.first->value(); 15811 | } 15812 | 15813 | template <typename T> 15814 | inline T value(const T* t) 15815 | { 15816 | return (*t); 15817 | } 15818 | 15819 | template <typename T> 15820 | inline T value(const T& t) 15821 | { 15822 | return t; 15823 | } 15824 | 15825 | template <typename T> 15826 | struct vararg_add_op exprtk_final : public opr_base<T> 15827 | { 15828 | typedef typename opr_base<T>::Type Type; 15829 | 15830 | template <typename Type, 15831 | typename Allocator, 15832 | template <typename, typename> class Sequence> 15833 | static inline T process(const Sequence<Type,Allocator>& arg_list) 15834 | { 15835 | switch (arg_list.size()) 15836 | { 15837 | case 0 : return T(0); 15838 | case 1 : return process_1(arg_list); 15839 | case 2 : return process_2(arg_list); 15840 | case 3 : return process_3(arg_list); 15841 | case 4 : return process_4(arg_list); 15842 | case 5 : return process_5(arg_list); 15843 | default : 15844 | { 15845 | T result = T(0); 15846 | 15847 | for (std::size_t i = 0; i < arg_list.size(); ++i) 15848 | { 15849 | result += value(arg_list[i]); 15850 | } 15851 | 15852 | return result; 15853 | } 15854 | } 15855 | } 15856 | 15857 | template <typename Sequence> 15858 | static inline T process_1(const Sequence& arg_list) 15859 | { 15860 | return value(arg_list[0]); 15861 | } 15862 | 15863 | template <typename Sequence> 15864 | static inline T process_2(const Sequence& arg_list) 15865 | { 15866 | return value(arg_list[0]) + value(arg_list[1]); 15867 | } 15868 | 15869 | template <typename Sequence> 15870 | static inline T process_3(const Sequence& arg_list) 15871 | { 15872 | return value(arg_list[0]) + value(arg_list[1]) + 15873 | value(arg_list[2]) ; 15874 | } 15875 | 15876 | template <typename Sequence> 15877 | static inline T process_4(const Sequence& arg_list) 15878 | { 15879 | return value(arg_list[0]) + value(arg_list[1]) + 15880 | value(arg_list[2]) + value(arg_list[3]) ; 15881 | } 15882 | 15883 | template <typename Sequence> 15884 | static inline T process_5(const Sequence& arg_list) 15885 | { 15886 | return value(arg_list[0]) + value(arg_list[1]) + 15887 | value(arg_list[2]) + value(arg_list[3]) + 15888 | value(arg_list[4]) ; 15889 | } 15890 | }; 15891 | 15892 | template <typename T> 15893 | struct vararg_mul_op exprtk_final : public opr_base<T> 15894 | { 15895 | typedef typename opr_base<T>::Type Type; 15896 | 15897 | template <typename Type, 15898 | typename Allocator, 15899 | template <typename, typename> class Sequence> 15900 | static inline T process(const Sequence<Type,Allocator>& arg_list) 15901 | { 15902 | switch (arg_list.size()) 15903 | { 15904 | case 0 : return T(0); 15905 | case 1 : return process_1(arg_list); 15906 | case 2 : return process_2(arg_list); 15907 | case 3 : return process_3(arg_list); 15908 | case 4 : return process_4(arg_list); 15909 | case 5 : return process_5(arg_list); 15910 | default : 15911 | { 15912 | T result = T(value(arg_list[0])); 15913 | 15914 | for (std::size_t i = 1; i < arg_list.size(); ++i) 15915 | { 15916 | result *= value(arg_list[i]); 15917 | } 15918 | 15919 | return result; 15920 | } 15921 | } 15922 | } 15923 | 15924 | template <typename Sequence> 15925 | static inline T process_1(const Sequence& arg_list) 15926 | { 15927 | return value(arg_list[0]); 15928 | } 15929 | 15930 | template <typename Sequence> 15931 | static inline T process_2(const Sequence& arg_list) 15932 | { 15933 | return value(arg_list[0]) * value(arg_list[1]); 15934 | } 15935 | 15936 | template <typename Sequence> 15937 | static inline T process_3(const Sequence& arg_list) 15938 | { 15939 | return value(arg_list[0]) * value(arg_list[1]) * 15940 | value(arg_list[2]) ; 15941 | } 15942 | 15943 | template <typename Sequence> 15944 | static inline T process_4(const Sequence& arg_list) 15945 | { 15946 | return value(arg_list[0]) * value(arg_list[1]) * 15947 | value(arg_list[2]) * value(arg_list[3]) ; 15948 | } 15949 | 15950 | template <typename Sequence> 15951 | static inline T process_5(const Sequence& arg_list) 15952 | { 15953 | return value(arg_list[0]) * value(arg_list[1]) * 15954 | value(arg_list[2]) * value(arg_list[3]) * 15955 | value(arg_list[4]) ; 15956 | } 15957 | }; 15958 | 15959 | template <typename T> 15960 | struct vararg_avg_op exprtk_final : public opr_base<T> 15961 | { 15962 | typedef typename opr_base<T>::Type Type; 15963 | 15964 | template <typename Type, 15965 | typename Allocator, 15966 | template <typename, typename> class Sequence> 15967 | static inline T process(const Sequence<Type,Allocator>& arg_list) 15968 | { 15969 | switch (arg_list.size()) 15970 | { 15971 | case 0 : return T(0); 15972 | case 1 : return process_1(arg_list); 15973 | case 2 : return process_2(arg_list); 15974 | case 3 : return process_3(arg_list); 15975 | case 4 : return process_4(arg_list); 15976 | case 5 : return process_5(arg_list); 15977 | default : return vararg_add_op<T>::process(arg_list) / T(arg_list.size()); 15978 | } 15979 | } 15980 | 15981 | template <typename Sequence> 15982 | static inline T process_1(const Sequence& arg_list) 15983 | { 15984 | return value(arg_list[0]); 15985 | } 15986 | 15987 | template <typename Sequence> 15988 | static inline T process_2(const Sequence& arg_list) 15989 | { 15990 | return (value(arg_list[0]) + value(arg_list[1])) / T(2); 15991 | } 15992 | 15993 | template <typename Sequence> 15994 | static inline T process_3(const Sequence& arg_list) 15995 | { 15996 | return (value(arg_list[0]) + value(arg_list[1]) + value(arg_list[2])) / T(3); 15997 | } 15998 | 15999 | template <typename Sequence> 16000 | static inline T process_4(const Sequence& arg_list) 16001 | { 16002 | return (value(arg_list[0]) + value(arg_list[1]) + 16003 | value(arg_list[2]) + value(arg_list[3])) / T(4); 16004 | } 16005 | 16006 | template <typename Sequence> 16007 | static inline T process_5(const Sequence& arg_list) 16008 | { 16009 | return (value(arg_list[0]) + value(arg_list[1]) + 16010 | value(arg_list[2]) + value(arg_list[3]) + 16011 | value(arg_list[4])) / T(5); 16012 | } 16013 | }; 16014 | 16015 | template <typename T> 16016 | struct vararg_min_op exprtk_final : public opr_base<T> 16017 | { 16018 | typedef typename opr_base<T>::Type Type; 16019 | 16020 | template <typename Type, 16021 | typename Allocator, 16022 | template <typename, typename> class Sequence> 16023 | static inline T process(const Sequence<Type,Allocator>& arg_list) 16024 | { 16025 | switch (arg_list.size()) 16026 | { 16027 | case 0 : return T(0); 16028 | case 1 : return process_1(arg_list); 16029 | case 2 : return process_2(arg_list); 16030 | case 3 : return process_3(arg_list); 16031 | case 4 : return process_4(arg_list); 16032 | case 5 : return process_5(arg_list); 16033 | default : 16034 | { 16035 | T result = T(value(arg_list[0])); 16036 | 16037 | for (std::size_t i = 1; i < arg_list.size(); ++i) 16038 | { 16039 | const T v = value(arg_list[i]); 16040 | 16041 | if (v < result) 16042 | result = v; 16043 | } 16044 | 16045 | return result; 16046 | } 16047 | } 16048 | } 16049 | 16050 | template <typename Sequence> 16051 | static inline T process_1(const Sequence& arg_list) 16052 | { 16053 | return value(arg_list[0]); 16054 | } 16055 | 16056 | template <typename Sequence> 16057 | static inline T process_2(const Sequence& arg_list) 16058 | { 16059 | return std::min<T>(value(arg_list[0]),value(arg_list[1])); 16060 | } 16061 | 16062 | template <typename Sequence> 16063 | static inline T process_3(const Sequence& arg_list) 16064 | { 16065 | return std::min<T>(std::min<T>(value(arg_list[0]),value(arg_list[1])),value(arg_list[2])); 16066 | } 16067 | 16068 | template <typename Sequence> 16069 | static inline T process_4(const Sequence& arg_list) 16070 | { 16071 | return std::min<T>( 16072 | std::min<T>(value(arg_list[0]), value(arg_list[1])), 16073 | std::min<T>(value(arg_list[2]), value(arg_list[3]))); 16074 | } 16075 | 16076 | template <typename Sequence> 16077 | static inline T process_5(const Sequence& arg_list) 16078 | { 16079 | return std::min<T>( 16080 | std::min<T>(std::min<T>(value(arg_list[0]), value(arg_list[1])), 16081 | std::min<T>(value(arg_list[2]), value(arg_list[3]))), 16082 | value(arg_list[4])); 16083 | } 16084 | }; 16085 | 16086 | template <typename T> 16087 | struct vararg_max_op exprtk_final : public opr_base<T> 16088 | { 16089 | typedef typename opr_base<T>::Type Type; 16090 | 16091 | template <typename Type, 16092 | typename Allocator, 16093 | template <typename, typename> class Sequence> 16094 | static inline T process(const Sequence<Type,Allocator>& arg_list) 16095 | { 16096 | switch (arg_list.size()) 16097 | { 16098 | case 0 : return T(0); 16099 | case 1 : return process_1(arg_list); 16100 | case 2 : return process_2(arg_list); 16101 | case 3 : return process_3(arg_list); 16102 | case 4 : return process_4(arg_list); 16103 | case 5 : return process_5(arg_list); 16104 | default : 16105 | { 16106 | T result = T(value(arg_list[0])); 16107 | 16108 | for (std::size_t i = 1; i < arg_list.size(); ++i) 16109 | { 16110 | const T v = value(arg_list[i]); 16111 | 16112 | if (v > result) 16113 | result = v; 16114 | } 16115 | 16116 | return result; 16117 | } 16118 | } 16119 | } 16120 | 16121 | template <typename Sequence> 16122 | static inline T process_1(const Sequence& arg_list) 16123 | { 16124 | return value(arg_list[0]); 16125 | } 16126 | 16127 | template <typename Sequence> 16128 | static inline T process_2(const Sequence& arg_list) 16129 | { 16130 | return std::max<T>(value(arg_list[0]),value(arg_list[1])); 16131 | } 16132 | 16133 | template <typename Sequence> 16134 | static inline T process_3(const Sequence& arg_list) 16135 | { 16136 | return std::max<T>(std::max<T>(value(arg_list[0]),value(arg_list[1])),value(arg_list[2])); 16137 | } 16138 | 16139 | template <typename Sequence> 16140 | static inline T process_4(const Sequence& arg_list) 16141 | { 16142 | return std::max<T>( 16143 | std::max<T>(value(arg_list[0]), value(arg_list[1])), 16144 | std::max<T>(value(arg_list[2]), value(arg_list[3]))); 16145 | } 16146 | 16147 | template <typename Sequence> 16148 | static inline T process_5(const Sequence& arg_list) 16149 | { 16150 | return std::max<T>( 16151 | std::max<T>(std::max<T>(value(arg_list[0]), value(arg_list[1])), 16152 | std::max<T>(value(arg_list[2]), value(arg_list[3]))), 16153 | value(arg_list[4])); 16154 | } 16155 | }; 16156 | 16157 | template <typename T> 16158 | struct vararg_mand_op exprtk_final : public opr_base<T> 16159 | { 16160 | typedef typename opr_base<T>::Type Type; 16161 | 16162 | template <typename Type, 16163 | typename Allocator, 16164 | template <typename, typename> class Sequence> 16165 | static inline T process(const Sequence<Type,Allocator>& arg_list) 16166 | { 16167 | switch (arg_list.size()) 16168 | { 16169 | case 1 : return process_1(arg_list); 16170 | case 2 : return process_2(arg_list); 16171 | case 3 : return process_3(arg_list); 16172 | case 4 : return process_4(arg_list); 16173 | case 5 : return process_5(arg_list); 16174 | default : 16175 | { 16176 | for (std::size_t i = 0; i < arg_list.size(); ++i) 16177 | { 16178 | if (std::equal_to<T>()(T(0), value(arg_list[i]))) 16179 | return T(0); 16180 | } 16181 | 16182 | return T(1); 16183 | } 16184 | } 16185 | } 16186 | 16187 | template <typename Sequence> 16188 | static inline T process_1(const Sequence& arg_list) 16189 | { 16190 | return std::not_equal_to<T>() 16191 | (T(0), value(arg_list[0])) ? T(1) : T(0); 16192 | } 16193 | 16194 | template <typename Sequence> 16195 | static inline T process_2(const Sequence& arg_list) 16196 | { 16197 | return ( 16198 | std::not_equal_to<T>()(T(0), value(arg_list[0])) && 16199 | std::not_equal_to<T>()(T(0), value(arg_list[1])) 16200 | ) ? T(1) : T(0); 16201 | } 16202 | 16203 | template <typename Sequence> 16204 | static inline T process_3(const Sequence& arg_list) 16205 | { 16206 | return ( 16207 | std::not_equal_to<T>()(T(0), value(arg_list[0])) && 16208 | std::not_equal_to<T>()(T(0), value(arg_list[1])) && 16209 | std::not_equal_to<T>()(T(0), value(arg_list[2])) 16210 | ) ? T(1) : T(0); 16211 | } 16212 | 16213 | template <typename Sequence> 16214 | static inline T process_4(const Sequence& arg_list) 16215 | { 16216 | return ( 16217 | std::not_equal_to<T>()(T(0), value(arg_list[0])) && 16218 | std::not_equal_to<T>()(T(0), value(arg_list[1])) && 16219 | std::not_equal_to<T>()(T(0), value(arg_list[2])) && 16220 | std::not_equal_to<T>()(T(0), value(arg_list[3])) 16221 | ) ? T(1) : T(0); 16222 | } 16223 | 16224 | template <typename Sequence> 16225 | static inline T process_5(const Sequence& arg_list) 16226 | { 16227 | return ( 16228 | std::not_equal_to<T>()(T(0), value(arg_list[0])) && 16229 | std::not_equal_to<T>()(T(0), value(arg_list[1])) && 16230 | std::not_equal_to<T>()(T(0), value(arg_list[2])) && 16231 | std::not_equal_to<T>()(T(0), value(arg_list[3])) && 16232 | std::not_equal_to<T>()(T(0), value(arg_list[4])) 16233 | ) ? T(1) : T(0); 16234 | } 16235 | }; 16236 | 16237 | template <typename T> 16238 | struct vararg_mor_op exprtk_final : public opr_base<T> 16239 | { 16240 | typedef typename opr_base<T>::Type Type; 16241 | 16242 | template <typename Type, 16243 | typename Allocator, 16244 | template <typename, typename> class Sequence> 16245 | static inline T process(const Sequence<Type,Allocator>& arg_list) 16246 | { 16247 | switch (arg_list.size()) 16248 | { 16249 | case 1 : return process_1(arg_list); 16250 | case 2 : return process_2(arg_list); 16251 | case 3 : return process_3(arg_list); 16252 | case 4 : return process_4(arg_list); 16253 | case 5 : return process_5(arg_list); 16254 | default : 16255 | { 16256 | for (std::size_t i = 0; i < arg_list.size(); ++i) 16257 | { 16258 | if (std::not_equal_to<T>()(T(0), value(arg_list[i]))) 16259 | return T(1); 16260 | } 16261 | 16262 | return T(0); 16263 | } 16264 | } 16265 | } 16266 | 16267 | template <typename Sequence> 16268 | static inline T process_1(const Sequence& arg_list) 16269 | { 16270 | return std::not_equal_to<T>() 16271 | (T(0), value(arg_list[0])) ? T(1) : T(0); 16272 | } 16273 | 16274 | template <typename Sequence> 16275 | static inline T process_2(const Sequence& arg_list) 16276 | { 16277 | return ( 16278 | std::not_equal_to<T>()(T(0), value(arg_list[0])) || 16279 | std::not_equal_to<T>()(T(0), value(arg_list[1])) 16280 | ) ? T(1) : T(0); 16281 | } 16282 | 16283 | template <typename Sequence> 16284 | static inline T process_3(const Sequence& arg_list) 16285 | { 16286 | return ( 16287 | std::not_equal_to<T>()(T(0), value(arg_list[0])) || 16288 | std::not_equal_to<T>()(T(0), value(arg_list[1])) || 16289 | std::not_equal_to<T>()(T(0), value(arg_list[2])) 16290 | ) ? T(1) : T(0); 16291 | } 16292 | 16293 | template <typename Sequence> 16294 | static inline T process_4(const Sequence& arg_list) 16295 | { 16296 | return ( 16297 | std::not_equal_to<T>()(T(0), value(arg_list[0])) || 16298 | std::not_equal_to<T>()(T(0), value(arg_list[1])) || 16299 | std::not_equal_to<T>()(T(0), value(arg_list[2])) || 16300 | std::not_equal_to<T>()(T(0), value(arg_list[3])) 16301 | ) ? T(1) : T(0); 16302 | } 16303 | 16304 | template <typename Sequence> 16305 | static inline T process_5(const Sequence& arg_list) 16306 | { 16307 | return ( 16308 | std::not_equal_to<T>()(T(0), value(arg_list[0])) || 16309 | std::not_equal_to<T>()(T(0), value(arg_list[1])) || 16310 | std::not_equal_to<T>()(T(0), value(arg_list[2])) || 16311 | std::not_equal_to<T>()(T(0), value(arg_list[3])) || 16312 | std::not_equal_to<T>()(T(0), value(arg_list[4])) 16313 | ) ? T(1) : T(0); 16314 | } 16315 | }; 16316 | 16317 | template <typename T> 16318 | struct vararg_multi_op exprtk_final : public opr_base<T> 16319 | { 16320 | typedef typename opr_base<T>::Type Type; 16321 | 16322 | template <typename Type, 16323 | typename Allocator, 16324 | template <typename, typename> class Sequence> 16325 | static inline T process(const Sequence<Type,Allocator>& arg_list) 16326 | { 16327 | switch (arg_list.size()) 16328 | { 16329 | case 0 : return std::numeric_limits<T>::quiet_NaN(); 16330 | case 1 : return process_1(arg_list); 16331 | case 2 : return process_2(arg_list); 16332 | case 3 : return process_3(arg_list); 16333 | case 4 : return process_4(arg_list); 16334 | case 5 : return process_5(arg_list); 16335 | case 6 : return process_6(arg_list); 16336 | case 7 : return process_7(arg_list); 16337 | case 8 : return process_8(arg_list); 16338 | default : 16339 | { 16340 | for (std::size_t i = 0; i < (arg_list.size() - 1); ++i) 16341 | { 16342 | value(arg_list[i]); 16343 | } 16344 | return value(arg_list.back()); 16345 | } 16346 | } 16347 | } 16348 | 16349 | template <typename Sequence> 16350 | static inline T process_1(const Sequence& arg_list) 16351 | { 16352 | return value(arg_list[0]); 16353 | } 16354 | 16355 | template <typename Sequence> 16356 | static inline T process_2(const Sequence& arg_list) 16357 | { 16358 | value(arg_list[0]); 16359 | return value(arg_list[1]); 16360 | } 16361 | 16362 | template <typename Sequence> 16363 | static inline T process_3(const Sequence& arg_list) 16364 | { 16365 | value(arg_list[0]); 16366 | value(arg_list[1]); 16367 | return value(arg_list[2]); 16368 | } 16369 | 16370 | template <typename Sequence> 16371 | static inline T process_4(const Sequence& arg_list) 16372 | { 16373 | value(arg_list[0]); 16374 | value(arg_list[1]); 16375 | value(arg_list[2]); 16376 | return value(arg_list[3]); 16377 | } 16378 | 16379 | template <typename Sequence> 16380 | static inline T process_5(const Sequence& arg_list) 16381 | { 16382 | value(arg_list[0]); 16383 | value(arg_list[1]); 16384 | value(arg_list[2]); 16385 | value(arg_list[3]); 16386 | return value(arg_list[4]); 16387 | } 16388 | 16389 | template <typename Sequence> 16390 | static inline T process_6(const Sequence& arg_list) 16391 | { 16392 | value(arg_list[0]); 16393 | value(arg_list[1]); 16394 | value(arg_list[2]); 16395 | value(arg_list[3]); 16396 | value(arg_list[4]); 16397 | return value(arg_list[5]); 16398 | } 16399 | 16400 | template <typename Sequence> 16401 | static inline T process_7(const Sequence& arg_list) 16402 | { 16403 | value(arg_list[0]); 16404 | value(arg_list[1]); 16405 | value(arg_list[2]); 16406 | value(arg_list[3]); 16407 | value(arg_list[4]); 16408 | value(arg_list[5]); 16409 | return value(arg_list[6]); 16410 | } 16411 | 16412 | template <typename Sequence> 16413 | static inline T process_8(const Sequence& arg_list) 16414 | { 16415 | value(arg_list[0]); 16416 | value(arg_list[1]); 16417 | value(arg_list[2]); 16418 | value(arg_list[3]); 16419 | value(arg_list[4]); 16420 | value(arg_list[5]); 16421 | value(arg_list[6]); 16422 | return value(arg_list[7]); 16423 | } 16424 | }; 16425 | 16426 | template <typename T> 16427 | struct vec_add_op 16428 | { 16429 | typedef vector_interface<T>* ivector_ptr; 16430 | 16431 | static inline T process(const ivector_ptr v) 16432 | { 16433 | const T* vec = v->vec()->vds().data(); 16434 | const std::size_t vec_size = v->size(); 16435 | 16436 | loop_unroll::details lud(vec_size); 16437 | 16438 | if (vec_size <= static_cast<std::size_t>(lud.batch_size)) 16439 | { 16440 | T result = T(0); 16441 | int i = 0; 16442 | 16443 | switch (vec_size) 16444 | { 16445 | #define case_stmt(N,fall_through) \ 16446 | case N : result += vec[i++]; \ 16447 | fall_through \ 16448 | 16449 | #ifndef exprtk_disable_superscalar_unroll 16450 | case_stmt(16, exprtk_fallthrough) case_stmt(15, exprtk_fallthrough) 16451 | case_stmt(14, exprtk_fallthrough) case_stmt(13, exprtk_fallthrough) 16452 | case_stmt(12, exprtk_fallthrough) case_stmt(11, exprtk_fallthrough) 16453 | case_stmt(10, exprtk_fallthrough) case_stmt( 9, exprtk_fallthrough) 16454 | case_stmt( 8, exprtk_fallthrough) case_stmt( 7, exprtk_fallthrough) 16455 | case_stmt( 6, exprtk_fallthrough) case_stmt( 5, exprtk_fallthrough) 16456 | 16457 | #endif 16458 | case_stmt( 4, exprtk_fallthrough) case_stmt( 3, exprtk_fallthrough) 16459 | case_stmt( 2, exprtk_fallthrough) case_stmt( 1, (void)0;) 16460 | } 16461 | 16462 | #undef case_stmt 16463 | 16464 | return result; 16465 | } 16466 | 16467 | T r[] = { 16468 | T(0), T(0), T(0), T(0), T(0), T(0), T(0), T(0), 16469 | T(0), T(0), T(0), T(0), T(0), T(0), T(0), T(0) 16470 | }; 16471 | 16472 | const T* upper_bound = vec + lud.upper_bound; 16473 | 16474 | while (vec < upper_bound) 16475 | { 16476 | #define exprtk_loop(N) \ 16477 | r[N] += vec[N]; \ 16478 | 16479 | exprtk_loop( 0) exprtk_loop( 1) 16480 | exprtk_loop( 2) exprtk_loop( 3) 16481 | #ifndef exprtk_disable_superscalar_unroll 16482 | exprtk_loop( 4) exprtk_loop( 5) 16483 | exprtk_loop( 6) exprtk_loop( 7) 16484 | exprtk_loop( 8) exprtk_loop( 9) 16485 | exprtk_loop(10) exprtk_loop(11) 16486 | exprtk_loop(12) exprtk_loop(13) 16487 | exprtk_loop(14) exprtk_loop(15) 16488 | #endif 16489 | 16490 | vec += lud.batch_size; 16491 | } 16492 | 16493 | int i = 0; 16494 | 16495 | switch (lud.remainder) 16496 | { 16497 | #define case_stmt(N,fall_through) \ 16498 | case N : r[0] += vec[i++]; \ 16499 | fall_through \ 16500 | 16501 | #ifndef exprtk_disable_superscalar_unroll 16502 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 16503 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 16504 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 16505 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 16506 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 16507 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 16508 | #endif 16509 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 16510 | case_stmt( 1, (void)0;) 16511 | } 16512 | 16513 | #undef exprtk_loop 16514 | #undef case_stmt 16515 | 16516 | return (r[ 0] + r[ 1] + r[ 2] + r[ 3]) 16517 | #ifndef exprtk_disable_superscalar_unroll 16518 | + (r[ 4] + r[ 5] + r[ 6] + r[ 7]) 16519 | + (r[ 8] + r[ 9] + r[10] + r[11]) 16520 | + (r[12] + r[13] + r[14] + r[15]) 16521 | #endif 16522 | ; 16523 | } 16524 | }; 16525 | 16526 | template <typename T> 16527 | struct vec_mul_op 16528 | { 16529 | typedef vector_interface<T>* ivector_ptr; 16530 | 16531 | static inline T process(const ivector_ptr v) 16532 | { 16533 | const T* vec = v->vec()->vds().data(); 16534 | const std::size_t vec_size = v->vec()->size(); 16535 | 16536 | loop_unroll::details lud(vec_size); 16537 | 16538 | if (vec_size <= static_cast<std::size_t>(lud.batch_size)) 16539 | { 16540 | T result = T(1); 16541 | int i = 0; 16542 | 16543 | switch (vec_size) 16544 | { 16545 | #define case_stmt(N,fall_through) \ 16546 | case N : result *= vec[i++]; \ 16547 | fall_through \ 16548 | 16549 | #ifndef exprtk_disable_superscalar_unroll 16550 | case_stmt(16, exprtk_fallthrough) case_stmt(15, exprtk_fallthrough) 16551 | case_stmt(14, exprtk_fallthrough) case_stmt(13, exprtk_fallthrough) 16552 | case_stmt(12, exprtk_fallthrough) case_stmt(11, exprtk_fallthrough) 16553 | case_stmt(10, exprtk_fallthrough) case_stmt( 9, exprtk_fallthrough) 16554 | case_stmt( 8, exprtk_fallthrough) case_stmt( 7, exprtk_fallthrough) 16555 | case_stmt( 6, exprtk_fallthrough) case_stmt( 5, exprtk_fallthrough) 16556 | #endif 16557 | case_stmt( 4, exprtk_fallthrough) case_stmt( 3, exprtk_fallthrough) 16558 | case_stmt( 2, exprtk_fallthrough) case_stmt( 1, (void)0;) 16559 | } 16560 | 16561 | #undef case_stmt 16562 | 16563 | return result; 16564 | } 16565 | 16566 | T r[] = { 16567 | T(1), T(1), T(1), T(1), T(1), T(1), T(1), T(1), 16568 | T(1), T(1), T(1), T(1), T(1), T(1), T(1), T(1) 16569 | }; 16570 | 16571 | const T* upper_bound = vec + lud.upper_bound; 16572 | 16573 | while (vec < upper_bound) 16574 | { 16575 | #define exprtk_loop(N) \ 16576 | r[N] *= vec[N]; \ 16577 | 16578 | exprtk_loop( 0) exprtk_loop( 1) 16579 | exprtk_loop( 2) exprtk_loop( 3) 16580 | #ifndef exprtk_disable_superscalar_unroll 16581 | exprtk_loop( 4) exprtk_loop( 5) 16582 | exprtk_loop( 6) exprtk_loop( 7) 16583 | exprtk_loop( 8) exprtk_loop( 9) 16584 | exprtk_loop(10) exprtk_loop(11) 16585 | exprtk_loop(12) exprtk_loop(13) 16586 | exprtk_loop(14) exprtk_loop(15) 16587 | #endif 16588 | 16589 | vec += lud.batch_size; 16590 | } 16591 | 16592 | int i = 0; 16593 | 16594 | switch (lud.remainder) 16595 | { 16596 | #define case_stmt(N,fall_through) \ 16597 | case N : r[0] *= vec[i++]; \ 16598 | fall_through \ 16599 | 16600 | #ifndef exprtk_disable_superscalar_unroll 16601 | case_stmt(15, exprtk_fallthrough) case_stmt(14, exprtk_fallthrough) 16602 | case_stmt(13, exprtk_fallthrough) case_stmt(12, exprtk_fallthrough) 16603 | case_stmt(11, exprtk_fallthrough) case_stmt(10, exprtk_fallthrough) 16604 | case_stmt( 9, exprtk_fallthrough) case_stmt( 8, exprtk_fallthrough) 16605 | case_stmt( 7, exprtk_fallthrough) case_stmt( 6, exprtk_fallthrough) 16606 | case_stmt( 5, exprtk_fallthrough) case_stmt( 4, exprtk_fallthrough) 16607 | #endif 16608 | case_stmt( 3, exprtk_fallthrough) case_stmt( 2, exprtk_fallthrough) 16609 | case_stmt( 1, (void)0;) 16610 | } 16611 | 16612 | #undef exprtk_loop 16613 | #undef case_stmt 16614 | 16615 | return (r[ 0] * r[ 1] * r[ 2] * r[ 3]) 16616 | #ifndef exprtk_disable_superscalar_unroll 16617 | * (r[ 4] * r[ 5] * r[ 6] * r[ 7]) 16618 | * (r[ 8] * r[ 9] * r[10] * r[11]) 16619 | * (r[12] * r[13] * r[14] * r[15]) 16620 | #endif 16621 | ; 16622 | } 16623 | }; 16624 | 16625 | template <typename T> 16626 | struct vec_avg_op 16627 | { 16628 | typedef vector_interface<T>* ivector_ptr; 16629 | 16630 | static inline T process(const ivector_ptr v) 16631 | { 16632 | const T vec_size = T(v->vec()->size()); 16633 | return vec_add_op<T>::process(v) / vec_size; 16634 | } 16635 | }; 16636 | 16637 | template <typename T> 16638 | struct vec_min_op 16639 | { 16640 | typedef vector_interface<T>* ivector_ptr; 16641 | 16642 | static inline T process(const ivector_ptr v) 16643 | { 16644 | const T* vec = v->vec()->vds().data(); 16645 | const std::size_t vec_size = v->vec()->size(); 16646 | 16647 | T result = vec[0]; 16648 | 16649 | for (std::size_t i = 1; i < vec_size; ++i) 16650 | { 16651 | const T v_i = vec[i]; 16652 | 16653 | if (v_i < result) 16654 | result = v_i; 16655 | } 16656 | 16657 | return result; 16658 | } 16659 | }; 16660 | 16661 | template <typename T> 16662 | struct vec_max_op 16663 | { 16664 | typedef vector_interface<T>* ivector_ptr; 16665 | 16666 | static inline T process(const ivector_ptr v) 16667 | { 16668 | const T* vec = v->vec()->vds().data(); 16669 | const std::size_t vec_size = v->vec()->size(); 16670 | 16671 | T result = vec[0]; 16672 | 16673 | for (std::size_t i = 1; i < vec_size; ++i) 16674 | { 16675 | const T v_i = vec[i]; 16676 | 16677 | if (v_i > result) 16678 | result = v_i; 16679 | } 16680 | 16681 | return result; 16682 | } 16683 | }; 16684 | 16685 | template <typename T> 16686 | class vov_base_node : public expression_node<T> 16687 | { 16688 | public: 16689 | 16690 | virtual ~vov_base_node() 16691 | {} 16692 | 16693 | inline virtual operator_type operation() const 16694 | { 16695 | return details::e_default; 16696 | } 16697 | 16698 | virtual const T& v0() const = 0; 16699 | 16700 | virtual const T& v1() const = 0; 16701 | }; 16702 | 16703 | template <typename T> 16704 | class cov_base_node : public expression_node<T> 16705 | { 16706 | public: 16707 | 16708 | virtual ~cov_base_node() 16709 | {} 16710 | 16711 | inline virtual operator_type operation() const 16712 | { 16713 | return details::e_default; 16714 | } 16715 | 16716 | virtual const T c() const = 0; 16717 | 16718 | virtual const T& v() const = 0; 16719 | }; 16720 | 16721 | template <typename T> 16722 | class voc_base_node : public expression_node<T> 16723 | { 16724 | public: 16725 | 16726 | virtual ~voc_base_node() 16727 | {} 16728 | 16729 | inline virtual operator_type operation() const 16730 | { 16731 | return details::e_default; 16732 | } 16733 | 16734 | virtual const T c() const = 0; 16735 | 16736 | virtual const T& v() const = 0; 16737 | }; 16738 | 16739 | template <typename T> 16740 | class vob_base_node : public expression_node<T> 16741 | { 16742 | public: 16743 | 16744 | virtual ~vob_base_node() 16745 | {} 16746 | 16747 | virtual const T& v() const = 0; 16748 | }; 16749 | 16750 | template <typename T> 16751 | class bov_base_node : public expression_node<T> 16752 | { 16753 | public: 16754 | 16755 | virtual ~bov_base_node() 16756 | {} 16757 | 16758 | virtual const T& v() const = 0; 16759 | }; 16760 | 16761 | template <typename T> 16762 | class cob_base_node : public expression_node<T> 16763 | { 16764 | public: 16765 | 16766 | virtual ~cob_base_node() 16767 | {} 16768 | 16769 | inline virtual operator_type operation() const 16770 | { 16771 | return details::e_default; 16772 | } 16773 | 16774 | virtual const T c() const = 0; 16775 | 16776 | virtual void set_c(const T) = 0; 16777 | 16778 | virtual expression_node<T>* move_branch(const std::size_t& index) = 0; 16779 | }; 16780 | 16781 | template <typename T> 16782 | class boc_base_node : public expression_node<T> 16783 | { 16784 | public: 16785 | 16786 | virtual ~boc_base_node() 16787 | {} 16788 | 16789 | inline virtual operator_type operation() const 16790 | { 16791 | return details::e_default; 16792 | } 16793 | 16794 | virtual const T c() const = 0; 16795 | 16796 | virtual void set_c(const T) = 0; 16797 | 16798 | virtual expression_node<T>* move_branch(const std::size_t& index) = 0; 16799 | }; 16800 | 16801 | template <typename T> 16802 | class uv_base_node : public expression_node<T> 16803 | { 16804 | public: 16805 | 16806 | virtual ~uv_base_node() 16807 | {} 16808 | 16809 | inline virtual operator_type operation() const 16810 | { 16811 | return details::e_default; 16812 | } 16813 | 16814 | virtual const T& v() const = 0; 16815 | }; 16816 | 16817 | template <typename T> 16818 | class sos_base_node : public expression_node<T> 16819 | { 16820 | public: 16821 | 16822 | virtual ~sos_base_node() 16823 | {} 16824 | 16825 | inline virtual operator_type operation() const 16826 | { 16827 | return details::e_default; 16828 | } 16829 | }; 16830 | 16831 | template <typename T> 16832 | class sosos_base_node : public expression_node<T> 16833 | { 16834 | public: 16835 | 16836 | virtual ~sosos_base_node() 16837 | {} 16838 | 16839 | inline virtual operator_type operation() const 16840 | { 16841 | return details::e_default; 16842 | } 16843 | }; 16844 | 16845 | template <typename T> 16846 | class T0oT1oT2_base_node : public expression_node<T> 16847 | { 16848 | public: 16849 | 16850 | virtual ~T0oT1oT2_base_node() 16851 | {} 16852 | 16853 | virtual std::string type_id() const = 0; 16854 | }; 16855 | 16856 | template <typename T> 16857 | class T0oT1oT2oT3_base_node : public expression_node<T> 16858 | { 16859 | public: 16860 | 16861 | virtual ~T0oT1oT2oT3_base_node() 16862 | {} 16863 | 16864 | virtual std::string type_id() const = 0; 16865 | }; 16866 | 16867 | template <typename T, typename Operation> 16868 | class unary_variable_node exprtk_final : public uv_base_node<T> 16869 | { 16870 | public: 16871 | 16872 | typedef expression_node<T>* expression_ptr; 16873 | typedef Operation operation_t; 16874 | 16875 | explicit unary_variable_node(const T& var) 16876 | : v_(var) 16877 | {} 16878 | 16879 | inline T value() const exprtk_override 16880 | { 16881 | return Operation::process(v_); 16882 | } 16883 | 16884 | inline typename expression_node<T>::node_type type() const exprtk_override 16885 | { 16886 | return Operation::type(); 16887 | } 16888 | 16889 | inline operator_type operation() const exprtk_override 16890 | { 16891 | return Operation::operation(); 16892 | } 16893 | 16894 | inline const T& v() const exprtk_override 16895 | { 16896 | return v_; 16897 | } 16898 | 16899 | private: 16900 | 16901 | unary_variable_node(const unary_variable_node<T,Operation>&) exprtk_delete; 16902 | unary_variable_node<T,Operation>& operator=(const unary_variable_node<T,Operation>&) exprtk_delete; 16903 | 16904 | const T& v_; 16905 | }; 16906 | 16907 | template <typename T> 16908 | class uvouv_node exprtk_final : public expression_node<T> 16909 | { 16910 | public: 16911 | 16912 | // UOpr1(v0) Op UOpr2(v1) 16913 | typedef typename details::functor_t<T> functor_t; 16914 | typedef typename functor_t::bfunc_t bfunc_t; 16915 | typedef typename functor_t::ufunc_t ufunc_t; 16916 | typedef expression_node<T>* expression_ptr; 16917 | 16918 | explicit uvouv_node(const T& var0,const T& var1, 16919 | ufunc_t uf0, ufunc_t uf1, bfunc_t bf) 16920 | : v0_(var0) 16921 | , v1_(var1) 16922 | , u0_(uf0 ) 16923 | , u1_(uf1 ) 16924 | , f_ (bf ) 16925 | {} 16926 | 16927 | inline T value() const exprtk_override 16928 | { 16929 | return f_(u0_(v0_),u1_(v1_)); 16930 | } 16931 | 16932 | inline typename expression_node<T>::node_type type() const exprtk_override 16933 | { 16934 | return expression_node<T>::e_uvouv; 16935 | } 16936 | 16937 | inline const T& v0() 16938 | { 16939 | return v0_; 16940 | } 16941 | 16942 | inline const T& v1() 16943 | { 16944 | return v1_; 16945 | } 16946 | 16947 | inline ufunc_t u0() 16948 | { 16949 | return u0_; 16950 | } 16951 | 16952 | inline ufunc_t u1() 16953 | { 16954 | return u1_; 16955 | } 16956 | 16957 | inline ufunc_t f() 16958 | { 16959 | return f_; 16960 | } 16961 | 16962 | private: 16963 | 16964 | uvouv_node(const uvouv_node<T>&) exprtk_delete; 16965 | uvouv_node<T>& operator=(const uvouv_node<T>&) exprtk_delete; 16966 | 16967 | const T& v0_; 16968 | const T& v1_; 16969 | const ufunc_t u0_; 16970 | const ufunc_t u1_; 16971 | const bfunc_t f_; 16972 | }; 16973 | 16974 | template <typename T, typename Operation> 16975 | class unary_branch_node exprtk_final : public expression_node<T> 16976 | { 16977 | public: 16978 | 16979 | typedef Operation operation_t; 16980 | typedef expression_node<T>* expression_ptr; 16981 | typedef std::pair<expression_ptr,bool> branch_t; 16982 | 16983 | explicit unary_branch_node(expression_ptr branch) 16984 | { 16985 | construct_branch_pair(branch_, branch); 16986 | } 16987 | 16988 | inline T value() const exprtk_override 16989 | { 16990 | return Operation::process(branch_.first->value()); 16991 | } 16992 | 16993 | inline typename expression_node<T>::node_type type() const exprtk_override 16994 | { 16995 | return Operation::type(); 16996 | } 16997 | 16998 | inline bool valid() const exprtk_override 16999 | { 17000 | return branch_.first && branch_.first->valid(); 17001 | } 17002 | 17003 | inline operator_type operation() 17004 | { 17005 | return Operation::operation(); 17006 | } 17007 | 17008 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 17009 | { 17010 | return branch_.first; 17011 | } 17012 | 17013 | inline void release() 17014 | { 17015 | branch_.second = false; 17016 | } 17017 | 17018 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 17019 | { 17020 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 17021 | } 17022 | 17023 | std::size_t node_depth() const exprtk_override 17024 | { 17025 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 17026 | } 17027 | 17028 | private: 17029 | 17030 | unary_branch_node(const unary_branch_node<T,Operation>&) exprtk_delete; 17031 | unary_branch_node<T,Operation>& operator=(const unary_branch_node<T,Operation>&) exprtk_delete; 17032 | 17033 | branch_t branch_; 17034 | }; 17035 | 17036 | template <typename T> struct is_const { enum {result = 0}; }; 17037 | template <typename T> struct is_const <const T> { enum {result = 1}; }; 17038 | template <typename T> struct is_const_ref { enum {result = 0}; }; 17039 | template <typename T> struct is_const_ref <const T&> { enum {result = 1}; }; 17040 | template <typename T> struct is_ref { enum {result = 0}; }; 17041 | template <typename T> struct is_ref<T&> { enum {result = 1}; }; 17042 | template <typename T> struct is_ref<const T&> { enum {result = 0}; }; 17043 | 17044 | template <std::size_t State> 17045 | struct param_to_str { static std::string result() { static const std::string r("v"); return r; } }; 17046 | 17047 | template <> 17048 | struct param_to_str<0> { static std::string result() { static const std::string r("c"); return r; } }; 17049 | 17050 | #define exprtk_crtype(Type) \ 17051 | param_to_str<is_const_ref< Type >::result>::result() \ 17052 | 17053 | template <typename T> 17054 | struct T0oT1oT2process 17055 | { 17056 | typedef typename details::functor_t<T> functor_t; 17057 | typedef typename functor_t::bfunc_t bfunc_t; 17058 | 17059 | struct mode0 17060 | { 17061 | static inline T process(const T& t0, const T& t1, const T& t2, const bfunc_t bf0, const bfunc_t bf1) 17062 | { 17063 | // (T0 o0 T1) o1 T2 17064 | return bf1(bf0(t0,t1),t2); 17065 | } 17066 | 17067 | template <typename T0, typename T1, typename T2> 17068 | static inline std::string id() 17069 | { 17070 | static const std::string result = "(" + exprtk_crtype(T0) + "o" + 17071 | exprtk_crtype(T1) + ")o(" + 17072 | exprtk_crtype(T2) + ")" ; 17073 | return result; 17074 | } 17075 | }; 17076 | 17077 | struct mode1 17078 | { 17079 | static inline T process(const T& t0, const T& t1, const T& t2, const bfunc_t bf0, const bfunc_t bf1) 17080 | { 17081 | // T0 o0 (T1 o1 T2) 17082 | return bf0(t0,bf1(t1,t2)); 17083 | } 17084 | 17085 | template <typename T0, typename T1, typename T2> 17086 | static inline std::string id() 17087 | { 17088 | static const std::string result = "(" + exprtk_crtype(T0) + ")o(" + 17089 | exprtk_crtype(T1) + "o" + 17090 | exprtk_crtype(T2) + ")" ; 17091 | return result; 17092 | } 17093 | }; 17094 | }; 17095 | 17096 | template <typename T> 17097 | struct T0oT1oT20T3process 17098 | { 17099 | typedef typename details::functor_t<T> functor_t; 17100 | typedef typename functor_t::bfunc_t bfunc_t; 17101 | 17102 | struct mode0 17103 | { 17104 | static inline T process(const T& t0, const T& t1, 17105 | const T& t2, const T& t3, 17106 | const bfunc_t bf0, const bfunc_t bf1, const bfunc_t bf2) 17107 | { 17108 | // (T0 o0 T1) o1 (T2 o2 T3) 17109 | return bf1(bf0(t0,t1),bf2(t2,t3)); 17110 | } 17111 | 17112 | template <typename T0, typename T1, typename T2, typename T3> 17113 | static inline std::string id() 17114 | { 17115 | static const std::string result = "(" + exprtk_crtype(T0) + "o" + 17116 | exprtk_crtype(T1) + ")o" + 17117 | "(" + exprtk_crtype(T2) + "o" + 17118 | exprtk_crtype(T3) + ")" ; 17119 | return result; 17120 | } 17121 | }; 17122 | 17123 | struct mode1 17124 | { 17125 | static inline T process(const T& t0, const T& t1, 17126 | const T& t2, const T& t3, 17127 | const bfunc_t bf0, const bfunc_t bf1, const bfunc_t bf2) 17128 | { 17129 | // (T0 o0 (T1 o1 (T2 o2 T3)) 17130 | return bf0(t0,bf1(t1,bf2(t2,t3))); 17131 | } 17132 | template <typename T0, typename T1, typename T2, typename T3> 17133 | static inline std::string id() 17134 | { 17135 | static const std::string result = "(" + exprtk_crtype(T0) + ")o((" + 17136 | exprtk_crtype(T1) + ")o(" + 17137 | exprtk_crtype(T2) + "o" + 17138 | exprtk_crtype(T3) + "))" ; 17139 | return result; 17140 | } 17141 | }; 17142 | 17143 | struct mode2 17144 | { 17145 | static inline T process(const T& t0, const T& t1, 17146 | const T& t2, const T& t3, 17147 | const bfunc_t bf0, const bfunc_t bf1, const bfunc_t bf2) 17148 | { 17149 | // (T0 o0 ((T1 o1 T2) o2 T3) 17150 | return bf0(t0,bf2(bf1(t1,t2),t3)); 17151 | } 17152 | 17153 | template <typename T0, typename T1, typename T2, typename T3> 17154 | static inline std::string id() 17155 | { 17156 | static const std::string result = "(" + exprtk_crtype(T0) + ")o((" + 17157 | exprtk_crtype(T1) + "o" + 17158 | exprtk_crtype(T2) + ")o(" + 17159 | exprtk_crtype(T3) + "))" ; 17160 | return result; 17161 | } 17162 | }; 17163 | 17164 | struct mode3 17165 | { 17166 | static inline T process(const T& t0, const T& t1, 17167 | const T& t2, const T& t3, 17168 | const bfunc_t bf0, const bfunc_t bf1, const bfunc_t bf2) 17169 | { 17170 | // (((T0 o0 T1) o1 T2) o2 T3) 17171 | return bf2(bf1(bf0(t0,t1),t2),t3); 17172 | } 17173 | 17174 | template <typename T0, typename T1, typename T2, typename T3> 17175 | static inline std::string id() 17176 | { 17177 | static const std::string result = "((" + exprtk_crtype(T0) + "o" + 17178 | exprtk_crtype(T1) + ")o(" + 17179 | exprtk_crtype(T2) + "))o(" + 17180 | exprtk_crtype(T3) + ")" 17181 | return result; 17182 | } 17183 | }; 17184 | 17185 | struct mode4 17186 | { 17187 | static inline T process(const T& t0, const T& t1, 17188 | const T& t2, const T& t3, 17189 | const bfunc_t bf0, const bfunc_t bf1, const bfunc_t bf2) 17190 | { 17191 | // ((T0 o0 (T1 o1 T2)) o2 T3 17192 | return bf2(bf0(t0,bf1(t1,t2)),t3); 17193 | } 17194 | 17195 | template <typename T0, typename T1, typename T2, typename T3> 17196 | static inline std::string id() 17197 | { 17198 | static const std::string result = "((" + exprtk_crtype(T0) + ")o(" + 17199 | exprtk_crtype(T1) + "o" + 17200 | exprtk_crtype(T2) + "))o(" + 17201 | exprtk_crtype(T3) + ")" ; 17202 | return result; 17203 | } 17204 | }; 17205 | }; 17206 | 17207 | #undef exprtk_crtype 17208 | 17209 | template <typename T, typename T0, typename T1> 17210 | struct nodetype_T0oT1 { static const typename expression_node<T>::node_type result; }; 17211 | template <typename T, typename T0, typename T1> 17212 | const typename expression_node<T>::node_type nodetype_T0oT1<T,T0,T1>::result = expression_node<T>::e_none; 17213 | 17214 | #define synthesis_node_type_define(T0_, T1_, v_) \ 17215 | template <typename T, typename T0, typename T1> \ 17216 | struct nodetype_T0oT1<T,T0_,T1_> { static const typename expression_node<T>::node_type result; }; \ 17217 | template <typename T, typename T0, typename T1> \ 17218 | const typename expression_node<T>::node_type nodetype_T0oT1<T,T0_,T1_>::result = expression_node<T>:: v_; \ 17219 | 17220 | synthesis_node_type_define(const T0&, const T1&, e_vov) 17221 | synthesis_node_type_define(const T0&, const T1 , e_voc) 17222 | synthesis_node_type_define(const T0 , const T1&, e_cov) 17223 | synthesis_node_type_define( T0&, T1&, e_none) 17224 | synthesis_node_type_define(const T0 , const T1 , e_none) 17225 | synthesis_node_type_define( T0&, const T1 , e_none) 17226 | synthesis_node_type_define(const T0 , T1&, e_none) 17227 | synthesis_node_type_define(const T0&, T1&, e_none) 17228 | synthesis_node_type_define( T0&, const T1&, e_none) 17229 | #undef synthesis_node_type_define 17230 | 17231 | template <typename T, typename T0, typename T1, typename T2> 17232 | struct nodetype_T0oT1oT2 { static const typename expression_node<T>::node_type result; }; 17233 | template <typename T, typename T0, typename T1, typename T2> 17234 | const typename expression_node<T>::node_type nodetype_T0oT1oT2<T,T0,T1,T2>::result = expression_node<T>::e_none; 17235 | 17236 | #define synthesis_node_type_define(T0_, T1_, T2_, v_) \ 17237 | template <typename T, typename T0, typename T1, typename T2> \ 17238 | struct nodetype_T0oT1oT2<T,T0_,T1_,T2_> { static const typename expression_node<T>::node_type result; }; \ 17239 | template <typename T, typename T0, typename T1, typename T2> \ 17240 | const typename expression_node<T>::node_type nodetype_T0oT1oT2<T,T0_,T1_,T2_>::result = expression_node<T>:: v_; \ 17241 | 17242 | synthesis_node_type_define(const T0&, const T1&, const T2&, e_vovov) 17243 | synthesis_node_type_define(const T0&, const T1&, const T2 , e_vovoc) 17244 | synthesis_node_type_define(const T0&, const T1 , const T2&, e_vocov) 17245 | synthesis_node_type_define(const T0 , const T1&, const T2&, e_covov) 17246 | synthesis_node_type_define(const T0 , const T1&, const T2 , e_covoc) 17247 | synthesis_node_type_define(const T0 , const T1 , const T2 , e_none ) 17248 | synthesis_node_type_define(const T0 , const T1 , const T2&, e_none ) 17249 | synthesis_node_type_define(const T0&, const T1 , const T2 , e_none ) 17250 | synthesis_node_type_define( T0&, T1&, T2&, e_none ) 17251 | #undef synthesis_node_type_define 17252 | 17253 | template <typename T, typename T0, typename T1, typename T2, typename T3> 17254 | struct nodetype_T0oT1oT2oT3 { static const typename expression_node<T>::node_type result; }; 17255 | template <typename T, typename T0, typename T1, typename T2, typename T3> 17256 | const typename expression_node<T>::node_type nodetype_T0oT1oT2oT3<T,T0,T1,T2,T3>::result = expression_node<T>::e_none; 17257 | 17258 | #define synthesis_node_type_define(T0_, T1_, T2_, T3_, v_) \ 17259 | template <typename T, typename T0, typename T1, typename T2, typename T3> \ 17260 | struct nodetype_T0oT1oT2oT3<T,T0_,T1_,T2_,T3_> { static const typename expression_node<T>::node_type result; }; \ 17261 | template <typename T, typename T0, typename T1, typename T2, typename T3> \ 17262 | const typename expression_node<T>::node_type nodetype_T0oT1oT2oT3<T,T0_,T1_,T2_,T3_>::result = expression_node<T>:: v_; \ 17263 | 17264 | synthesis_node_type_define(const T0&, const T1&, const T2&, const T3&, e_vovovov) 17265 | synthesis_node_type_define(const T0&, const T1&, const T2&, const T3 , e_vovovoc) 17266 | synthesis_node_type_define(const T0&, const T1&, const T2 , const T3&, e_vovocov) 17267 | synthesis_node_type_define(const T0&, const T1 , const T2&, const T3&, e_vocovov) 17268 | synthesis_node_type_define(const T0 , const T1&, const T2&, const T3&, e_covovov) 17269 | synthesis_node_type_define(const T0 , const T1&, const T2 , const T3&, e_covocov) 17270 | synthesis_node_type_define(const T0&, const T1 , const T2&, const T3 , e_vocovoc) 17271 | synthesis_node_type_define(const T0 , const T1&, const T2&, const T3 , e_covovoc) 17272 | synthesis_node_type_define(const T0&, const T1 , const T2 , const T3&, e_vococov) 17273 | synthesis_node_type_define(const T0 , const T1 , const T2 , const T3 , e_none ) 17274 | synthesis_node_type_define(const T0 , const T1 , const T2 , const T3&, e_none ) 17275 | synthesis_node_type_define(const T0 , const T1 , const T2&, const T3 , e_none ) 17276 | synthesis_node_type_define(const T0 , const T1&, const T2 , const T3 , e_none ) 17277 | synthesis_node_type_define(const T0&, const T1 , const T2 , const T3 , e_none ) 17278 | synthesis_node_type_define(const T0 , const T1 , const T2&, const T3&, e_none ) 17279 | synthesis_node_type_define(const T0&, const T1&, const T2 , const T3 , e_none ) 17280 | #undef synthesis_node_type_define 17281 | 17282 | template <typename T, typename T0, typename T1> 17283 | class T0oT1 exprtk_final : public expression_node<T> 17284 | { 17285 | public: 17286 | 17287 | typedef typename details::functor_t<T> functor_t; 17288 | typedef typename functor_t::bfunc_t bfunc_t; 17289 | typedef T value_type; 17290 | typedef T0oT1<T,T0,T1> node_type; 17291 | 17292 | T0oT1(T0 p0, T1 p1, const bfunc_t p2) 17293 | : t0_(p0) 17294 | , t1_(p1) 17295 | , f_ (p2) 17296 | {} 17297 | 17298 | inline typename expression_node<T>::node_type type() const exprtk_override 17299 | { 17300 | static const typename expression_node<T>::node_type result = nodetype_T0oT1<T,T0,T1>::result; 17301 | return result; 17302 | } 17303 | 17304 | inline operator_type operation() const exprtk_override 17305 | { 17306 | return e_default; 17307 | } 17308 | 17309 | inline T value() const exprtk_override 17310 | { 17311 | return f_(t0_,t1_); 17312 | } 17313 | 17314 | inline T0 t0() const 17315 | { 17316 | return t0_; 17317 | } 17318 | 17319 | inline T1 t1() const 17320 | { 17321 | return t1_; 17322 | } 17323 | 17324 | inline bfunc_t f() const 17325 | { 17326 | return f_; 17327 | } 17328 | 17329 | template <typename Allocator> 17330 | static inline expression_node<T>* allocate(Allocator& allocator, 17331 | T0 p0, T1 p1, 17332 | bfunc_t p2) 17333 | { 17334 | return allocator 17335 | .template allocate_type<node_type, T0, T1, bfunc_t&> 17336 | (p0, p1, p2); 17337 | } 17338 | 17339 | private: 17340 | 17341 | T0oT1(const T0oT1<T,T0,T1>&) exprtk_delete; 17342 | T0oT1<T,T0,T1>& operator=(const T0oT1<T,T0,T1>&) { return (*this); } 17343 | 17344 | T0 t0_; 17345 | T1 t1_; 17346 | const bfunc_t f_; 17347 | }; 17348 | 17349 | template <typename T, typename T0, typename T1, typename T2, typename ProcessMode> 17350 | class T0oT1oT2 exprtk_final : public T0oT1oT2_base_node<T> 17351 | { 17352 | public: 17353 | 17354 | typedef typename details::functor_t<T> functor_t; 17355 | typedef typename functor_t::bfunc_t bfunc_t; 17356 | typedef T value_type; 17357 | typedef T0oT1oT2<T,T0,T1,T2,ProcessMode> node_type; 17358 | typedef ProcessMode process_mode_t; 17359 | 17360 | T0oT1oT2(T0 p0, T1 p1, T2 p2, const bfunc_t p3, const bfunc_t p4) 17361 | : t0_(p0) 17362 | , t1_(p1) 17363 | , t2_(p2) 17364 | , f0_(p3) 17365 | , f1_(p4) 17366 | {} 17367 | 17368 | inline typename expression_node<T>::node_type type() const exprtk_override 17369 | { 17370 | static const typename expression_node<T>::node_type result = nodetype_T0oT1oT2<T,T0,T1,T2>::result; 17371 | return result; 17372 | } 17373 | 17374 | inline operator_type operation() 17375 | { 17376 | return e_default; 17377 | } 17378 | 17379 | inline T value() const exprtk_override 17380 | { 17381 | return ProcessMode::process(t0_, t1_, t2_, f0_, f1_); 17382 | } 17383 | 17384 | inline T0 t0() const 17385 | { 17386 | return t0_; 17387 | } 17388 | 17389 | inline T1 t1() const 17390 | { 17391 | return t1_; 17392 | } 17393 | 17394 | inline T2 t2() const 17395 | { 17396 | return t2_; 17397 | } 17398 | 17399 | bfunc_t f0() const 17400 | { 17401 | return f0_; 17402 | } 17403 | 17404 | bfunc_t f1() const 17405 | { 17406 | return f1_; 17407 | } 17408 | 17409 | std::string type_id() const exprtk_override 17410 | { 17411 | return id(); 17412 | } 17413 | 17414 | static inline std::string id() 17415 | { 17416 | return process_mode_t::template id<T0,T1,T2>(); 17417 | } 17418 | 17419 | template <typename Allocator> 17420 | static inline expression_node<T>* allocate(Allocator& allocator, T0 p0, T1 p1, T2 p2, bfunc_t p3, bfunc_t p4) 17421 | { 17422 | return allocator 17423 | .template allocate_type<node_type, T0, T1, T2, bfunc_t, bfunc_t> 17424 | (p0, p1, p2, p3, p4); 17425 | } 17426 | 17427 | private: 17428 | 17429 | T0oT1oT2(const node_type&) exprtk_delete; 17430 | node_type& operator=(const node_type&) exprtk_delete; 17431 | 17432 | T0 t0_; 17433 | T1 t1_; 17434 | T2 t2_; 17435 | const bfunc_t f0_; 17436 | const bfunc_t f1_; 17437 | }; 17438 | 17439 | template <typename T, typename T0_, typename T1_, typename T2_, typename T3_, typename ProcessMode> 17440 | class T0oT1oT2oT3 exprtk_final : public T0oT1oT2oT3_base_node<T> 17441 | { 17442 | public: 17443 | 17444 | typedef typename details::functor_t<T> functor_t; 17445 | typedef typename functor_t::bfunc_t bfunc_t; 17446 | typedef T value_type; 17447 | typedef T0_ T0; 17448 | typedef T1_ T1; 17449 | typedef T2_ T2; 17450 | typedef T3_ T3; 17451 | typedef T0oT1oT2oT3<T,T0,T1,T2,T3,ProcessMode> node_type; 17452 | typedef ProcessMode process_mode_t; 17453 | 17454 | T0oT1oT2oT3(T0 p0, T1 p1, T2 p2, T3 p3, bfunc_t p4, bfunc_t p5, bfunc_t p6) 17455 | : t0_(p0) 17456 | , t1_(p1) 17457 | , t2_(p2) 17458 | , t3_(p3) 17459 | , f0_(p4) 17460 | , f1_(p5) 17461 | , f2_(p6) 17462 | {} 17463 | 17464 | inline T value() const exprtk_override 17465 | { 17466 | return ProcessMode::process(t0_, t1_, t2_, t3_, f0_, f1_, f2_); 17467 | } 17468 | 17469 | inline T0 t0() const 17470 | { 17471 | return t0_; 17472 | } 17473 | 17474 | inline T1 t1() const 17475 | { 17476 | return t1_; 17477 | } 17478 | 17479 | inline T2 t2() const 17480 | { 17481 | return t2_; 17482 | } 17483 | 17484 | inline T3 t3() const 17485 | { 17486 | return t3_; 17487 | } 17488 | 17489 | inline bfunc_t f0() const 17490 | { 17491 | return f0_; 17492 | } 17493 | 17494 | inline bfunc_t f1() const 17495 | { 17496 | return f1_; 17497 | } 17498 | 17499 | inline bfunc_t f2() const 17500 | { 17501 | return f2_; 17502 | } 17503 | 17504 | inline std::string type_id() const exprtk_override 17505 | { 17506 | return id(); 17507 | } 17508 | 17509 | static inline std::string id() 17510 | { 17511 | return process_mode_t::template id<T0, T1, T2, T3>(); 17512 | } 17513 | 17514 | template <typename Allocator> 17515 | static inline expression_node<T>* allocate(Allocator& allocator, 17516 | T0 p0, T1 p1, T2 p2, T3 p3, 17517 | bfunc_t p4, bfunc_t p5, bfunc_t p6) 17518 | { 17519 | return allocator 17520 | .template allocate_type<node_type, T0, T1, T2, T3, bfunc_t, bfunc_t> 17521 | (p0, p1, p2, p3, p4, p5, p6); 17522 | } 17523 | 17524 | private: 17525 | 17526 | T0oT1oT2oT3(const node_type&) exprtk_delete; 17527 | node_type& operator=(const node_type&) exprtk_delete; 17528 | 17529 | T0 t0_; 17530 | T1 t1_; 17531 | T2 t2_; 17532 | T3 t3_; 17533 | const bfunc_t f0_; 17534 | const bfunc_t f1_; 17535 | const bfunc_t f2_; 17536 | }; 17537 | 17538 | template <typename T, typename T0, typename T1, typename T2> 17539 | class T0oT1oT2_sf3 exprtk_final : public T0oT1oT2_base_node<T> 17540 | { 17541 | public: 17542 | 17543 | typedef typename details::functor_t<T> functor_t; 17544 | typedef typename functor_t::tfunc_t tfunc_t; 17545 | typedef T value_type; 17546 | typedef T0oT1oT2_sf3<T,T0,T1,T2> node_type; 17547 | 17548 | T0oT1oT2_sf3(T0 p0, T1 p1, T2 p2, const tfunc_t p3) 17549 | : t0_(p0) 17550 | , t1_(p1) 17551 | , t2_(p2) 17552 | , f_ (p3) 17553 | {} 17554 | 17555 | inline typename expression_node<T>::node_type type() const exprtk_override 17556 | { 17557 | static const typename expression_node<T>::node_type result = nodetype_T0oT1oT2<T,T0,T1,T2>::result; 17558 | return result; 17559 | } 17560 | 17561 | inline operator_type operation() const exprtk_override 17562 | { 17563 | return e_default; 17564 | } 17565 | 17566 | inline T value() const exprtk_override 17567 | { 17568 | return f_(t0_, t1_, t2_); 17569 | } 17570 | 17571 | inline T0 t0() const 17572 | { 17573 | return t0_; 17574 | } 17575 | 17576 | inline T1 t1() const 17577 | { 17578 | return t1_; 17579 | } 17580 | 17581 | inline T2 t2() const 17582 | { 17583 | return t2_; 17584 | } 17585 | 17586 | tfunc_t f() const 17587 | { 17588 | return f_; 17589 | } 17590 | 17591 | std::string type_id() const 17592 | { 17593 | return id(); 17594 | } 17595 | 17596 | static inline std::string id() 17597 | { 17598 | return "sf3" 17599 | } 17600 | 17601 | template <typename Allocator> 17602 | static inline expression_node<T>* allocate(Allocator& allocator, T0 p0, T1 p1, T2 p2, tfunc_t p3) 17603 | { 17604 | return allocator 17605 | .template allocate_type<node_type, T0, T1, T2, tfunc_t> 17606 | (p0, p1, p2, p3); 17607 | } 17608 | 17609 | private: 17610 | 17611 | T0oT1oT2_sf3(const node_type&) exprtk_delete; 17612 | node_type& operator=(const node_type&) exprtk_delete; 17613 | 17614 | T0 t0_; 17615 | T1 t1_; 17616 | T2 t2_; 17617 | const tfunc_t f_; 17618 | }; 17619 | 17620 | template <typename T, typename T0, typename T1, typename T2> 17621 | class sf3ext_type_node : public T0oT1oT2_base_node<T> 17622 | { 17623 | public: 17624 | 17625 | virtual ~sf3ext_type_node() 17626 | {} 17627 | 17628 | virtual T0 t0() const = 0; 17629 | 17630 | virtual T1 t1() const = 0; 17631 | 17632 | virtual T2 t2() const = 0; 17633 | }; 17634 | 17635 | template <typename T, typename T0, typename T1, typename T2, typename SF3Operation> 17636 | class T0oT1oT2_sf3ext exprtk_final : public sf3ext_type_node<T,T0,T1,T2> 17637 | { 17638 | public: 17639 | 17640 | typedef T value_type; 17641 | typedef T0oT1oT2_sf3ext<T, T0, T1, T2, SF3Operation> node_type; 17642 | 17643 | T0oT1oT2_sf3ext(T0 p0, T1 p1, T2 p2) 17644 | : t0_(p0) 17645 | , t1_(p1) 17646 | , t2_(p2) 17647 | {} 17648 | 17649 | inline typename expression_node<T>::node_type type() const exprtk_override 17650 | { 17651 | static const typename expression_node<T>::node_type result = nodetype_T0oT1oT2<T,T0,T1,T2>::result; 17652 | return result; 17653 | } 17654 | 17655 | inline operator_type operation() 17656 | { 17657 | return e_default; 17658 | } 17659 | 17660 | inline T value() const exprtk_override 17661 | { 17662 | return SF3Operation::process(t0_, t1_, t2_); 17663 | } 17664 | 17665 | T0 t0() const exprtk_override 17666 | { 17667 | return t0_; 17668 | } 17669 | 17670 | T1 t1() const exprtk_override 17671 | { 17672 | return t1_; 17673 | } 17674 | 17675 | T2 t2() const exprtk_override 17676 | { 17677 | return t2_; 17678 | } 17679 | 17680 | std::string type_id() const exprtk_override 17681 | { 17682 | return id(); 17683 | } 17684 | 17685 | static inline std::string id() 17686 | { 17687 | return SF3Operation::id(); 17688 | } 17689 | 17690 | template <typename Allocator> 17691 | static inline expression_node<T>* allocate(Allocator& allocator, T0 p0, T1 p1, T2 p2) 17692 | { 17693 | return allocator 17694 | .template allocate_type<node_type, T0, T1, T2> 17695 | (p0, p1, p2); 17696 | } 17697 | 17698 | private: 17699 | 17700 | T0oT1oT2_sf3ext(const node_type&) exprtk_delete; 17701 | node_type& operator=(const node_type&) exprtk_delete; 17702 | 17703 | T0 t0_; 17704 | T1 t1_; 17705 | T2 t2_; 17706 | }; 17707 | 17708 | template <typename T> 17709 | inline bool is_sf3ext_node(const expression_node<T>* n) 17710 | { 17711 | switch (n->type()) 17712 | { 17713 | case expression_node<T>::e_vovov : return true; 17714 | case expression_node<T>::e_vovoc : return true; 17715 | case expression_node<T>::e_vocov : return true; 17716 | case expression_node<T>::e_covov : return true; 17717 | case expression_node<T>::e_covoc : return true; 17718 | default : return false; 17719 | } 17720 | } 17721 | 17722 | template <typename T, typename T0, typename T1, typename T2, typename T3> 17723 | class T0oT1oT2oT3_sf4 exprtk_final : public T0oT1oT2_base_node<T> 17724 | { 17725 | public: 17726 | 17727 | typedef typename details::functor_t<T> functor_t; 17728 | typedef typename functor_t::qfunc_t qfunc_t; 17729 | typedef T value_type; 17730 | typedef T0oT1oT2oT3_sf4<T, T0, T1, T2, T3> node_type; 17731 | 17732 | T0oT1oT2oT3_sf4(T0 p0, T1 p1, T2 p2, T3 p3, const qfunc_t p4) 17733 | : t0_(p0) 17734 | , t1_(p1) 17735 | , t2_(p2) 17736 | , t3_(p3) 17737 | , f_ (p4) 17738 | {} 17739 | 17740 | inline typename expression_node<T>::node_type type() const exprtk_override 17741 | { 17742 | static const typename expression_node<T>::node_type result = nodetype_T0oT1oT2oT3<T,T0,T1,T2,T3>::result; 17743 | return result; 17744 | } 17745 | 17746 | inline operator_type operation() const exprtk_override 17747 | { 17748 | return e_default; 17749 | } 17750 | 17751 | inline T value() const exprtk_override 17752 | { 17753 | return f_(t0_, t1_, t2_, t3_); 17754 | } 17755 | 17756 | inline T0 t0() const 17757 | { 17758 | return t0_; 17759 | } 17760 | 17761 | inline T1 t1() const 17762 | { 17763 | return t1_; 17764 | } 17765 | 17766 | inline T2 t2() const 17767 | { 17768 | return t2_; 17769 | } 17770 | 17771 | inline T3 t3() const 17772 | { 17773 | return t3_; 17774 | } 17775 | 17776 | qfunc_t f() const 17777 | { 17778 | return f_; 17779 | } 17780 | 17781 | std::string type_id() const 17782 | { 17783 | return id(); 17784 | } 17785 | 17786 | static inline std::string id() 17787 | { 17788 | return "sf4" 17789 | } 17790 | 17791 | template <typename Allocator> 17792 | static inline expression_node<T>* allocate(Allocator& allocator, T0 p0, T1 p1, T2 p2, T3 p3, qfunc_t p4) 17793 | { 17794 | return allocator 17795 | .template allocate_type<node_type, T0, T1, T2, T3, qfunc_t> 17796 | (p0, p1, p2, p3, p4); 17797 | } 17798 | 17799 | private: 17800 | 17801 | T0oT1oT2oT3_sf4(const node_type&) exprtk_delete; 17802 | node_type& operator=(const node_type&) exprtk_delete; 17803 | 17804 | T0 t0_; 17805 | T1 t1_; 17806 | T2 t2_; 17807 | T3 t3_; 17808 | const qfunc_t f_; 17809 | }; 17810 | 17811 | template <typename T, typename T0, typename T1, typename T2, typename T3, typename SF4Operation> 17812 | class T0oT1oT2oT3_sf4ext exprtk_final : public T0oT1oT2oT3_base_node<T> 17813 | { 17814 | public: 17815 | 17816 | typedef T value_type; 17817 | typedef T0oT1oT2oT3_sf4ext<T, T0, T1, T2, T3, SF4Operation> node_type; 17818 | 17819 | T0oT1oT2oT3_sf4ext(T0 p0, T1 p1, T2 p2, T3 p3) 17820 | : t0_(p0) 17821 | , t1_(p1) 17822 | , t2_(p2) 17823 | , t3_(p3) 17824 | {} 17825 | 17826 | inline typename expression_node<T>::node_type type() const exprtk_override 17827 | { 17828 | static const typename expression_node<T>::node_type result = nodetype_T0oT1oT2oT3<T,T0,T1,T2,T3>::result; 17829 | return result; 17830 | } 17831 | 17832 | inline T value() const exprtk_override 17833 | { 17834 | return SF4Operation::process(t0_, t1_, t2_, t3_); 17835 | } 17836 | 17837 | inline T0 t0() const 17838 | { 17839 | return t0_; 17840 | } 17841 | 17842 | inline T1 t1() const 17843 | { 17844 | return t1_; 17845 | } 17846 | 17847 | inline T2 t2() const 17848 | { 17849 | return t2_; 17850 | } 17851 | 17852 | inline T3 t3() const 17853 | { 17854 | return t3_; 17855 | } 17856 | 17857 | std::string type_id() const exprtk_override 17858 | { 17859 | return id(); 17860 | } 17861 | 17862 | static inline std::string id() 17863 | { 17864 | return SF4Operation::id(); 17865 | } 17866 | 17867 | template <typename Allocator> 17868 | static inline expression_node<T>* allocate(Allocator& allocator, T0 p0, T1 p1, T2 p2, T3 p3) 17869 | { 17870 | return allocator 17871 | .template allocate_type<node_type, T0, T1, T2, T3> 17872 | (p0, p1, p2, p3); 17873 | } 17874 | 17875 | private: 17876 | 17877 | T0oT1oT2oT3_sf4ext(const node_type&) exprtk_delete; 17878 | node_type& operator=(const node_type&) exprtk_delete; 17879 | 17880 | T0 t0_; 17881 | T1 t1_; 17882 | T2 t2_; 17883 | T3 t3_; 17884 | }; 17885 | 17886 | template <typename T> 17887 | inline bool is_sf4ext_node(const expression_node<T>* n) 17888 | { 17889 | switch (n->type()) 17890 | { 17891 | case expression_node<T>::e_vovovov : return true; 17892 | case expression_node<T>::e_vovovoc : return true; 17893 | case expression_node<T>::e_vovocov : return true; 17894 | case expression_node<T>::e_vocovov : return true; 17895 | case expression_node<T>::e_covovov : return true; 17896 | case expression_node<T>::e_covocov : return true; 17897 | case expression_node<T>::e_vocovoc : return true; 17898 | case expression_node<T>::e_covovoc : return true; 17899 | case expression_node<T>::e_vococov : return true; 17900 | default : return false; 17901 | } 17902 | } 17903 | 17904 | template <typename T, typename T0, typename T1> 17905 | struct T0oT1_define 17906 | { 17907 | typedef details::T0oT1<T, T0, T1> type0; 17908 | }; 17909 | 17910 | template <typename T, typename T0, typename T1, typename T2> 17911 | struct T0oT1oT2_define 17912 | { 17913 | typedef details::T0oT1oT2<T, T0, T1, T2, typename T0oT1oT2process<T>::mode0> type0; 17914 | typedef details::T0oT1oT2<T, T0, T1, T2, typename T0oT1oT2process<T>::mode1> type1; 17915 | typedef details::T0oT1oT2_sf3<T, T0, T1, T2> sf3_type; 17916 | typedef details::sf3ext_type_node<T, T0, T1, T2> sf3_type_node; 17917 | }; 17918 | 17919 | template <typename T, typename T0, typename T1, typename T2, typename T3> 17920 | struct T0oT1oT2oT3_define 17921 | { 17922 | typedef details::T0oT1oT2oT3<T, T0, T1, T2, T3, typename T0oT1oT20T3process<T>::mode0> type0; 17923 | typedef details::T0oT1oT2oT3<T, T0, T1, T2, T3, typename T0oT1oT20T3process<T>::mode1> type1; 17924 | typedef details::T0oT1oT2oT3<T, T0, T1, T2, T3, typename T0oT1oT20T3process<T>::mode2> type2; 17925 | typedef details::T0oT1oT2oT3<T, T0, T1, T2, T3, typename T0oT1oT20T3process<T>::mode3> type3; 17926 | typedef details::T0oT1oT2oT3<T, T0, T1, T2, T3, typename T0oT1oT20T3process<T>::mode4> type4; 17927 | typedef details::T0oT1oT2oT3_sf4<T, T0, T1, T2, T3> sf4_type; 17928 | }; 17929 | 17930 | template <typename T, typename Operation> 17931 | class vov_node exprtk_final : public vov_base_node<T> 17932 | { 17933 | public: 17934 | 17935 | typedef expression_node<T>* expression_ptr; 17936 | typedef Operation operation_t; 17937 | 17938 | // variable op variable node 17939 | explicit vov_node(const T& var0, const T& var1) 17940 | : v0_(var0) 17941 | , v1_(var1) 17942 | {} 17943 | 17944 | inline T value() const exprtk_override 17945 | { 17946 | return Operation::process(v0_,v1_); 17947 | } 17948 | 17949 | inline typename expression_node<T>::node_type type() const exprtk_override 17950 | { 17951 | return Operation::type(); 17952 | } 17953 | 17954 | inline operator_type operation() const exprtk_override 17955 | { 17956 | return Operation::operation(); 17957 | } 17958 | 17959 | inline const T& v0() const exprtk_override 17960 | { 17961 | return v0_; 17962 | } 17963 | 17964 | inline const T& v1() const exprtk_override 17965 | { 17966 | return v1_; 17967 | } 17968 | 17969 | protected: 17970 | 17971 | const T& v0_; 17972 | const T& v1_; 17973 | 17974 | private: 17975 | 17976 | vov_node(const vov_node<T,Operation>&) exprtk_delete; 17977 | vov_node<T,Operation>& operator=(const vov_node<T,Operation>&) exprtk_delete; 17978 | }; 17979 | 17980 | template <typename T, typename Operation> 17981 | class cov_node exprtk_final : public cov_base_node<T> 17982 | { 17983 | public: 17984 | 17985 | typedef expression_node<T>* expression_ptr; 17986 | typedef Operation operation_t; 17987 | 17988 | // constant op variable node 17989 | explicit cov_node(const T& const_var, const T& var) 17990 | : c_(const_var) 17991 | , v_(var) 17992 | {} 17993 | 17994 | inline T value() const exprtk_override 17995 | { 17996 | return Operation::process(c_,v_); 17997 | } 17998 | 17999 | inline typename expression_node<T>::node_type type() const exprtk_override 18000 | { 18001 | return Operation::type(); 18002 | } 18003 | 18004 | inline operator_type operation() const exprtk_override 18005 | { 18006 | return Operation::operation(); 18007 | } 18008 | 18009 | inline const T c() const exprtk_override 18010 | { 18011 | return c_; 18012 | } 18013 | 18014 | inline const T& v() const exprtk_override 18015 | { 18016 | return v_; 18017 | } 18018 | 18019 | protected: 18020 | 18021 | const T c_; 18022 | const T& v_; 18023 | 18024 | private: 18025 | 18026 | cov_node(const cov_node<T,Operation>&) exprtk_delete; 18027 | cov_node<T,Operation>& operator=(const cov_node<T,Operation>&) exprtk_delete; 18028 | }; 18029 | 18030 | template <typename T, typename Operation> 18031 | class voc_node exprtk_final : public voc_base_node<T> 18032 | { 18033 | public: 18034 | 18035 | typedef expression_node<T>* expression_ptr; 18036 | typedef Operation operation_t; 18037 | 18038 | // variable op constant node 18039 | explicit voc_node(const T& var, const T& const_var) 18040 | : v_(var) 18041 | , c_(const_var) 18042 | {} 18043 | 18044 | inline T value() const exprtk_override 18045 | { 18046 | return Operation::process(v_,c_); 18047 | } 18048 | 18049 | inline operator_type operation() const exprtk_override 18050 | { 18051 | return Operation::operation(); 18052 | } 18053 | 18054 | inline const T c() const exprtk_override 18055 | { 18056 | return c_; 18057 | } 18058 | 18059 | inline const T& v() const exprtk_override 18060 | { 18061 | return v_; 18062 | } 18063 | 18064 | protected: 18065 | 18066 | const T& v_; 18067 | const T c_; 18068 | 18069 | private: 18070 | 18071 | voc_node(const voc_node<T,Operation>&) exprtk_delete; 18072 | voc_node<T,Operation>& operator=(const voc_node<T,Operation>&) exprtk_delete; 18073 | }; 18074 | 18075 | template <typename T, typename Operation> 18076 | class vob_node exprtk_final : public vob_base_node<T> 18077 | { 18078 | public: 18079 | 18080 | typedef expression_node<T>* expression_ptr; 18081 | typedef std::pair<expression_ptr,bool> branch_t; 18082 | typedef Operation operation_t; 18083 | 18084 | // variable op binary node 18085 | explicit vob_node(const T& var, const expression_ptr branch) 18086 | : v_(var) 18087 | { 18088 | construct_branch_pair(branch_, branch); 18089 | assert(valid()); 18090 | } 18091 | 18092 | inline T value() const exprtk_override 18093 | { 18094 | return Operation::process(v_,branch_.first->value()); 18095 | } 18096 | 18097 | inline const T& v() const exprtk_override 18098 | { 18099 | return v_; 18100 | } 18101 | 18102 | inline bool valid() const exprtk_override 18103 | { 18104 | return branch_.first && branch_.first->valid(); 18105 | } 18106 | 18107 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 18108 | { 18109 | return branch_.first; 18110 | } 18111 | 18112 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18113 | { 18114 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18115 | } 18116 | 18117 | std::size_t node_depth() const exprtk_override 18118 | { 18119 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18120 | } 18121 | 18122 | private: 18123 | 18124 | vob_node(const vob_node<T,Operation>&) exprtk_delete; 18125 | vob_node<T,Operation>& operator=(const vob_node<T,Operation>&) exprtk_delete; 18126 | 18127 | const T& v_; 18128 | branch_t branch_; 18129 | }; 18130 | 18131 | template <typename T, typename Operation> 18132 | class bov_node exprtk_final : public bov_base_node<T> 18133 | { 18134 | public: 18135 | 18136 | typedef expression_node<T>* expression_ptr; 18137 | typedef std::pair<expression_ptr,bool> branch_t; 18138 | typedef Operation operation_t; 18139 | 18140 | // binary node op variable node 18141 | explicit bov_node(const expression_ptr branch, const T& var) 18142 | : v_(var) 18143 | { 18144 | construct_branch_pair(branch_, branch); 18145 | assert(valid()); 18146 | } 18147 | 18148 | inline T value() const exprtk_override 18149 | { 18150 | return Operation::process(branch_.first->value(),v_); 18151 | } 18152 | 18153 | inline const T& v() const exprtk_override 18154 | { 18155 | return v_; 18156 | } 18157 | 18158 | inline bool valid() const exprtk_override 18159 | { 18160 | return branch_.first && branch_.first->valid(); 18161 | } 18162 | 18163 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 18164 | { 18165 | return branch_.first; 18166 | } 18167 | 18168 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18169 | { 18170 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18171 | } 18172 | 18173 | std::size_t node_depth() const exprtk_override 18174 | { 18175 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18176 | } 18177 | 18178 | private: 18179 | 18180 | bov_node(const bov_node<T,Operation>&) exprtk_delete; 18181 | bov_node<T,Operation>& operator=(const bov_node<T,Operation>&) exprtk_delete; 18182 | 18183 | const T& v_; 18184 | branch_t branch_; 18185 | }; 18186 | 18187 | template <typename T, typename Operation> 18188 | class cob_node exprtk_final : public cob_base_node<T> 18189 | { 18190 | public: 18191 | 18192 | typedef expression_node<T>* expression_ptr; 18193 | typedef std::pair<expression_ptr,bool> branch_t; 18194 | typedef Operation operation_t; 18195 | 18196 | // constant op variable node 18197 | explicit cob_node(const T const_var, const expression_ptr branch) 18198 | : c_(const_var) 18199 | { 18200 | construct_branch_pair(branch_, branch); 18201 | assert(valid()); 18202 | } 18203 | 18204 | inline T value() const exprtk_override 18205 | { 18206 | return Operation::process(c_,branch_.first->value()); 18207 | } 18208 | 18209 | inline operator_type operation() const exprtk_override 18210 | { 18211 | return Operation::operation(); 18212 | } 18213 | 18214 | inline const T c() const exprtk_override 18215 | { 18216 | return c_; 18217 | } 18218 | 18219 | inline void set_c(const T new_c) exprtk_override 18220 | { 18221 | (*const_cast<T*>(&c_)) = new_c; 18222 | } 18223 | 18224 | inline bool valid() const exprtk_override 18225 | { 18226 | return branch_.first && branch_.first->valid(); 18227 | } 18228 | 18229 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 18230 | { 18231 | return branch_.first; 18232 | } 18233 | 18234 | inline expression_node<T>* move_branch(const std::size_t&) exprtk_override 18235 | { 18236 | branch_.second = false; 18237 | return branch_.first; 18238 | } 18239 | 18240 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18241 | { 18242 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18243 | } 18244 | 18245 | std::size_t node_depth() const exprtk_override 18246 | { 18247 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18248 | } 18249 | 18250 | private: 18251 | 18252 | cob_node(const cob_node<T,Operation>&) exprtk_delete; 18253 | cob_node<T,Operation>& operator=(const cob_node<T,Operation>&) exprtk_delete; 18254 | 18255 | const T c_; 18256 | branch_t branch_; 18257 | }; 18258 | 18259 | template <typename T, typename Operation> 18260 | class boc_node exprtk_final : public boc_base_node<T> 18261 | { 18262 | public: 18263 | 18264 | typedef expression_node<T>* expression_ptr; 18265 | typedef std::pair<expression_ptr,bool> branch_t; 18266 | typedef Operation operation_t; 18267 | 18268 | // binary node op constant node 18269 | explicit boc_node(const expression_ptr branch, const T const_var) 18270 | : c_(const_var) 18271 | { 18272 | construct_branch_pair(branch_, branch); 18273 | assert(valid()); 18274 | } 18275 | 18276 | inline T value() const exprtk_override 18277 | { 18278 | return Operation::process(branch_.first->value(),c_); 18279 | } 18280 | 18281 | inline operator_type operation() const exprtk_override 18282 | { 18283 | return Operation::operation(); 18284 | } 18285 | 18286 | inline const T c() const exprtk_override 18287 | { 18288 | return c_; 18289 | } 18290 | 18291 | inline void set_c(const T new_c) exprtk_override 18292 | { 18293 | (*const_cast<T*>(&c_)) = new_c; 18294 | } 18295 | 18296 | inline bool valid() const exprtk_override 18297 | { 18298 | return branch_.first && branch_.first->valid(); 18299 | } 18300 | 18301 | inline expression_node<T>* branch(const std::size_t&) const exprtk_override 18302 | { 18303 | return branch_.first; 18304 | } 18305 | 18306 | inline expression_node<T>* move_branch(const std::size_t&) exprtk_override 18307 | { 18308 | branch_.second = false; 18309 | return branch_.first; 18310 | } 18311 | 18312 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18313 | { 18314 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18315 | } 18316 | 18317 | std::size_t node_depth() const exprtk_override 18318 | { 18319 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18320 | } 18321 | 18322 | private: 18323 | 18324 | boc_node(const boc_node<T,Operation>&) exprtk_delete; 18325 | boc_node<T,Operation>& operator=(const boc_node<T,Operation>&) exprtk_delete; 18326 | 18327 | const T c_; 18328 | branch_t branch_; 18329 | }; 18330 | 18331 | #ifndef exprtk_disable_string_capabilities 18332 | template <typename T, typename SType0, typename SType1, typename Operation> 18333 | class sos_node exprtk_final : public sos_base_node<T> 18334 | { 18335 | public: 18336 | 18337 | typedef expression_node<T>* expression_ptr; 18338 | typedef Operation operation_t; 18339 | 18340 | // string op string node 18341 | explicit sos_node(SType0 p0, SType1 p1) 18342 | : s0_(p0) 18343 | , s1_(p1) 18344 | {} 18345 | 18346 | inline T value() const exprtk_override 18347 | { 18348 | return Operation::process(s0_,s1_); 18349 | } 18350 | 18351 | inline typename expression_node<T>::node_type type() const exprtk_override 18352 | { 18353 | return Operation::type(); 18354 | } 18355 | 18356 | inline operator_type operation() const exprtk_override 18357 | { 18358 | return Operation::operation(); 18359 | } 18360 | 18361 | inline std::string& s0() 18362 | { 18363 | return s0_; 18364 | } 18365 | 18366 | inline std::string& s1() 18367 | { 18368 | return s1_; 18369 | } 18370 | 18371 | protected: 18372 | 18373 | SType0 s0_; 18374 | SType1 s1_; 18375 | 18376 | private: 18377 | 18378 | sos_node(const sos_node<T,SType0,SType1,Operation>&) exprtk_delete; 18379 | sos_node<T,SType0,SType1,Operation>& operator=(const sos_node<T,SType0,SType1,Operation>&) exprtk_delete; 18380 | }; 18381 | 18382 | template <typename T, typename SType0, typename SType1, typename RangePack, typename Operation> 18383 | class str_xrox_node exprtk_final : public sos_base_node<T> 18384 | { 18385 | public: 18386 | 18387 | typedef expression_node<T>* expression_ptr; 18388 | typedef Operation operation_t; 18389 | typedef str_xrox_node<T,SType0,SType1,RangePack,Operation> node_type; 18390 | 18391 | // string-range op string node 18392 | explicit str_xrox_node(SType0 p0, SType1 p1, RangePack rp0) 18393 | : s0_ (p0 ) 18394 | , s1_ (p1 ) 18395 | , rp0_(rp0) 18396 | {} 18397 | 18398 | ~str_xrox_node() exprtk_override 18399 | { 18400 | rp0_.free(); 18401 | } 18402 | 18403 | inline T value() const exprtk_override 18404 | { 18405 | std::size_t r0 = 0; 18406 | std::size_t r1 = 0; 18407 | 18408 | if (rp0_(r0, r1, s0_.size())) 18409 | return Operation::process(s0_.substr(r0, (r1 - r0) + 1), s1_); 18410 | else 18411 | return T(0); 18412 | } 18413 | 18414 | inline typename expression_node<T>::node_type type() const exprtk_override 18415 | { 18416 | return Operation::type(); 18417 | } 18418 | 18419 | inline operator_type operation() const exprtk_override 18420 | { 18421 | return Operation::operation(); 18422 | } 18423 | 18424 | inline std::string& s0() 18425 | { 18426 | return s0_; 18427 | } 18428 | 18429 | inline std::string& s1() 18430 | { 18431 | return s1_; 18432 | } 18433 | 18434 | protected: 18435 | 18436 | SType0 s0_; 18437 | SType1 s1_; 18438 | RangePack rp0_; 18439 | 18440 | private: 18441 | 18442 | str_xrox_node(const node_type&) exprtk_delete; 18443 | node_type& operator=(const node_type&) exprtk_delete; 18444 | }; 18445 | 18446 | template <typename T, typename SType0, typename SType1, typename RangePack, typename Operation> 18447 | class str_xoxr_node exprtk_final : public sos_base_node<T> 18448 | { 18449 | public: 18450 | 18451 | typedef expression_node<T>* expression_ptr; 18452 | typedef Operation operation_t; 18453 | typedef str_xoxr_node<T,SType0,SType1,RangePack,Operation> node_type; 18454 | 18455 | // string op string range node 18456 | explicit str_xoxr_node(SType0 p0, SType1 p1, RangePack rp1) 18457 | : s0_ (p0 ) 18458 | , s1_ (p1 ) 18459 | , rp1_(rp1) 18460 | {} 18461 | 18462 | ~str_xoxr_node() 18463 | { 18464 | rp1_.free(); 18465 | } 18466 | 18467 | inline T value() const exprtk_override 18468 | { 18469 | std::size_t r0 = 0; 18470 | std::size_t r1 = 0; 18471 | 18472 | if (rp1_(r0, r1, s1_.size())) 18473 | { 18474 | return Operation::process 18475 | ( 18476 | s0_, 18477 | s1_.substr(r0, (r1 - r0) + 1) 18478 | ); 18479 | } 18480 | else 18481 | return T(0); 18482 | } 18483 | 18484 | inline typename expression_node<T>::node_type type() const exprtk_override 18485 | { 18486 | return Operation::type(); 18487 | } 18488 | 18489 | inline operator_type operation() const exprtk_override 18490 | { 18491 | return Operation::operation(); 18492 | } 18493 | 18494 | inline std::string& s0() 18495 | { 18496 | return s0_; 18497 | } 18498 | 18499 | inline std::string& s1() 18500 | { 18501 | return s1_; 18502 | } 18503 | 18504 | protected: 18505 | 18506 | SType0 s0_; 18507 | SType1 s1_; 18508 | RangePack rp1_; 18509 | 18510 | private: 18511 | 18512 | str_xoxr_node(const node_type&) exprtk_delete; 18513 | node_type& operator=(const node_type&) exprtk_delete; 18514 | }; 18515 | 18516 | template <typename T, typename SType0, typename SType1, typename RangePack, typename Operation> 18517 | class str_xroxr_node exprtk_final : public sos_base_node<T> 18518 | { 18519 | public: 18520 | 18521 | typedef expression_node<T>* expression_ptr; 18522 | typedef Operation operation_t; 18523 | typedef str_xroxr_node<T,SType0,SType1,RangePack,Operation> node_type; 18524 | 18525 | // string-range op string-range node 18526 | explicit str_xroxr_node(SType0 p0, SType1 p1, RangePack rp0, RangePack rp1) 18527 | : s0_ (p0 ) 18528 | , s1_ (p1 ) 18529 | , rp0_(rp0) 18530 | , rp1_(rp1) 18531 | {} 18532 | 18533 | ~str_xroxr_node() exprtk_override 18534 | { 18535 | rp0_.free(); 18536 | rp1_.free(); 18537 | } 18538 | 18539 | inline T value() const exprtk_override 18540 | { 18541 | std::size_t r0_0 = 0; 18542 | std::size_t r0_1 = 0; 18543 | std::size_t r1_0 = 0; 18544 | std::size_t r1_1 = 0; 18545 | 18546 | if ( 18547 | rp0_(r0_0, r1_0, s0_.size()) && 18548 | rp1_(r0_1, r1_1, s1_.size()) 18549 | ) 18550 | { 18551 | return Operation::process 18552 | ( 18553 | s0_.substr(r0_0, (r1_0 - r0_0) + 1), 18554 | s1_.substr(r0_1, (r1_1 - r0_1) + 1) 18555 | ); 18556 | } 18557 | else 18558 | return T(0); 18559 | } 18560 | 18561 | inline typename expression_node<T>::node_type type() const exprtk_override 18562 | { 18563 | return Operation::type(); 18564 | } 18565 | 18566 | inline operator_type operation() const exprtk_override 18567 | { 18568 | return Operation::operation(); 18569 | } 18570 | 18571 | inline std::string& s0() 18572 | { 18573 | return s0_; 18574 | } 18575 | 18576 | inline std::string& s1() 18577 | { 18578 | return s1_; 18579 | } 18580 | 18581 | protected: 18582 | 18583 | SType0 s0_; 18584 | SType1 s1_; 18585 | RangePack rp0_; 18586 | RangePack rp1_; 18587 | 18588 | private: 18589 | 18590 | str_xroxr_node(const node_type&) exprtk_delete; 18591 | node_type& operator=(const node_type&) exprtk_delete; 18592 | }; 18593 | 18594 | template <typename T, typename Operation> 18595 | class str_sogens_node exprtk_final : public binary_node<T> 18596 | { 18597 | public: 18598 | 18599 | typedef expression_node <T>* expression_ptr; 18600 | typedef string_base_node<T>* str_base_ptr; 18601 | typedef range_pack <T> range_t; 18602 | typedef range_t* range_ptr; 18603 | typedef range_interface <T> irange_t; 18604 | typedef irange_t* irange_ptr; 18605 | 18606 | using binary_node<T>::branch; 18607 | 18608 | str_sogens_node(const operator_type& opr, 18609 | expression_ptr branch0, 18610 | expression_ptr branch1) 18611 | : binary_node<T>(opr, branch0, branch1) 18612 | , str0_base_ptr_ (0) 18613 | , str1_base_ptr_ (0) 18614 | , str0_range_ptr_(0) 18615 | , str1_range_ptr_(0) 18616 | , initialised_ (false) 18617 | { 18618 | if (is_generally_string_node(branch(0))) 18619 | { 18620 | str0_base_ptr_ = dynamic_cast<str_base_ptr>(branch(0)); 18621 | 18622 | if (0 == str0_base_ptr_) 18623 | return; 18624 | 18625 | irange_ptr range = dynamic_cast<irange_ptr>(branch(0)); 18626 | 18627 | if (0 == range) 18628 | return; 18629 | 18630 | str0_range_ptr_ = &(range->range_ref()); 18631 | } 18632 | 18633 | if (is_generally_string_node(branch(1))) 18634 | { 18635 | str1_base_ptr_ = dynamic_cast<str_base_ptr>(branch(1)); 18636 | 18637 | if (0 == str1_base_ptr_) 18638 | return; 18639 | 18640 | irange_ptr range = dynamic_cast<irange_ptr>(branch(1)); 18641 | 18642 | if (0 == range) 18643 | return; 18644 | 18645 | str1_range_ptr_ = &(range->range_ref()); 18646 | } 18647 | 18648 | initialised_ = 18649 | str0_base_ptr_ && 18650 | str1_base_ptr_ && 18651 | str0_range_ptr_ && 18652 | str1_range_ptr_; 18653 | 18654 | assert(valid()); 18655 | } 18656 | 18657 | inline T value() const exprtk_override 18658 | { 18659 | branch(0)->value(); 18660 | branch(1)->value(); 18661 | 18662 | std::size_t str0_r0 = 0; 18663 | std::size_t str0_r1 = 0; 18664 | 18665 | std::size_t str1_r0 = 0; 18666 | std::size_t str1_r1 = 0; 18667 | 18668 | const range_t& range0 = (*str0_range_ptr_); 18669 | const range_t& range1 = (*str1_range_ptr_); 18670 | 18671 | if ( 18672 | range0(str0_r0, str0_r1, str0_base_ptr_->size()) && 18673 | range1(str1_r0, str1_r1, str1_base_ptr_->size()) 18674 | ) 18675 | { 18676 | return Operation::process 18677 | ( 18678 | str0_base_ptr_->str().substr(str0_r0,(str0_r1 - str0_r0)), 18679 | str1_base_ptr_->str().substr(str1_r0,(str1_r1 - str1_r0)) 18680 | ); 18681 | } 18682 | 18683 | return std::numeric_limits<T>::quiet_NaN(); 18684 | } 18685 | 18686 | inline typename expression_node<T>::node_type type() const exprtk_override 18687 | { 18688 | return Operation::type(); 18689 | } 18690 | 18691 | inline bool valid() const exprtk_override 18692 | { 18693 | return initialised_; 18694 | } 18695 | 18696 | private: 18697 | 18698 | str_sogens_node(const str_sogens_node<T,Operation>&) exprtk_delete; 18699 | str_sogens_node<T,Operation>& operator=(const str_sogens_node<T,Operation>&) exprtk_delete; 18700 | 18701 | str_base_ptr str0_base_ptr_; 18702 | str_base_ptr str1_base_ptr_; 18703 | range_ptr str0_range_ptr_; 18704 | range_ptr str1_range_ptr_; 18705 | bool initialised_; 18706 | }; 18707 | 18708 | template <typename T, typename SType0, typename SType1, typename SType2, typename Operation> 18709 | class sosos_node exprtk_final : public sosos_base_node<T> 18710 | { 18711 | public: 18712 | 18713 | typedef expression_node<T>* expression_ptr; 18714 | typedef Operation operation_t; 18715 | typedef sosos_node<T, SType0, SType1, SType2, Operation> node_type; 18716 | 18717 | // string op string op string node 18718 | explicit sosos_node(SType0 p0, SType1 p1, SType2 p2) 18719 | : s0_(p0) 18720 | , s1_(p1) 18721 | , s2_(p2) 18722 | {} 18723 | 18724 | inline T value() const exprtk_override 18725 | { 18726 | return Operation::process(s0_, s1_, s2_); 18727 | } 18728 | 18729 | inline typename expression_node<T>::node_type type() const exprtk_override 18730 | { 18731 | return Operation::type(); 18732 | } 18733 | 18734 | inline operator_type operation() const exprtk_override 18735 | { 18736 | return Operation::operation(); 18737 | } 18738 | 18739 | inline std::string& s0() 18740 | { 18741 | return s0_; 18742 | } 18743 | 18744 | inline std::string& s1() 18745 | { 18746 | return s1_; 18747 | } 18748 | 18749 | inline std::string& s2() 18750 | { 18751 | return s2_; 18752 | } 18753 | 18754 | protected: 18755 | 18756 | SType0 s0_; 18757 | SType1 s1_; 18758 | SType2 s2_; 18759 | 18760 | private: 18761 | 18762 | sosos_node(const node_type&) exprtk_delete; 18763 | node_type& operator=(const node_type&) exprtk_delete; 18764 | }; 18765 | #endif 18766 | 18767 | template <typename T, typename PowOp> 18768 | class ipow_node exprtk_final: public expression_node<T> 18769 | { 18770 | public: 18771 | 18772 | typedef expression_node<T>* expression_ptr; 18773 | typedef PowOp operation_t; 18774 | 18775 | explicit ipow_node(const T& v) 18776 | : v_(v) 18777 | {} 18778 | 18779 | inline T value() const exprtk_override 18780 | { 18781 | return PowOp::result(v_); 18782 | } 18783 | 18784 | inline typename expression_node<T>::node_type type() const exprtk_override 18785 | { 18786 | return expression_node<T>::e_ipow; 18787 | } 18788 | 18789 | private: 18790 | 18791 | ipow_node(const ipow_node<T,PowOp>&) exprtk_delete; 18792 | ipow_node<T,PowOp>& operator=(const ipow_node<T,PowOp>&) exprtk_delete; 18793 | 18794 | const T& v_; 18795 | }; 18796 | 18797 | template <typename T, typename PowOp> 18798 | class bipow_node exprtk_final : public expression_node<T> 18799 | { 18800 | public: 18801 | 18802 | typedef expression_node<T>* expression_ptr; 18803 | typedef std::pair<expression_ptr, bool> branch_t; 18804 | typedef PowOp operation_t; 18805 | 18806 | explicit bipow_node(expression_ptr branch) 18807 | { 18808 | construct_branch_pair(branch_, branch); 18809 | assert(valid()); 18810 | } 18811 | 18812 | inline T value() const exprtk_override 18813 | { 18814 | return PowOp::result(branch_.first->value()); 18815 | } 18816 | 18817 | inline typename expression_node<T>::node_type type() const exprtk_override 18818 | { 18819 | return expression_node<T>::e_ipow; 18820 | } 18821 | 18822 | inline bool valid() const exprtk_override 18823 | { 18824 | return branch_.first && branch_.first->valid(); 18825 | } 18826 | 18827 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18828 | { 18829 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18830 | } 18831 | 18832 | std::size_t node_depth() const exprtk_override 18833 | { 18834 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18835 | } 18836 | 18837 | private: 18838 | 18839 | bipow_node(const bipow_node<T,PowOp>&) exprtk_delete; 18840 | bipow_node<T,PowOp>& operator=(const bipow_node<T,PowOp>&) exprtk_delete; 18841 | 18842 | branch_t branch_; 18843 | }; 18844 | 18845 | template <typename T, typename PowOp> 18846 | class ipowinv_node exprtk_final : public expression_node<T> 18847 | { 18848 | public: 18849 | 18850 | typedef expression_node<T>* expression_ptr; 18851 | typedef PowOp operation_t; 18852 | 18853 | explicit ipowinv_node(const T& v) 18854 | : v_(v) 18855 | {} 18856 | 18857 | inline T value() const exprtk_override 18858 | { 18859 | return (T(1) / PowOp::result(v_)); 18860 | } 18861 | 18862 | inline typename expression_node<T>::node_type type() const exprtk_override 18863 | { 18864 | return expression_node<T>::e_ipowinv; 18865 | } 18866 | 18867 | private: 18868 | 18869 | ipowinv_node(const ipowinv_node<T,PowOp>&) exprtk_delete; 18870 | ipowinv_node<T,PowOp>& operator=(const ipowinv_node<T,PowOp>&) exprtk_delete; 18871 | 18872 | const T& v_; 18873 | }; 18874 | 18875 | template <typename T, typename PowOp> 18876 | class bipowinv_node exprtk_final : public expression_node<T> 18877 | { 18878 | public: 18879 | 18880 | typedef expression_node<T>* expression_ptr; 18881 | typedef std::pair<expression_ptr, bool> branch_t; 18882 | typedef PowOp operation_t; 18883 | 18884 | explicit bipowinv_node(expression_ptr branch) 18885 | { 18886 | construct_branch_pair(branch_, branch); 18887 | assert(valid()); 18888 | } 18889 | 18890 | inline T value() const exprtk_override 18891 | { 18892 | return (T(1) / PowOp::result(branch_.first->value())); 18893 | } 18894 | 18895 | inline typename expression_node<T>::node_type type() const exprtk_override 18896 | { 18897 | return expression_node<T>::e_ipowinv; 18898 | } 18899 | 18900 | inline bool valid() const exprtk_override 18901 | { 18902 | return branch_.first && branch_.first->valid(); 18903 | } 18904 | 18905 | void collect_nodes(typename expression_node<T>::noderef_list_t& node_delete_list) exprtk_override 18906 | { 18907 | expression_node<T>::ndb_t::collect(branch_, node_delete_list); 18908 | } 18909 | 18910 | std::size_t node_depth() const exprtk_override 18911 | { 18912 | return expression_node<T>::ndb_t::compute_node_depth(branch_); 18913 | } 18914 | 18915 | private: 18916 | 18917 | bipowinv_node(const bipowinv_node<T,PowOp>&) exprtk_delete; 18918 | bipowinv_node<T,PowOp>& operator=(const bipowinv_node<T,PowOp>&) exprtk_delete; 18919 | 18920 | branch_t branch_; 18921 | }; 18922 | 18923 | template <typename T> 18924 | inline bool is_vov_node(const expression_node<T>* node) 18925 | { 18926 | return (0 != dynamic_cast<const vov_base_node<T>*>(node)); 18927 | } 18928 | 18929 | template <typename T> 18930 | inline bool is_cov_node(const expression_node<T>* node) 18931 | { 18932 | return (0 != dynamic_cast<const cov_base_node<T>*>(node)); 18933 | } 18934 | 18935 | template <typename T> 18936 | inline bool is_voc_node(const expression_node<T>* node) 18937 | { 18938 | return (0 != dynamic_cast<const voc_base_node<T>*>(node)); 18939 | } 18940 | 18941 | template <typename T> 18942 | inline bool is_cob_node(const expression_node<T>* node) 18943 | { 18944 | return (0 != dynamic_cast<const cob_base_node<T>*>(node)); 18945 | } 18946 | 18947 | template <typename T> 18948 | inline bool is_boc_node(const expression_node<T>* node) 18949 | { 18950 | return (0 != dynamic_cast<const boc_base_node<T>*>(node)); 18951 | } 18952 | 18953 | template <typename T> 18954 | inline bool is_t0ot1ot2_node(const expression_node<T>* node) 18955 | { 18956 | return (0 != dynamic_cast<const T0oT1oT2_base_node<T>*>(node)); 18957 | } 18958 | 18959 | template <typename T> 18960 | inline bool is_t0ot1ot2ot3_node(const expression_node<T>* node) 18961 | { 18962 | return (0 != dynamic_cast<const T0oT1oT2oT3_base_node<T>*>(node)); 18963 | } 18964 | 18965 | template <typename T> 18966 | inline bool is_uv_node(const expression_node<T>* node) 18967 | { 18968 | return (0 != dynamic_cast<const uv_base_node<T>*>(node)); 18969 | } 18970 | 18971 | template <typename T> 18972 | inline bool is_string_node(const expression_node<T>* node) 18973 | { 18974 | return node && (expression_node<T>::e_stringvar == node->type()); 18975 | } 18976 | 18977 | template <typename T> 18978 | inline bool is_string_range_node(const expression_node<T>* node) 18979 | { 18980 | return node && (expression_node<T>::e_stringvarrng == node->type()); 18981 | } 18982 | 18983 | template <typename T> 18984 | inline bool is_const_string_node(const expression_node<T>* node) 18985 | { 18986 | return node && (expression_node<T>::e_stringconst == node->type()); 18987 | } 18988 | 18989 | template <typename T> 18990 | inline bool is_const_string_range_node(const expression_node<T>* node) 18991 | { 18992 | return node && (expression_node<T>::e_cstringvarrng == node->type()); 18993 | } 18994 | 18995 | template <typename T> 18996 | inline bool is_string_assignment_node(const expression_node<T>* node) 18997 | { 18998 | return node && (expression_node<T>::e_strass == node->type()); 18999 | } 19000 | 19001 | template <typename T> 19002 | inline bool is_string_concat_node(const expression_node<T>* node) 19003 | { 19004 | return node && (expression_node<T>::e_strconcat == node->type()); 19005 | } 19006 | 19007 | template <typename T> 19008 | inline bool is_string_function_node(const expression_node<T>* node) 19009 | { 19010 | return node && (expression_node<T>::e_strfunction == node->type()); 19011 | } 19012 | 19013 | template <typename T> 19014 | inline bool is_string_condition_node(const expression_node<T>* node) 19015 | { 19016 | return node && (expression_node<T>::e_strcondition == node->type()); 19017 | } 19018 | 19019 | template <typename T> 19020 | inline bool is_string_ccondition_node(const expression_node<T>* node) 19021 | { 19022 | return node && (expression_node<T>::e_strccondition == node->type()); 19023 | } 19024 | 19025 | template <typename T> 19026 | inline bool is_string_vararg_node(const expression_node<T>* node) 19027 | { 19028 | return node && (expression_node<T>::e_stringvararg == node->type()); 19029 | } 19030 | 19031 | template <typename T> 19032 | inline bool is_genricstring_range_node(const expression_node<T>* node) 19033 | { 19034 | return node && (expression_node<T>::e_strgenrange == node->type()); 19035 | } 19036 | 19037 | template <typename T> 19038 | inline bool is_generally_string_node(const expression_node<T>* node) 19039 | { 19040 | if (node) 19041 | { 19042 | switch (node->type()) 19043 | { 19044 | case expression_node<T>::e_stringvar : 19045 | case expression_node<T>::e_stringconst : 19046 | case expression_node<T>::e_stringvarrng : 19047 | case expression_node<T>::e_cstringvarrng : 19048 | case expression_node<T>::e_strgenrange : 19049 | case expression_node<T>::e_strass : 19050 | case expression_node<T>::e_strconcat : 19051 | case expression_node<T>::e_strfunction : 19052 | case expression_node<T>::e_strcondition : 19053 | case expression_node<T>::e_strccondition : 19054 | case expression_node<T>::e_stringvararg : return true; 19055 | default : return false; 19056 | } 19057 | } 19058 | 19059 | return false; 19060 | } 19061 | 19062 | template <typename T> 19063 | inline bool is_loop_node(const expression_node<T>* node) 19064 | { 19065 | if (node) 19066 | { 19067 | switch (node->type()) 19068 | { 19069 | case expression_node<T>::e_for : 19070 | case expression_node<T>::e_repeat : 19071 | case expression_node<T>::e_while : return true; 19072 | default : return false; 19073 | } 19074 | } 19075 | 19076 | return false; 19077 | } 19078 | 19079 | template <typename T> 19080 | inline bool is_block_node(const expression_node<T>* node) 19081 | { 19082 | if (node) 19083 | { 19084 | if (is_loop_node(node)) 19085 | { 19086 | return true; 19087 | } 19088 | 19089 | switch (node->type()) 19090 | { 19091 | case expression_node<T>::e_conditional : 19092 | case expression_node<T>::e_mswitch : 19093 | case expression_node<T>::e_switch : 19094 | case expression_node<T>::e_vararg : return true; 19095 | default : return false; 19096 | } 19097 | } 19098 | 19099 | return false; 19100 | } 19101 | 19102 | class node_allocator 19103 | { 19104 | public: 19105 | 19106 | template <typename ResultNode, typename OpType, typename ExprNode> 19107 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[1]) 19108 | { 19109 | expression_node<typename ResultNode::value_type>* result = 19110 | allocate<ResultNode>(operation, branch[0]); 19111 | result->node_depth(); 19112 | return result; 19113 | } 19114 | 19115 | template <typename ResultNode, typename OpType, typename ExprNode> 19116 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[2]) 19117 | { 19118 | expression_node<typename ResultNode::value_type>* result = 19119 | allocate<ResultNode>(operation, branch[0], branch[1]); 19120 | result->node_depth(); 19121 | return result; 19122 | } 19123 | 19124 | template <typename ResultNode, typename OpType, typename ExprNode> 19125 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[3]) 19126 | { 19127 | expression_node<typename ResultNode::value_type>* result = 19128 | allocate<ResultNode>(operation, branch[0], branch[1], branch[2]); 19129 | result->node_depth(); 19130 | return result; 19131 | } 19132 | 19133 | template <typename ResultNode, typename OpType, typename ExprNode> 19134 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[4]) 19135 | { 19136 | expression_node<typename ResultNode::value_type>* result = 19137 | allocate<ResultNode>(operation, branch[0], branch[1], branch[2], branch[3]); 19138 | result->node_depth(); 19139 | return result; 19140 | } 19141 | 19142 | template <typename ResultNode, typename OpType, typename ExprNode> 19143 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[5]) 19144 | { 19145 | expression_node<typename ResultNode::value_type>* result = 19146 | allocate<ResultNode>(operation, branch[0],branch[1], branch[2], branch[3], branch[4]); 19147 | result->node_depth(); 19148 | return result; 19149 | } 19150 | 19151 | template <typename ResultNode, typename OpType, typename ExprNode> 19152 | inline expression_node<typename ResultNode::value_type>* allocate(OpType& operation, ExprNode (&branch)[6]) 19153 | { 19154 | expression_node<typename ResultNode::value_type>* result = 19155 | allocate<ResultNode>(operation, branch[0], branch[1], branch[2], branch[3], branch[4], branch[5]); 19156 | result->node_depth(); 19157 | return result; 19158 | } 19159 | 19160 | template <typename node_type> 19161 | inline expression_node<typename node_type::value_type>* allocate() const 19162 | { 19163 | return (new node_type()); 19164 | } 19165 | 19166 | template <typename node_type, 19167 | typename Type, 19168 | typename Allocator, 19169 | template <typename, typename> class Sequence> 19170 | inline expression_node<typename node_type::value_type>* allocate(const Sequence<Type,Allocator>& seq) const 19171 | { 19172 | expression_node<typename node_type::value_type>* 19173 | result = (new node_type(seq)); 19174 | result->node_depth(); 19175 | return result; 19176 | } 19177 | 19178 | template <typename node_type, typename T1> 19179 | inline expression_node<typename node_type::value_type>* allocate(T1& t1) const 19180 | { 19181 | expression_node<typename node_type::value_type>* 19182 | result = (new node_type(t1)); 19183 | result->node_depth(); 19184 | return result; 19185 | } 19186 | 19187 | template <typename node_type, typename T1> 19188 | inline expression_node<typename node_type::value_type>* allocate_c(const T1& t1) const 19189 | { 19190 | expression_node<typename node_type::value_type>* 19191 | result = (new node_type(t1)); 19192 | result->node_depth(); 19193 | return result; 19194 | } 19195 | 19196 | template <typename node_type, 19197 | typename T1, typename T2> 19198 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2) const 19199 | { 19200 | expression_node<typename node_type::value_type>* 19201 | result = (new node_type(t1, t2)); 19202 | result->node_depth(); 19203 | return result; 19204 | } 19205 | 19206 | template <typename node_type, 19207 | typename T1, typename T2> 19208 | inline expression_node<typename node_type::value_type>* allocate_cr(const T1& t1, T2& t2) const 19209 | { 19210 | expression_node<typename node_type::value_type>* 19211 | result = (new node_type(t1, t2)); 19212 | result->node_depth(); 19213 | return result; 19214 | } 19215 | 19216 | template <typename node_type, 19217 | typename T1, typename T2> 19218 | inline expression_node<typename node_type::value_type>* allocate_rc(T1& t1, const T2& t2) const 19219 | { 19220 | expression_node<typename node_type::value_type>* 19221 | result = (new node_type(t1, t2)); 19222 | result->node_depth(); 19223 | return result; 19224 | } 19225 | 19226 | template <typename node_type, 19227 | typename T1, typename T2> 19228 | inline expression_node<typename node_type::value_type>* allocate_rr(T1& t1, T2& t2) const 19229 | { 19230 | expression_node<typename node_type::value_type>* 19231 | result = (new node_type(t1, t2)); 19232 | result->node_depth(); 19233 | return result; 19234 | } 19235 | 19236 | template <typename node_type, 19237 | typename T1, typename T2> 19238 | inline expression_node<typename node_type::value_type>* allocate_tt(T1 t1, T2 t2) const 19239 | { 19240 | expression_node<typename node_type::value_type>* 19241 | result = (new node_type(t1, t2)); 19242 | result->node_depth(); 19243 | return result; 19244 | } 19245 | 19246 | template <typename node_type, 19247 | typename T1, typename T2, typename T3> 19248 | inline expression_node<typename node_type::value_type>* allocate_ttt(T1 t1, T2 t2, T3 t3) const 19249 | { 19250 | expression_node<typename node_type::value_type>* 19251 | result = (new node_type(t1, t2, t3)); 19252 | result->node_depth(); 19253 | return result; 19254 | } 19255 | 19256 | template <typename node_type, 19257 | typename T1, typename T2, typename T3, typename T4> 19258 | inline expression_node<typename node_type::value_type>* allocate_tttt(T1 t1, T2 t2, T3 t3, T4 t4) const 19259 | { 19260 | expression_node<typename node_type::value_type>* 19261 | result = (new node_type(t1, t2, t3, t4)); 19262 | result->node_depth(); 19263 | return result; 19264 | } 19265 | 19266 | template <typename node_type, 19267 | typename T1, typename T2, typename T3> 19268 | inline expression_node<typename node_type::value_type>* allocate_rrr(T1& t1, T2& t2, T3& t3) const 19269 | { 19270 | expression_node<typename node_type::value_type>* 19271 | result = (new node_type(t1, t2, t3)); 19272 | result->node_depth(); 19273 | return result; 19274 | } 19275 | 19276 | template <typename node_type, 19277 | typename T1, typename T2, typename T3, typename T4> 19278 | inline expression_node<typename node_type::value_type>* allocate_rrrr(T1& t1, T2& t2, T3& t3, T4& t4) const 19279 | { 19280 | expression_node<typename node_type::value_type>* 19281 | result = (new node_type(t1, t2, t3, t4)); 19282 | result->node_depth(); 19283 | return result; 19284 | } 19285 | 19286 | template <typename node_type, 19287 | typename T1, typename T2, typename T3, typename T4, typename T5> 19288 | inline expression_node<typename node_type::value_type>* allocate_rrrrr(T1& t1, T2& t2, T3& t3, T4& t4, T5& t5) const 19289 | { 19290 | expression_node<typename node_type::value_type>* 19291 | result = (new node_type(t1, t2, t3, t4, t5)); 19292 | result->node_depth(); 19293 | return result; 19294 | } 19295 | 19296 | template <typename node_type, 19297 | typename T1, typename T2, typename T3> 19298 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19299 | const T3& t3) const 19300 | { 19301 | expression_node<typename node_type::value_type>* 19302 | result = (new node_type(t1, t2, t3)); 19303 | result->node_depth(); 19304 | return result; 19305 | } 19306 | 19307 | template <typename node_type, 19308 | typename T1, typename T2, 19309 | typename T3, typename T4> 19310 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19311 | const T3& t3, const T4& t4) const 19312 | { 19313 | expression_node<typename node_type::value_type>* 19314 | result = (new node_type(t1, t2, t3, t4)); 19315 | result->node_depth(); 19316 | return result; 19317 | } 19318 | 19319 | template <typename node_type, 19320 | typename T1, typename T2, 19321 | typename T3, typename T4, typename T5> 19322 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19323 | const T3& t3, const T4& t4, 19324 | const T5& t5) const 19325 | { 19326 | expression_node<typename node_type::value_type>* 19327 | result = (new node_type(t1, t2, t3, t4, t5)); 19328 | result->node_depth(); 19329 | return result; 19330 | } 19331 | 19332 | template <typename node_type, 19333 | typename T1, typename T2, 19334 | typename T3, typename T4, typename T5, typename T6> 19335 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19336 | const T3& t3, const T4& t4, 19337 | const T5& t5, const T6& t6) const 19338 | { 19339 | expression_node<typename node_type::value_type>* 19340 | result = (new node_type(t1, t2, t3, t4, t5, t6)); 19341 | result->node_depth(); 19342 | return result; 19343 | } 19344 | 19345 | template <typename node_type, 19346 | typename T1, typename T2, 19347 | typename T3, typename T4, 19348 | typename T5, typename T6, typename T7> 19349 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19350 | const T3& t3, const T4& t4, 19351 | const T5& t5, const T6& t6, 19352 | const T7& t7) const 19353 | { 19354 | expression_node<typename node_type::value_type>* 19355 | result = (new node_type(t1, t2, t3, t4, t5, t6, t7)); 19356 | result->node_depth(); 19357 | return result; 19358 | } 19359 | 19360 | template <typename node_type, 19361 | typename T1, typename T2, 19362 | typename T3, typename T4, 19363 | typename T5, typename T6, 19364 | typename T7, typename T8> 19365 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19366 | const T3& t3, const T4& t4, 19367 | const T5& t5, const T6& t6, 19368 | const T7& t7, const T8& t8) const 19369 | { 19370 | expression_node<typename node_type::value_type>* 19371 | result = (new node_type(t1, t2, t3, t4, t5, t6, t7, t8)); 19372 | result->node_depth(); 19373 | return result; 19374 | } 19375 | 19376 | template <typename node_type, 19377 | typename T1, typename T2, 19378 | typename T3, typename T4, 19379 | typename T5, typename T6, 19380 | typename T7, typename T8, typename T9> 19381 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19382 | const T3& t3, const T4& t4, 19383 | const T5& t5, const T6& t6, 19384 | const T7& t7, const T8& t8, 19385 | const T9& t9) const 19386 | { 19387 | expression_node<typename node_type::value_type>* 19388 | result = (new node_type(t1, t2, t3, t4, t5, t6, t7, t8, t9)); 19389 | result->node_depth(); 19390 | return result; 19391 | } 19392 | 19393 | template <typename node_type, 19394 | typename T1, typename T2, 19395 | typename T3, typename T4, 19396 | typename T5, typename T6, 19397 | typename T7, typename T8, 19398 | typename T9, typename T10> 19399 | inline expression_node<typename node_type::value_type>* allocate(const T1& t1, const T2& t2, 19400 | const T3& t3, const T4& t4, 19401 | const T5& t5, const T6& t6, 19402 | const T7& t7, const T8& t8, 19403 | const T9& t9, const T10& t10) const 19404 | { 19405 | expression_node<typename node_type::value_type>* 19406 | result = (new node_type(t1, t2, t3, t4, t5, t6, t7, t8, t9, t10)); 19407 | result->node_depth(); 19408 | return result; 19409 | } 19410 | 19411 | template <typename node_type, 19412 | typename T1, typename T2, typename T3> 19413 | inline expression_node<typename node_type::value_type>* allocate_type(T1 t1, T2 t2, T3 t3) const 19414 | { 19415 | expression_node<typename node_type::value_type>* 19416 | result = (new node_type(t1, t2, t3)); 19417 | result->node_depth(); 19418 | return result; 19419 | } 19420 | 19421 | template <typename node_type, 19422 | typename T1, typename T2, 19423 | typename T3, typename T4> 19424 | inline expression_node<typename node_type::value_type>* allocate_type(T1 t1, T2 t2, 19425 | T3 t3, T4 t4) const 19426 | { 19427 | expression_node<typename node_type::value_type>* 19428 | result = (new node_type(t1, t2, t3, t4)); 19429 | result->node_depth(); 19430 | return result; 19431 | } 19432 | 19433 | template <typename node_type, 19434 | typename T1, typename T2, 19435 | typename T3, typename T4, 19436 | typename T5> 19437 | inline expression_node<typename node_type::value_type>* allocate_type(T1 t1, T2 t2, 19438 | T3 t3, T4 t4, 19439 | T5 t5) const 19440 | { 19441 | expression_node<typename node_type::value_type>* 19442 | result = (new node_type(t1, t2, t3, t4, t5)); 19443 | result->node_depth(); 19444 | return result; 19445 | } 19446 | 19447 | template <typename node_type, 19448 | typename T1, typename T2, 19449 | typename T3, typename T4, 19450 | typename T5, typename T6> 19451 | inline expression_node<typename node_type::value_type>* allocate_type(T1 t1, T2 t2, 19452 | T3 t3, T4 t4, 19453 | T5 t5, T6 t6) const 19454 | { 19455 | expression_node<typename node_type::value_type>* 19456 | result = (new node_type(t1, t2, t3, t4, t5, t6)); 19457 | result->node_depth(); 19458 | return result; 19459 | } 19460 | 19461 | template <typename node_type, 19462 | typename T1, typename T2, 19463 | typename T3, typename T4, 19464 | typename T5, typename T6, typename T7> 19465 | inline expression_node<typename node_type::value_type>* allocate_type(T1 t1, T2 t2, 19466 | T3 t3, T4 t4, 19467 | T5 t5, T6 t6, 19468 | T7 t7) const 19469 | { 19470 | expression_node<typename node_type::value_type>* 19471 | result = (new node_type(t1, t2, t3, t4, t5, t6, t7)); 19472 | result->node_depth(); 19473 | return result; 19474 | } 19475 | 19476 | template <typename T> 19477 | void inline free(expression_node<T>*& e) const 19478 | { 19479 | exprtk_debug(("node_allocator::free() - deleting expression_node " 19480 | "type: %03d addr: %p\n", 19481 | static_cast<int>(e->type()), 19482 | reinterpret_cast<void*>(e))); 19483 | delete e; 19484 | e = 0; 19485 | } 19486 | }; 19487 | 19488 | inline void load_operations_map(std::multimap<std::string,details::base_operation_t,details::ilesscompare>& m) 19489 | { 19490 | #define register_op(Symbol, Type, Args) \ 19491 | m.insert(std::make_pair(std::string(Symbol),details::base_operation_t(Type,Args))); \ 19492 | 19493 | register_op("abs" , e_abs , 1) 19494 | register_op("acos" , e_acos , 1) 19495 | register_op("acosh" , e_acosh , 1) 19496 | register_op("asin" , e_asin , 1) 19497 | register_op("asinh" , e_asinh , 1) 19498 | register_op("atan" , e_atan , 1) 19499 | register_op("atanh" , e_atanh , 1) 19500 | register_op("ceil" , e_ceil , 1) 19501 | register_op("cos" , e_cos , 1) 19502 | register_op("cosh" , e_cosh , 1) 19503 | register_op("exp" , e_exp , 1) 19504 | register_op("expm1" , e_expm1 , 1) 19505 | register_op("floor" , e_floor , 1) 19506 | register_op("log" , e_log , 1) 19507 | register_op("log10" , e_log10 , 1) 19508 | register_op("log2" , e_log2 , 1) 19509 | register_op("log1p" , e_log1p , 1) 19510 | register_op("round" , e_round , 1) 19511 | register_op("sin" , e_sin , 1) 19512 | register_op("sinc" , e_sinc , 1) 19513 | register_op("sinh" , e_sinh , 1) 19514 | register_op("sec" , e_sec , 1) 19515 | register_op("csc" , e_csc , 1) 19516 | register_op("sqrt" , e_sqrt , 1) 19517 | register_op("tan" , e_tan , 1) 19518 | register_op("tanh" , e_tanh , 1) 19519 | register_op("cot" , e_cot , 1) 19520 | register_op("rad2deg" , e_r2d , 1) 19521 | register_op("deg2rad" , e_d2r , 1) 19522 | register_op("deg2grad" , e_d2g , 1) 19523 | register_op("grad2deg" , e_g2d , 1) 19524 | register_op("sgn" , e_sgn , 1) 19525 | register_op("not" , e_notl , 1) 19526 | register_op("erf" , e_erf , 1) 19527 | register_op("erfc" , e_erfc , 1) 19528 | register_op("ncdf" , e_ncdf , 1) 19529 | register_op("frac" , e_frac , 1) 19530 | register_op("trunc" , e_trunc , 1) 19531 | register_op("atan2" , e_atan2 , 2) 19532 | register_op("mod" , e_mod , 2) 19533 | register_op("logn" , e_logn , 2) 19534 | register_op("pow" , e_pow , 2) 19535 | register_op("root" , e_root , 2) 19536 | register_op("roundn" , e_roundn , 2) 19537 | register_op("equal" , e_equal , 2) 19538 | register_op("not_equal" , e_nequal , 2) 19539 | register_op("hypot" , e_hypot , 2) 19540 | register_op("shr" , e_shr , 2) 19541 | register_op("shl" , e_shl , 2) 19542 | register_op("clamp" , e_clamp , 3) 19543 | register_op("iclamp" , e_iclamp , 3) 19544 | register_op("inrange" , e_inrange , 3) 19545 | #undef register_op 19546 | } 19547 | 19548 | } // namespace details 19549 | 19550 | class function_traits 19551 | { 19552 | public: 19553 | 19554 | function_traits() 19555 | : allow_zero_parameters_(false) 19556 | , has_side_effects_(true) 19557 | , min_num_args_(0) 19558 | , max_num_args_(std::numeric_limits<std::size_t>::max()) 19559 | {} 19560 | 19561 | inline bool& allow_zero_parameters() 19562 | { 19563 | return allow_zero_parameters_; 19564 | } 19565 | 19566 | inline bool& has_side_effects() 19567 | { 19568 | return has_side_effects_; 19569 | } 19570 | 19571 | std::size_t& min_num_args() 19572 | { 19573 | return min_num_args_; 19574 | } 19575 | 19576 | std::size_t& max_num_args() 19577 | { 19578 | return max_num_args_; 19579 | } 19580 | 19581 | private: 19582 | 19583 | bool allow_zero_parameters_; 19584 | bool has_side_effects_; 19585 | std::size_t min_num_args_; 19586 | std::size_t max_num_args_; 19587 | }; 19588 | 19589 | template <typename FunctionType> 19590 | void enable_zero_parameters(FunctionType& func) 19591 | { 19592 | func.allow_zero_parameters() = true; 19593 | 19594 | if (0 != func.min_num_args()) 19595 | { 19596 | func.min_num_args() = 0; 19597 | } 19598 | } 19599 | 19600 | template <typename FunctionType> 19601 | void disable_zero_parameters(FunctionType& func) 19602 | { 19603 | func.allow_zero_parameters() = false; 19604 | } 19605 | 19606 | template <typename FunctionType> 19607 | void enable_has_side_effects(FunctionType& func) 19608 | { 19609 | func.has_side_effects() = true; 19610 | } 19611 | 19612 | template <typename FunctionType> 19613 | void disable_has_side_effects(FunctionType& func) 19614 | { 19615 | func.has_side_effects() = false; 19616 | } 19617 | 19618 | template <typename FunctionType> 19619 | void set_min_num_args(FunctionType& func, const std::size_t& num_args) 19620 | { 19621 | func.min_num_args() = num_args; 19622 | 19623 | if ((0 != func.min_num_args()) && func.allow_zero_parameters()) 19624 | func.allow_zero_parameters() = false; 19625 | } 19626 | 19627 | template <typename FunctionType> 19628 | void set_max_num_args(FunctionType& func, const std::size_t& num_args) 19629 | { 19630 | func.max_num_args() = num_args; 19631 | } 19632 | 19633 | template <typename T> 19634 | class ifunction : public function_traits 19635 | { 19636 | public: 19637 | 19638 | explicit ifunction(const std::size_t& pc) 19639 | : param_count(pc) 19640 | {} 19641 | 19642 | virtual ~ifunction() 19643 | {} 19644 | 19645 | #define empty_method_body(N) \ 19646 | { \ 19647 | exprtk_debug(("ifunction::operator() - Operator(" #N ") has not been overridden\n")); \ 19648 | return std::numeric_limits<T>::quiet_NaN(); \ 19649 | } \ 19650 | 19651 | inline virtual T operator() () 19652 | empty_method_body(0) 19653 | 19654 | inline virtual T operator() (const T&) 19655 | empty_method_body(1) 19656 | 19657 | inline virtual T operator() (const T&,const T&) 19658 | empty_method_body(2) 19659 | 19660 | inline virtual T operator() (const T&, const T&, const T&) 19661 | empty_method_body(3) 19662 | 19663 | inline virtual T operator() (const T&, const T&, const T&, const T&) 19664 | empty_method_body(4) 19665 | 19666 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&) 19667 | empty_method_body(5) 19668 | 19669 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&) 19670 | empty_method_body(6) 19671 | 19672 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19673 | empty_method_body(7) 19674 | 19675 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19676 | empty_method_body(8) 19677 | 19678 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19679 | empty_method_body(9) 19680 | 19681 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19682 | empty_method_body(10) 19683 | 19684 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19685 | const T&) 19686 | empty_method_body(11) 19687 | 19688 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19689 | const T&, const T&) 19690 | empty_method_body(12) 19691 | 19692 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19693 | const T&, const T&, const T&) 19694 | empty_method_body(13) 19695 | 19696 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19697 | const T&, const T&, const T&, const T&) 19698 | empty_method_body(14) 19699 | 19700 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19701 | const T&, const T&, const T&, const T&, const T&) 19702 | empty_method_body(15) 19703 | 19704 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19705 | const T&, const T&, const T&, const T&, const T&, const T&) 19706 | empty_method_body(16) 19707 | 19708 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19709 | const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19710 | empty_method_body(17) 19711 | 19712 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19713 | const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19714 | empty_method_body(18) 19715 | 19716 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19717 | const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19718 | empty_method_body(19) 19719 | 19720 | inline virtual T operator() (const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, 19721 | const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&, const T&) 19722 | empty_method_body(20) 19723 | 19724 | #undef empty_method_body 19725 | 19726 | std::size_t param_count; 19727 | }; 19728 | 19729 | template <typename T> 19730 | class ivararg_function : public function_traits 19731 | { 19732 | public: 19733 | 19734 | virtual ~ivararg_function() 19735 | {} 19736 | 19737 | inline virtual T operator() (const std::vector<T>&) 19738 | { 19739 | exprtk_debug(("ivararg_function::operator() - Operator has not been overridden\n")); 19740 | return std::numeric_limits<T>::quiet_NaN(); 19741 | } 19742 | }; 19743 | 19744 | template <typename T> 19745 | class igeneric_function : public function_traits 19746 | { 19747 | public: 19748 | 19749 | enum return_type 19750 | { 19751 | e_rtrn_scalar = 0, 19752 | e_rtrn_string = 1, 19753 | e_rtrn_overload = 2 19754 | }; 19755 | 19756 | typedef T type; 19757 | typedef type_store<T> generic_type; 19758 | typedef typename generic_type::parameter_list parameter_list_t; 19759 | 19760 | explicit igeneric_function(const std::string& param_seq = "", const return_type rtr_type = e_rtrn_scalar) 19761 | : parameter_sequence(param_seq) 19762 | , rtrn_type(rtr_type) 19763 | {} 19764 | 19765 | virtual ~igeneric_function() 19766 | {} 19767 | 19768 | #define igeneric_function_empty_body(N) \ 19769 | { \ 19770 | exprtk_debug(("igeneric_function::operator() - Operator(" #N ") has not been overridden\n")); \ 19771 | return std::numeric_limits<T>::quiet_NaN(); \ 19772 | } \ 19773 | 19774 | // f(i_0,i_1,....,i_N) --> Scalar 19775 | inline virtual T operator() (parameter_list_t) 19776 | igeneric_function_empty_body(1) 19777 | 19778 | // f(i_0,i_1,....,i_N) --> String 19779 | inline virtual T operator() (std::string&, parameter_list_t) 19780 | igeneric_function_empty_body(2) 19781 | 19782 | // f(psi,i_0,i_1,....,i_N) --> Scalar 19783 | inline virtual T operator() (const std::size_t&, parameter_list_t) 19784 | igeneric_function_empty_body(3) 19785 | 19786 | // f(psi,i_0,i_1,....,i_N) --> String 19787 | inline virtual T operator() (const std::size_t&, std::string&, parameter_list_t) 19788 | igeneric_function_empty_body(4) 19789 | 19790 | #undef igeneric_function_empty_body 19791 | 19792 | std::string parameter_sequence; 19793 | return_type rtrn_type; 19794 | }; 19795 | 19796 | #ifndef exprtk_disable_string_capabilities 19797 | template <typename T> 19798 | class stringvar_base 19799 | { 19800 | public: 19801 | 19802 | typedef typename details::stringvar_node<T> stringvar_node_t; 19803 | 19804 | stringvar_base(const std::string& name, stringvar_node_t* svn) 19805 | : name_(name) 19806 | , string_varnode_(svn) 19807 | {} 19808 | 19809 | bool valid() const 19810 | { 19811 | return !name_.empty() && (0 != string_varnode_); 19812 | } 19813 | 19814 | std::string name() const 19815 | { 19816 | assert(string_varnode_); 19817 | return name_; 19818 | } 19819 | 19820 | void rebase(std::string& s) 19821 | { 19822 | assert(string_varnode_); 19823 | string_varnode_->rebase(s); 19824 | } 19825 | 19826 | private: 19827 | 19828 | std::string name_; 19829 | stringvar_node_t* string_varnode_; 19830 | }; 19831 | #endif 19832 | 19833 | template <typename T> class parser; 19834 | template <typename T> class expression_helper; 19835 | 19836 | template <typename T> 19837 | class symbol_table 19838 | { 19839 | public: 19840 | 19841 | enum symtab_mutability_type 19842 | { 19843 | e_unknown = 0, 19844 | e_mutable = 1, 19845 | e_immutable = 2 19846 | }; 19847 | 19848 | typedef T (*ff00_functor)(); 19849 | typedef T (*ff01_functor)(T); 19850 | typedef T (*ff02_functor)(T, T); 19851 | typedef T (*ff03_functor)(T, T, T); 19852 | typedef T (*ff04_functor)(T, T, T, T); 19853 | typedef T (*ff05_functor)(T, T, T, T, T); 19854 | typedef T (*ff06_functor)(T, T, T, T, T, T); 19855 | typedef T (*ff07_functor)(T, T, T, T, T, T, T); 19856 | typedef T (*ff08_functor)(T, T, T, T, T, T, T, T); 19857 | typedef T (*ff09_functor)(T, T, T, T, T, T, T, T, T); 19858 | typedef T (*ff10_functor)(T, T, T, T, T, T, T, T, T, T); 19859 | typedef T (*ff11_functor)(T, T, T, T, T, T, T, T, T, T, T); 19860 | typedef T (*ff12_functor)(T, T, T, T, T, T, T, T, T, T, T, T); 19861 | typedef T (*ff13_functor)(T, T, T, T, T, T, T, T, T, T, T, T, T); 19862 | typedef T (*ff14_functor)(T, T, T, T, T, T, T, T, T, T, T, T, T, T); 19863 | typedef T (*ff15_functor)(T, T, T, T, T, T, T, T, T, T, T, T, T, T, T); 19864 | 19865 | protected: 19866 | 19867 | struct freefunc00 exprtk_final : public exprtk::ifunction<T> 19868 | { 19869 | using exprtk::ifunction<T>::operator(); 19870 | 19871 | explicit freefunc00(ff00_functor ff) : exprtk::ifunction<T>(0), f(ff) {} 19872 | inline T operator() () exprtk_override 19873 | { return f(); } 19874 | ff00_functor f; 19875 | }; 19876 | 19877 | struct freefunc01 exprtk_final : public exprtk::ifunction<T> 19878 | { 19879 | using exprtk::ifunction<T>::operator(); 19880 | 19881 | explicit freefunc01(ff01_functor ff) : exprtk::ifunction<T>(1), f(ff) {} 19882 | inline T operator() (const T& v0) exprtk_override 19883 | { return f(v0); } 19884 | ff01_functor f; 19885 | }; 19886 | 19887 | struct freefunc02 exprtk_final : public exprtk::ifunction<T> 19888 | { 19889 | using exprtk::ifunction<T>::operator(); 19890 | 19891 | explicit freefunc02(ff02_functor ff) : exprtk::ifunction<T>(2), f(ff) {} 19892 | inline T operator() (const T& v0, const T& v1) exprtk_override 19893 | { return f(v0, v1); } 19894 | ff02_functor f; 19895 | }; 19896 | 19897 | struct freefunc03 exprtk_final : public exprtk::ifunction<T> 19898 | { 19899 | using exprtk::ifunction<T>::operator(); 19900 | 19901 | explicit freefunc03(ff03_functor ff) : exprtk::ifunction<T>(3), f(ff) {} 19902 | inline T operator() (const T& v0, const T& v1, const T& v2) exprtk_override 19903 | { return f(v0, v1, v2); } 19904 | ff03_functor f; 19905 | }; 19906 | 19907 | struct freefunc04 exprtk_final : public exprtk::ifunction<T> 19908 | { 19909 | using exprtk::ifunction<T>::operator(); 19910 | 19911 | explicit freefunc04(ff04_functor ff) : exprtk::ifunction<T>(4), f(ff) {} 19912 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3) exprtk_override 19913 | { return f(v0, v1, v2, v3); } 19914 | ff04_functor f; 19915 | }; 19916 | 19917 | struct freefunc05 : public exprtk::ifunction<T> 19918 | { 19919 | using exprtk::ifunction<T>::operator(); 19920 | 19921 | explicit freefunc05(ff05_functor ff) : exprtk::ifunction<T>(5), f(ff) {} 19922 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4) exprtk_override 19923 | { return f(v0, v1, v2, v3, v4); } 19924 | ff05_functor f; 19925 | }; 19926 | 19927 | struct freefunc06 exprtk_final : public exprtk::ifunction<T> 19928 | { 19929 | using exprtk::ifunction<T>::operator(); 19930 | 19931 | explicit freefunc06(ff06_functor ff) : exprtk::ifunction<T>(6), f(ff) {} 19932 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, const T& v5) exprtk_override 19933 | { return f(v0, v1, v2, v3, v4, v5); } 19934 | ff06_functor f; 19935 | }; 19936 | 19937 | struct freefunc07 exprtk_final : public exprtk::ifunction<T> 19938 | { 19939 | using exprtk::ifunction<T>::operator(); 19940 | 19941 | explicit freefunc07(ff07_functor ff) : exprtk::ifunction<T>(7), f(ff) {} 19942 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, 19943 | const T& v5, const T& v6) exprtk_override 19944 | { return f(v0, v1, v2, v3, v4, v5, v6); } 19945 | ff07_functor f; 19946 | }; 19947 | 19948 | struct freefunc08 exprtk_final : public exprtk::ifunction<T> 19949 | { 19950 | using exprtk::ifunction<T>::operator(); 19951 | 19952 | explicit freefunc08(ff08_functor ff) : exprtk::ifunction<T>(8), f(ff) {} 19953 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, 19954 | const T& v5, const T& v6, const T& v7) exprtk_override 19955 | { return f(v0, v1, v2, v3, v4, v5, v6, v7); } 19956 | ff08_functor f; 19957 | }; 19958 | 19959 | struct freefunc09 exprtk_final : public exprtk::ifunction<T> 19960 | { 19961 | using exprtk::ifunction<T>::operator(); 19962 | 19963 | explicit freefunc09(ff09_functor ff) : exprtk::ifunction<T>(9), f(ff) {} 19964 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, 19965 | const T& v5, const T& v6, const T& v7, const T& v8) exprtk_override 19966 | { return f(v0, v1, v2, v3, v4, v5, v6, v7, v8); } 19967 | ff09_functor f; 19968 | }; 19969 | 19970 | struct freefunc10 exprtk_final : public exprtk::ifunction<T> 19971 | { 19972 | using exprtk::ifunction<T>::operator(); 19973 | 19974 | explicit freefunc10(ff10_functor ff) : exprtk::ifunction<T>(10), f(ff) {} 19975 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, 19976 | const T& v5, const T& v6, const T& v7, const T& v8, const T& v9) exprtk_override 19977 | { return f(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9); } 19978 | ff10_functor f; 19979 | }; 19980 | 19981 | struct freefunc11 exprtk_final : public exprtk::ifunction<T> 19982 | { 19983 | using exprtk::ifunction<T>::operator(); 19984 | 19985 | explicit freefunc11(ff11_functor ff) : exprtk::ifunction<T>(11), f(ff) {} 19986 | inline T operator() (const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, 19987 | const T& v5, const T& v6, const T& v7, const T& v8, const T& v9, const T& v10) exprtk_override 19988 | { return f(v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10); } 19989 | ff11_functor f; 19990 | }; 19991 | 19992 | struct freefunc12 exprtk_final : public exprtk::ifunction<T> 19993 | { 19994 | using exprtk::ifunction<T>::operator(); 19995 | 19996 | explicit freefunc12(ff12_functor ff) : exprtk::ifunction<T>(12), f(ff) {} 19997 | inline T operator() (const T& v00, const T& v01, const T& v02, const T& v03, const T& v04, 19998 | const T& v05, const T& v06, const T& v07, const T& v08, const T& v09, 19999 | const T& v10, const T& v11) exprtk_override 20000 | { return f(v00, v01, v02, v03, v04, v05, v06, v07, v08, v09, v10, v11); } 20001 | ff12_functor f; 20002 | }; 20003 | 20004 | struct freefunc13 exprtk_final : public exprtk::ifunction<T> 20005 | { 20006 | using exprtk::ifunction<T>::operator(); 20007 | 20008 | explicit freefunc13(ff13_functor ff) : exprtk::ifunction<T>(13), f(ff) {} 20009 | inline T operator() (const T& v00, const T& v01, const T& v02, const T& v03, const T& v04, 20010 | const T& v05, const T& v06, const T& v07, const T& v08, const T& v09, 20011 | const T& v10, const T& v11, const T& v12) exprtk_override 20012 | { return f(v00, v01, v02, v03, v04, v05, v06, v07, v08, v09, v10, v11, v12); } 20013 | ff13_functor f; 20014 | }; 20015 | 20016 | struct freefunc14 exprtk_final : public exprtk::ifunction<T> 20017 | { 20018 | using exprtk::ifunction<T>::operator(); 20019 | 20020 | explicit freefunc14(ff14_functor ff) : exprtk::ifunction<T>(14), f(ff) {} 20021 | inline T operator() (const T& v00, const T& v01, const T& v02, const T& v03, const T& v04, 20022 | const T& v05, const T& v06, const T& v07, const T& v08, const T& v09, 20023 | const T& v10, const T& v11, const T& v12, const T& v13) exprtk_override 20024 | { return f(v00, v01, v02, v03, v04, v05, v06, v07, v08, v09, v10, v11, v12, v13); } 20025 | ff14_functor f; 20026 | }; 20027 | 20028 | struct freefunc15 exprtk_final : public exprtk::ifunction<T> 20029 | { 20030 | using exprtk::ifunction<T>::operator(); 20031 | 20032 | explicit freefunc15(ff15_functor ff) : exprtk::ifunction<T>(15), f(ff) {} 20033 | inline T operator() (const T& v00, const T& v01, const T& v02, const T& v03, const T& v04, 20034 | const T& v05, const T& v06, const T& v07, const T& v08, const T& v09, 20035 | const T& v10, const T& v11, const T& v12, const T& v13, const T& v14) exprtk_override 20036 | { return f(v00, v01, v02, v03, v04, v05, v06, v07, v08, v09, v10, v11, v12, v13, v14); } 20037 | ff15_functor f; 20038 | }; 20039 | 20040 | template <typename Type, typename RawType> 20041 | struct type_store 20042 | { 20043 | typedef details::expression_node<T>* expression_ptr; 20044 | typedef typename details::variable_node<T> variable_node_t; 20045 | typedef ifunction<T> ifunction_t; 20046 | typedef ivararg_function<T> ivararg_function_t; 20047 | typedef igeneric_function<T> igeneric_function_t; 20048 | typedef details::vector_holder<T> vector_t; 20049 | #ifndef exprtk_disable_string_capabilities 20050 | typedef typename details::stringvar_node<T> stringvar_node_t; 20051 | #endif 20052 | 20053 | typedef Type type_t; 20054 | typedef type_t* type_ptr; 20055 | typedef std::pair<bool,type_ptr> type_pair_t; 20056 | typedef std::map<std::string,type_pair_t,details::ilesscompare> type_map_t; 20057 | typedef typename type_map_t::iterator tm_itr_t; 20058 | typedef typename type_map_t::const_iterator tm_const_itr_t; 20059 | 20060 | enum { lut_size = 256 }; 20061 | 20062 | type_map_t map; 20063 | std::size_t size; 20064 | 20065 | type_store() 20066 | : size(0) 20067 | {} 20068 | 20069 | struct deleter 20070 | { 20071 | #define exprtk_define_process(Type) \ 20072 | static inline void process(std::pair<bool,Type*>& n) \ 20073 | { \ 20074 | delete n.second; \ 20075 | } \ 20076 | 20077 | exprtk_define_process(variable_node_t ) 20078 | exprtk_define_process(vector_t ) 20079 | #ifndef exprtk_disable_string_capabilities 20080 | exprtk_define_process(stringvar_node_t) 20081 | #endif 20082 | 20083 | #undef exprtk_define_process 20084 | 20085 | template <typename DeleteType> 20086 | static inline void process(std::pair<bool,DeleteType*>&) 20087 | {} 20088 | }; 20089 | 20090 | inline bool symbol_exists(const std::string& symbol_name) const 20091 | { 20092 | if (symbol_name.empty()) 20093 | return false; 20094 | else if (map.end() != map.find(symbol_name)) 20095 | return true; 20096 | else 20097 | return false; 20098 | } 20099 | 20100 | template <typename PtrType> 20101 | inline std::string entity_name(const PtrType& ptr) const 20102 | { 20103 | if (map.empty()) 20104 | return std::string(); 20105 | 20106 | tm_const_itr_t itr = map.begin(); 20107 | 20108 | while (map.end() != itr) 20109 | { 20110 | if (itr->second.second == ptr) 20111 | { 20112 | return itr->first; 20113 | } 20114 | else 20115 | ++itr; 20116 | } 20117 | 20118 | return std::string(); 20119 | } 20120 | 20121 | inline bool is_constant(const std::string& symbol_name) const 20122 | { 20123 | if (symbol_name.empty()) 20124 | return false; 20125 | else 20126 | { 20127 | const tm_const_itr_t itr = map.find(symbol_name); 20128 | 20129 | if (map.end() == itr) 20130 | return false; 20131 | else 20132 | return (*itr).second.first; 20133 | } 20134 | } 20135 | 20136 | template <typename Tie, typename RType> 20137 | inline bool add_impl(const std::string& symbol_name, RType t, const bool is_const) 20138 | { 20139 | if (symbol_name.size() > 1) 20140 | { 20141 | for (std::size_t i = 0; i < details::reserved_symbols_size; ++i) 20142 | { 20143 | if (details::imatch(symbol_name, details::reserved_symbols[i])) 20144 | { 20145 | return false; 20146 | } 20147 | } 20148 | } 20149 | 20150 | const tm_itr_t itr = map.find(symbol_name); 20151 | 20152 | if (map.end() == itr) 20153 | { 20154 | map[symbol_name] = Tie::make(t,is_const); 20155 | ++size; 20156 | } 20157 | 20158 | return true; 20159 | } 20160 | 20161 | struct tie_array 20162 | { 20163 | static inline std::pair<bool,vector_t*> make(std::pair<T*,std::size_t> v, const bool is_const = false) 20164 | { 20165 | return std::make_pair(is_const, new vector_t(v.first, v.second)); 20166 | } 20167 | }; 20168 | 20169 | struct tie_stdvec 20170 | { 20171 | template <typename Allocator> 20172 | static inline std::pair<bool,vector_t*> make(std::vector<T,Allocator>& v, const bool is_const = false) 20173 | { 20174 | return std::make_pair(is_const, new vector_t(v)); 20175 | } 20176 | }; 20177 | 20178 | struct tie_vecview 20179 | { 20180 | static inline std::pair<bool,vector_t*> make(exprtk::vector_view<T>& v, const bool is_const = false) 20181 | { 20182 | return std::make_pair(is_const, new vector_t(v)); 20183 | } 20184 | }; 20185 | 20186 | struct tie_stddeq 20187 | { 20188 | template <typename Allocator> 20189 | static inline std::pair<bool,vector_t*> make(std::deque<T,Allocator>& v, const bool is_const = false) 20190 | { 20191 | return std::make_pair(is_const, new vector_t(v)); 20192 | } 20193 | }; 20194 | 20195 | template <std::size_t v_size> 20196 | inline bool add(const std::string& symbol_name, T (&v)[v_size], const bool is_const = false) 20197 | { 20198 | return add_impl<tie_array,std::pair<T*,std::size_t> > 20199 | (symbol_name, std::make_pair(v,v_size), is_const); 20200 | } 20201 | 20202 | inline bool add(const std::string& symbol_name, T* v, const std::size_t v_size, const bool is_const = false) 20203 | { 20204 | return add_impl<tie_array,std::pair<T*,std::size_t> > 20205 | (symbol_name, std::make_pair(v,v_size), is_const); 20206 | } 20207 | 20208 | template <typename Allocator> 20209 | inline bool add(const std::string& symbol_name, std::vector<T,Allocator>& v, const bool is_const = false) 20210 | { 20211 | return add_impl<tie_stdvec,std::vector<T,Allocator>&> 20212 | (symbol_name, v, is_const); 20213 | } 20214 | 20215 | inline bool add(const std::string& symbol_name, exprtk::vector_view<T>& v, const bool is_const = false) 20216 | { 20217 | return add_impl<tie_vecview,exprtk::vector_view<T>&> 20218 | (symbol_name, v, is_const); 20219 | } 20220 | 20221 | template <typename Allocator> 20222 | inline bool add(const std::string& symbol_name, std::deque<T,Allocator>& v, const bool is_const = false) 20223 | { 20224 | return add_impl<tie_stddeq,std::deque<T,Allocator>&> 20225 | (symbol_name, v, is_const); 20226 | } 20227 | 20228 | inline bool add(const std::string& symbol_name, RawType& t_, const bool is_const = false) 20229 | { 20230 | struct tie 20231 | { 20232 | static inline std::pair<bool,variable_node_t*> make(T& t, const bool is_constant = false) 20233 | { 20234 | return std::make_pair(is_constant, new variable_node_t(t)); 20235 | } 20236 | 20237 | #ifndef exprtk_disable_string_capabilities 20238 | static inline std::pair<bool,stringvar_node_t*> make(std::string& t, const bool is_constant = false) 20239 | { 20240 | return std::make_pair(is_constant, new stringvar_node_t(t)); 20241 | } 20242 | #endif 20243 | 20244 | static inline std::pair<bool,function_t*> make(function_t& t, const bool is_constant = false) 20245 | { 20246 | return std::make_pair(is_constant,&t); 20247 | } 20248 | 20249 | static inline std::pair<bool,vararg_function_t*> make(vararg_function_t& t, const bool is_constant = false) 20250 | { 20251 | return std::make_pair(is_constant,&t); 20252 | } 20253 | 20254 | static inline std::pair<bool,generic_function_t*> make(generic_function_t& t, const bool is_constant = false) 20255 | { 20256 | return std::make_pair(is_constant,&t); 20257 | } 20258 | }; 20259 | 20260 | const tm_itr_t itr = map.find(symbol_name); 20261 | 20262 | if (map.end() == itr) 20263 | { 20264 | map[symbol_name] = tie::make(t_,is_const); 20265 | ++size; 20266 | } 20267 | 20268 | return true; 20269 | } 20270 | 20271 | inline type_ptr get(const std::string& symbol_name) const 20272 | { 20273 | const tm_const_itr_t itr = map.find(symbol_name); 20274 | 20275 | if (map.end() == itr) 20276 | return reinterpret_cast<type_ptr>(0); 20277 | else 20278 | return itr->second.second; 20279 | } 20280 | 20281 | template <typename TType, typename TRawType, typename PtrType> 20282 | struct ptr_match 20283 | { 20284 | static inline bool test(const PtrType, const void*) 20285 | { 20286 | return false; 20287 | } 20288 | }; 20289 | 20290 | template <typename TType, typename TRawType> 20291 | struct ptr_match<TType,TRawType,variable_node_t*> 20292 | { 20293 | static inline bool test(const variable_node_t* p, const void* ptr) 20294 | { 20295 | exprtk_debug(("ptr_match::test() - %p <--> %p\n", reinterpret_cast<const void*>(&(p->ref())), ptr)); 20296 | return (&(p->ref()) == ptr); 20297 | } 20298 | }; 20299 | 20300 | inline type_ptr get_from_varptr(const void* ptr) const 20301 | { 20302 | tm_const_itr_t itr = map.begin(); 20303 | 20304 | while (map.end() != itr) 20305 | { 20306 | type_ptr ret_ptr = itr->second.second; 20307 | 20308 | if (ptr_match<Type,RawType,type_ptr>::test(ret_ptr,ptr)) 20309 | { 20310 | return ret_ptr; 20311 | } 20312 | 20313 | ++itr; 20314 | } 20315 | 20316 | return type_ptr(0); 20317 | } 20318 | 20319 | inline bool remove(const std::string& symbol_name, const bool delete_node = true) 20320 | { 20321 | const tm_itr_t itr = map.find(symbol_name); 20322 | 20323 | if (map.end() != itr) 20324 | { 20325 | if (delete_node) 20326 | { 20327 | deleter::process((*itr).second); 20328 | } 20329 | 20330 | map.erase(itr); 20331 | --size; 20332 | 20333 | return true; 20334 | } 20335 | else 20336 | return false; 20337 | } 20338 | 20339 | inline RawType& type_ref(const std::string& symbol_name) 20340 | { 20341 | struct init_type 20342 | { 20343 | static inline double set(double) { return (0.0); } 20344 | static inline double set(long double) { return (0.0); } 20345 | static inline float set(float) { return (0.0f); } 20346 | static inline std::string set(std::string) { return std::string(""); } 20347 | }; 20348 | 20349 | static RawType null_type = init_type::set(RawType()); 20350 | 20351 | const tm_const_itr_t itr = map.find(symbol_name); 20352 | 20353 | if (map.end() == itr) 20354 | return null_type; 20355 | else 20356 | return itr->second.second->ref(); 20357 | } 20358 | 20359 | inline void clear(const bool delete_node = true) 20360 | { 20361 | if (!map.empty()) 20362 | { 20363 | if (delete_node) 20364 | { 20365 | tm_itr_t itr = map.begin(); 20366 | tm_itr_t end = map.end (); 20367 | 20368 | while (end != itr) 20369 | { 20370 | deleter::process((*itr).second); 20371 | ++itr; 20372 | } 20373 | } 20374 | 20375 | map.clear(); 20376 | } 20377 | 20378 | size = 0; 20379 | } 20380 | 20381 | template <typename Allocator, 20382 | template <typename, typename> class Sequence> 20383 | inline std::size_t get_list(Sequence<std::pair<std::string,RawType>,Allocator>& list) const 20384 | { 20385 | std::size_t count = 0; 20386 | 20387 | if (!map.empty()) 20388 | { 20389 | tm_const_itr_t itr = map.begin(); 20390 | tm_const_itr_t end = map.end (); 20391 | 20392 | while (end != itr) 20393 | { 20394 | list.push_back(std::make_pair((*itr).first,itr->second.second->ref())); 20395 | ++itr; 20396 | ++count; 20397 | } 20398 | } 20399 | 20400 | return count; 20401 | } 20402 | 20403 | template <typename Allocator, 20404 | template <typename, typename> class Sequence> 20405 | inline std::size_t get_list(Sequence<std::string,Allocator>& vlist) const 20406 | { 20407 | std::size_t count = 0; 20408 | 20409 | if (!map.empty()) 20410 | { 20411 | tm_const_itr_t itr = map.begin(); 20412 | tm_const_itr_t end = map.end (); 20413 | 20414 | while (end != itr) 20415 | { 20416 | vlist.push_back((*itr).first); 20417 | ++itr; 20418 | ++count; 20419 | } 20420 | } 20421 | 20422 | return count; 20423 | } 20424 | }; 20425 | 20426 | typedef details::expression_node<T>* expression_ptr; 20427 | typedef typename details::variable_node<T> variable_t; 20428 | typedef typename details::vector_holder<T> vector_holder_t; 20429 | typedef variable_t* variable_ptr; 20430 | #ifndef exprtk_disable_string_capabilities 20431 | typedef typename details::stringvar_node<T> stringvar_t; 20432 | typedef stringvar_t* stringvar_ptr; 20433 | #endif 20434 | typedef ifunction <T> function_t; 20435 | typedef ivararg_function <T> vararg_function_t; 20436 | typedef igeneric_function<T> generic_function_t; 20437 | typedef function_t* function_ptr; 20438 | typedef vararg_function_t* vararg_function_ptr; 20439 | typedef generic_function_t* generic_function_ptr; 20440 | 20441 | static const std::size_t lut_size = 256; 20442 | 20443 | // Symbol Table Holder 20444 | struct control_block 20445 | { 20446 | struct st_data 20447 | { 20448 | type_store<variable_t , T > variable_store; 20449 | type_store<function_t , function_t > function_store; 20450 | type_store<vararg_function_t , vararg_function_t > vararg_function_store; 20451 | type_store<generic_function_t, generic_function_t> generic_function_store; 20452 | type_store<generic_function_t, generic_function_t> string_function_store; 20453 | type_store<generic_function_t, generic_function_t> overload_function_store; 20454 | type_store<vector_holder_t , vector_holder_t > vector_store; 20455 | #ifndef exprtk_disable_string_capabilities 20456 | type_store<stringvar_t , std::string > stringvar_store; 20457 | #endif 20458 | 20459 | st_data() 20460 | { 20461 | for (std::size_t i = 0; i < details::reserved_words_size; ++i) 20462 | { 20463 | reserved_symbol_table_.insert(details::reserved_words[i]); 20464 | } 20465 | 20466 | for (std::size_t i = 0; i < details::reserved_symbols_size; ++i) 20467 | { 20468 | reserved_symbol_table_.insert(details::reserved_symbols[i]); 20469 | } 20470 | } 20471 | 20472 | ~st_data() 20473 | { 20474 | for (std::size_t i = 0; i < free_function_list_.size(); ++i) 20475 | { 20476 | delete free_function_list_[i]; 20477 | } 20478 | } 20479 | 20480 | inline bool is_reserved_symbol(const std::string& symbol) const 20481 | { 20482 | return (reserved_symbol_table_.end() != reserved_symbol_table_.find(symbol)); 20483 | } 20484 | 20485 | static inline st_data* create() 20486 | { 20487 | return (new st_data); 20488 | } 20489 | 20490 | static inline void destroy(st_data*& sd) 20491 | { 20492 | delete sd; 20493 | sd = reinterpret_cast<st_data*>(0); 20494 | } 20495 | 20496 | std::list<T> local_symbol_list_; 20497 | std::list<std::string> local_stringvar_list_; 20498 | std::set<std::string> reserved_symbol_table_; 20499 | std::vector<ifunction<T>*> free_function_list_; 20500 | }; 20501 | 20502 | control_block() 20503 | : ref_count(1) 20504 | , data_(st_data::create()) 20505 | , mutability_(e_mutable) 20506 | {} 20507 | 20508 | explicit control_block(st_data* data) 20509 | : ref_count(1) 20510 | , data_(data) 20511 | , mutability_(e_mutable) 20512 | {} 20513 | 20514 | ~control_block() 20515 | { 20516 | if (data_ && (0 == ref_count)) 20517 | { 20518 | st_data::destroy(data_); 20519 | } 20520 | } 20521 | 20522 | static inline control_block* create() 20523 | { 20524 | return (new control_block); 20525 | } 20526 | 20527 | template <typename SymTab> 20528 | static inline void destroy(control_block*& cntrl_blck, SymTab* sym_tab) 20529 | { 20530 | if (cntrl_blck) 20531 | { 20532 | if ( 20533 | (0 != cntrl_blck->ref_count) && 20534 | (0 == --cntrl_blck->ref_count) 20535 | ) 20536 | { 20537 | if (sym_tab) 20538 | sym_tab->clear(); 20539 | 20540 | delete cntrl_blck; 20541 | } 20542 | 20543 | cntrl_blck = 0; 20544 | } 20545 | } 20546 | 20547 | void set_mutability(const symtab_mutability_type mutability) 20548 | { 20549 | mutability_ = mutability; 20550 | } 20551 | 20552 | std::size_t ref_count; 20553 | st_data* data_; 20554 | symtab_mutability_type mutability_; 20555 | }; 20556 | 20557 | public: 20558 | 20559 | explicit symbol_table(const symtab_mutability_type mutability = e_mutable) 20560 | : control_block_(control_block::create()) 20561 | { 20562 | control_block_->set_mutability(mutability); 20563 | clear(); 20564 | } 20565 | 20566 | ~symbol_table() 20567 | { 20568 | exprtk::details::dump_ptr("~symbol_table", this); 20569 | control_block::destroy(control_block_, this); 20570 | } 20571 | 20572 | symbol_table(const symbol_table<T>& st) 20573 | { 20574 | control_block_ = st.control_block_; 20575 | control_block_->ref_count++; 20576 | } 20577 | 20578 | inline symbol_table<T>& operator=(const symbol_table<T>& st) 20579 | { 20580 | if (this != &st) 20581 | { 20582 | control_block::destroy(control_block_,reinterpret_cast<symbol_table<T>*>(0)); 20583 | 20584 | control_block_ = st.control_block_; 20585 | control_block_->ref_count++; 20586 | } 20587 | 20588 | return (*this); 20589 | } 20590 | 20591 | inline bool operator==(const symbol_table<T>& st) const 20592 | { 20593 | return (this == &st) || (control_block_ == st.control_block_); 20594 | } 20595 | 20596 | inline symtab_mutability_type mutability() const 20597 | { 20598 | return valid() ? control_block_->mutability_ : e_unknown; 20599 | } 20600 | 20601 | inline void clear_variables(const bool delete_node = true) 20602 | { 20603 | local_data().variable_store.clear(delete_node); 20604 | } 20605 | 20606 | inline void clear_functions() 20607 | { 20608 | local_data().function_store.clear(); 20609 | } 20610 | 20611 | inline void clear_strings() 20612 | { 20613 | #ifndef exprtk_disable_string_capabilities 20614 | local_data().stringvar_store.clear(); 20615 | #endif 20616 | } 20617 | 20618 | inline void clear_vectors() 20619 | { 20620 | local_data().vector_store.clear(); 20621 | } 20622 | 20623 | inline void clear_local_constants() 20624 | { 20625 | local_data().local_symbol_list_.clear(); 20626 | } 20627 | 20628 | inline void clear() 20629 | { 20630 | if (!valid()) return; 20631 | clear_variables (); 20632 | clear_functions (); 20633 | clear_strings (); 20634 | clear_vectors (); 20635 | clear_local_constants(); 20636 | } 20637 | 20638 | inline std::size_t variable_count() const 20639 | { 20640 | if (valid()) 20641 | return local_data().variable_store.size; 20642 | else 20643 | return 0; 20644 | } 20645 | 20646 | #ifndef exprtk_disable_string_capabilities 20647 | inline std::size_t stringvar_count() const 20648 | { 20649 | if (valid()) 20650 | return local_data().stringvar_store.size; 20651 | else 20652 | return 0; 20653 | } 20654 | #endif 20655 | 20656 | inline std::size_t function_count() const 20657 | { 20658 | if (valid()) 20659 | return local_data().function_store.size; 20660 | else 20661 | return 0; 20662 | } 20663 | 20664 | inline std::size_t vector_count() const 20665 | { 20666 | if (valid()) 20667 | return local_data().vector_store.size; 20668 | else 20669 | return 0; 20670 | } 20671 | 20672 | inline variable_ptr get_variable(const std::string& variable_name) const 20673 | { 20674 | if (!valid()) 20675 | return reinterpret_cast<variable_ptr>(0); 20676 | else if (!valid_symbol(variable_name)) 20677 | return reinterpret_cast<variable_ptr>(0); 20678 | else 20679 | return local_data().variable_store.get(variable_name); 20680 | } 20681 | 20682 | inline variable_ptr get_variable(const T& var_ref) const 20683 | { 20684 | if (!valid()) 20685 | return reinterpret_cast<variable_ptr>(0); 20686 | else 20687 | return local_data().variable_store.get_from_varptr( 20688 | reinterpret_cast<const void*>(&var_ref)); 20689 | } 20690 | 20691 | #ifndef exprtk_disable_string_capabilities 20692 | inline stringvar_ptr get_stringvar(const std::string& string_name) const 20693 | { 20694 | if (!valid()) 20695 | return reinterpret_cast<stringvar_ptr>(0); 20696 | else if (!valid_symbol(string_name)) 20697 | return reinterpret_cast<stringvar_ptr>(0); 20698 | else 20699 | return local_data().stringvar_store.get(string_name); 20700 | } 20701 | 20702 | inline stringvar_base<T> get_stringvar_base(const std::string& string_name) const 20703 | { 20704 | static stringvar_base<T> null_stringvar_base("",reinterpret_cast<stringvar_ptr>(0)); 20705 | if (!valid()) 20706 | return null_stringvar_base; 20707 | else if (!valid_symbol(string_name)) 20708 | return null_stringvar_base; 20709 | 20710 | stringvar_ptr stringvar = local_data().stringvar_store.get(string_name); 20711 | 20712 | if (0 == stringvar) 20713 | { 20714 | return null_stringvar_base; 20715 | } 20716 | 20717 | return stringvar_base<T>(string_name,stringvar); 20718 | } 20719 | #endif 20720 | 20721 | inline function_ptr get_function(const std::string& function_name) const 20722 | { 20723 | if (!valid()) 20724 | return reinterpret_cast<function_ptr>(0); 20725 | else if (!valid_symbol(function_name)) 20726 | return reinterpret_cast<function_ptr>(0); 20727 | else 20728 | return local_data().function_store.get(function_name); 20729 | } 20730 | 20731 | inline vararg_function_ptr get_vararg_function(const std::string& vararg_function_name) const 20732 | { 20733 | if (!valid()) 20734 | return reinterpret_cast<vararg_function_ptr>(0); 20735 | else if (!valid_symbol(vararg_function_name)) 20736 | return reinterpret_cast<vararg_function_ptr>(0); 20737 | else 20738 | return local_data().vararg_function_store.get(vararg_function_name); 20739 | } 20740 | 20741 | inline generic_function_ptr get_generic_function(const std::string& function_name) const 20742 | { 20743 | if (!valid()) 20744 | return reinterpret_cast<generic_function_ptr>(0); 20745 | else if (!valid_symbol(function_name)) 20746 | return reinterpret_cast<generic_function_ptr>(0); 20747 | else 20748 | return local_data().generic_function_store.get(function_name); 20749 | } 20750 | 20751 | inline generic_function_ptr get_string_function(const std::string& function_name) const 20752 | { 20753 | if (!valid()) 20754 | return reinterpret_cast<generic_function_ptr>(0); 20755 | else if (!valid_symbol(function_name)) 20756 | return reinterpret_cast<generic_function_ptr>(0); 20757 | else 20758 | return local_data().string_function_store.get(function_name); 20759 | } 20760 | 20761 | inline generic_function_ptr get_overload_function(const std::string& function_name) const 20762 | { 20763 | if (!valid()) 20764 | return reinterpret_cast<generic_function_ptr>(0); 20765 | else if (!valid_symbol(function_name)) 20766 | return reinterpret_cast<generic_function_ptr>(0); 20767 | else 20768 | return local_data().overload_function_store.get(function_name); 20769 | } 20770 | 20771 | typedef vector_holder_t* vector_holder_ptr; 20772 | 20773 | inline vector_holder_ptr get_vector(const std::string& vector_name) const 20774 | { 20775 | if (!valid()) 20776 | return reinterpret_cast<vector_holder_ptr>(0); 20777 | else if (!valid_symbol(vector_name)) 20778 | return reinterpret_cast<vector_holder_ptr>(0); 20779 | else 20780 | return local_data().vector_store.get(vector_name); 20781 | } 20782 | 20783 | inline T& variable_ref(const std::string& symbol_name) 20784 | { 20785 | static T null_var = T(0); 20786 | if (!valid()) 20787 | return null_var; 20788 | else if (!valid_symbol(symbol_name)) 20789 | return null_var; 20790 | else 20791 | return local_data().variable_store.type_ref(symbol_name); 20792 | } 20793 | 20794 | #ifndef exprtk_disable_string_capabilities 20795 | inline std::string& stringvar_ref(const std::string& symbol_name) 20796 | { 20797 | static std::string null_stringvar; 20798 | if (!valid()) 20799 | return null_stringvar; 20800 | else if (!valid_symbol(symbol_name)) 20801 | return null_stringvar; 20802 | else 20803 | return local_data().stringvar_store.type_ref(symbol_name); 20804 | } 20805 | #endif 20806 | 20807 | inline bool is_constant_node(const std::string& symbol_name) const 20808 | { 20809 | if (!valid()) 20810 | return false; 20811 | else if (!valid_symbol(symbol_name)) 20812 | return false; 20813 | else 20814 | return local_data().variable_store.is_constant(symbol_name); 20815 | } 20816 | 20817 | #ifndef exprtk_disable_string_capabilities 20818 | inline bool is_constant_string(const std::string& symbol_name) const 20819 | { 20820 | if (!valid()) 20821 | return false; 20822 | else if (!valid_symbol(symbol_name)) 20823 | return false; 20824 | else if (!local_data().stringvar_store.symbol_exists(symbol_name)) 20825 | return false; 20826 | else 20827 | return local_data().stringvar_store.is_constant(symbol_name); 20828 | } 20829 | #endif 20830 | 20831 | inline bool create_variable(const std::string& variable_name, const T& value = T(0)) 20832 | { 20833 | if (!valid()) 20834 | return false; 20835 | else if (!valid_symbol(variable_name)) 20836 | return false; 20837 | else if (symbol_exists(variable_name)) 20838 | return false; 20839 | 20840 | local_data().local_symbol_list_.push_back(value); 20841 | T& t = local_data().local_symbol_list_.back(); 20842 | 20843 | return add_variable(variable_name,t); 20844 | } 20845 | 20846 | #ifndef exprtk_disable_string_capabilities 20847 | inline bool create_stringvar(const std::string& stringvar_name, const std::string& value = std::string("")) 20848 | { 20849 | if (!valid()) 20850 | return false; 20851 | else if (!valid_symbol(stringvar_name)) 20852 | return false; 20853 | else if (symbol_exists(stringvar_name)) 20854 | return false; 20855 | 20856 | local_data().local_stringvar_list_.push_back(value); 20857 | std::string& s = local_data().local_stringvar_list_.back(); 20858 | 20859 | return add_stringvar(stringvar_name,s); 20860 | } 20861 | #endif 20862 | 20863 | inline bool add_variable(const std::string& variable_name, T& t, const bool is_constant = false) 20864 | { 20865 | if (!valid()) 20866 | return false; 20867 | else if (!valid_symbol(variable_name)) 20868 | return false; 20869 | else if (symbol_exists(variable_name)) 20870 | return false; 20871 | else 20872 | return local_data().variable_store.add(variable_name, t, is_constant); 20873 | } 20874 | 20875 | inline bool add_constant(const std::string& constant_name, const T& value) 20876 | { 20877 | if (!valid()) 20878 | return false; 20879 | else if (!valid_symbol(constant_name)) 20880 | return false; 20881 | else if (symbol_exists(constant_name)) 20882 | return false; 20883 | 20884 | local_data().local_symbol_list_.push_back(value); 20885 | T& t = local_data().local_symbol_list_.back(); 20886 | 20887 | return add_variable(constant_name, t, true); 20888 | } 20889 | 20890 | #ifndef exprtk_disable_string_capabilities 20891 | inline bool add_stringvar(const std::string& stringvar_name, std::string& s, const bool is_constant = false) 20892 | { 20893 | if (!valid()) 20894 | return false; 20895 | else if (!valid_symbol(stringvar_name)) 20896 | return false; 20897 | else if (symbol_exists(stringvar_name)) 20898 | return false; 20899 | else 20900 | return local_data().stringvar_store.add(stringvar_name, s, is_constant); 20901 | } 20902 | #endif 20903 | 20904 | inline bool add_function(const std::string& function_name, function_t& function) 20905 | { 20906 | if (!valid()) 20907 | return false; 20908 | else if (!valid_symbol(function_name)) 20909 | return false; 20910 | else if (symbol_exists(function_name)) 20911 | return false; 20912 | else 20913 | return local_data().function_store.add(function_name,function); 20914 | } 20915 | 20916 | inline bool add_function(const std::string& vararg_function_name, vararg_function_t& vararg_function) 20917 | { 20918 | if (!valid()) 20919 | return false; 20920 | else if (!valid_symbol(vararg_function_name)) 20921 | return false; 20922 | else if (symbol_exists(vararg_function_name)) 20923 | return false; 20924 | else 20925 | return local_data().vararg_function_store.add(vararg_function_name,vararg_function); 20926 | } 20927 | 20928 | inline bool add_function(const std::string& function_name, generic_function_t& function) 20929 | { 20930 | if (!valid()) 20931 | return false; 20932 | else if (!valid_symbol(function_name)) 20933 | return false; 20934 | else if (symbol_exists(function_name)) 20935 | return false; 20936 | else 20937 | { 20938 | switch (function.rtrn_type) 20939 | { 20940 | case generic_function_t::e_rtrn_scalar : 20941 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|")) ? 20942 | local_data().generic_function_store.add(function_name,function) : false; 20943 | 20944 | case generic_function_t::e_rtrn_string : 20945 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|")) ? 20946 | local_data().string_function_store.add(function_name,function) : false; 20947 | 20948 | case generic_function_t::e_rtrn_overload : 20949 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|:")) ? 20950 | local_data().overload_function_store.add(function_name,function) : false; 20951 | } 20952 | } 20953 | 20954 | return false; 20955 | } 20956 | 20957 | #define exprtk_define_freefunction(NN) \ 20958 | inline bool add_function(const std::string& function_name, ff##NN##_functor function) \ 20959 | { \ 20960 | if (!valid()) \ 20961 | { return false; } \ 20962 | if (!valid_symbol(function_name)) \ 20963 | { return false; } \ 20964 | if (symbol_exists(function_name)) \ 20965 | { return false; } \ 20966 | \ 20967 | exprtk::ifunction<T>* ifunc = new freefunc##NN(function); \ 20968 | \ 20969 | local_data().free_function_list_.push_back(ifunc); \ 20970 | \ 20971 | return add_function(function_name,(*local_data().free_function_list_.back())); \ 20972 | } \ 20973 | 20974 | exprtk_define_freefunction(00) exprtk_define_freefunction(01) 20975 | exprtk_define_freefunction(02) exprtk_define_freefunction(03) 20976 | exprtk_define_freefunction(04) exprtk_define_freefunction(05) 20977 | exprtk_define_freefunction(06) exprtk_define_freefunction(07) 20978 | exprtk_define_freefunction(08) exprtk_define_freefunction(09) 20979 | exprtk_define_freefunction(10) exprtk_define_freefunction(11) 20980 | exprtk_define_freefunction(12) exprtk_define_freefunction(13) 20981 | exprtk_define_freefunction(14) exprtk_define_freefunction(15) 20982 | 20983 | #undef exprtk_define_freefunction 20984 | 20985 | inline bool add_reserved_function(const std::string& function_name, function_t& function) 20986 | { 20987 | if (!valid()) 20988 | return false; 20989 | else if (!valid_symbol(function_name,false)) 20990 | return false; 20991 | else if (symbol_exists(function_name,false)) 20992 | return false; 20993 | else 20994 | return local_data().function_store.add(function_name,function); 20995 | } 20996 | 20997 | inline bool add_reserved_function(const std::string& vararg_function_name, vararg_function_t& vararg_function) 20998 | { 20999 | if (!valid()) 21000 | return false; 21001 | else if (!valid_symbol(vararg_function_name,false)) 21002 | return false; 21003 | else if (symbol_exists(vararg_function_name,false)) 21004 | return false; 21005 | else 21006 | return local_data().vararg_function_store.add(vararg_function_name,vararg_function); 21007 | } 21008 | 21009 | inline bool add_reserved_function(const std::string& function_name, generic_function_t& function) 21010 | { 21011 | if (!valid()) 21012 | return false; 21013 | else if (!valid_symbol(function_name,false)) 21014 | return false; 21015 | else if (symbol_exists(function_name,false)) 21016 | return false; 21017 | else 21018 | { 21019 | switch (function.rtrn_type) 21020 | { 21021 | case generic_function_t::e_rtrn_scalar : 21022 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|")) ? 21023 | local_data().generic_function_store.add(function_name,function) : false; 21024 | 21025 | case generic_function_t::e_rtrn_string : 21026 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|")) ? 21027 | local_data().string_function_store.add(function_name,function) : false; 21028 | 21029 | case generic_function_t::e_rtrn_overload : 21030 | return (std::string::npos == function.parameter_sequence.find_first_not_of("STVZ*?|:")) ? 21031 | local_data().overload_function_store.add(function_name,function) : false; 21032 | } 21033 | } 21034 | 21035 | return false; 21036 | } 21037 | 21038 | #define exprtk_define_reserved_function(NN) \ 21039 | inline bool add_reserved_function(const std::string& function_name, ff##NN##_functor function) \ 21040 | { \ 21041 | if (!valid()) \ 21042 | { return false; } \ 21043 | if (!valid_symbol(function_name,false)) \ 21044 | { return false; } \ 21045 | if (symbol_exists(function_name,false)) \ 21046 | { return false; } \ 21047 | \ 21048 | exprtk::ifunction<T>* ifunc = new freefunc##NN(function); \ 21049 | \ 21050 | local_data().free_function_list_.push_back(ifunc); \ 21051 | \ 21052 | return add_reserved_function(function_name,(*local_data().free_function_list_.back())); \ 21053 | } \ 21054 | 21055 | exprtk_define_reserved_function(00) exprtk_define_reserved_function(01) 21056 | exprtk_define_reserved_function(02) exprtk_define_reserved_function(03) 21057 | exprtk_define_reserved_function(04) exprtk_define_reserved_function(05) 21058 | exprtk_define_reserved_function(06) exprtk_define_reserved_function(07) 21059 | exprtk_define_reserved_function(08) exprtk_define_reserved_function(09) 21060 | exprtk_define_reserved_function(10) exprtk_define_reserved_function(11) 21061 | exprtk_define_reserved_function(12) exprtk_define_reserved_function(13) 21062 | exprtk_define_reserved_function(14) exprtk_define_reserved_function(15) 21063 | 21064 | #undef exprtk_define_reserved_function 21065 | 21066 | template <std::size_t N> 21067 | inline bool add_vector(const std::string& vector_name, T (&v)[N]) 21068 | { 21069 | if (!valid()) 21070 | return false; 21071 | else if (!valid_symbol(vector_name)) 21072 | return false; 21073 | else if (symbol_exists(vector_name)) 21074 | return false; 21075 | else 21076 | return local_data().vector_store.add(vector_name,v); 21077 | } 21078 | 21079 | inline bool add_vector(const std::string& vector_name, T* v, const std::size_t& v_size) 21080 | { 21081 | if (!valid()) 21082 | return false; 21083 | else if (!valid_symbol(vector_name)) 21084 | return false; 21085 | else if (symbol_exists(vector_name)) 21086 | return false; 21087 | else if (0 == v_size) 21088 | return false; 21089 | else 21090 | return local_data().vector_store.add(vector_name, v, v_size); 21091 | } 21092 | 21093 | template <typename Allocator> 21094 | inline bool add_vector(const std::string& vector_name, std::vector<T,Allocator>& v) 21095 | { 21096 | if (!valid()) 21097 | return false; 21098 | else if (!valid_symbol(vector_name)) 21099 | return false; 21100 | else if (symbol_exists(vector_name)) 21101 | return false; 21102 | else if (0 == v.size()) 21103 | return false; 21104 | else 21105 | return local_data().vector_store.add(vector_name,v); 21106 | } 21107 | 21108 | inline bool add_vector(const std::string& vector_name, exprtk::vector_view<T>& v) 21109 | { 21110 | if (!valid()) 21111 | return false; 21112 | else if (!valid_symbol(vector_name)) 21113 | return false; 21114 | else if (symbol_exists(vector_name)) 21115 | return false; 21116 | else if (0 == v.size()) 21117 | return false; 21118 | else 21119 | return local_data().vector_store.add(vector_name,v); 21120 | } 21121 | 21122 | inline bool remove_variable(const std::string& variable_name, const bool delete_node = true) 21123 | { 21124 | if (!valid()) 21125 | return false; 21126 | else 21127 | return local_data().variable_store.remove(variable_name, delete_node); 21128 | } 21129 | 21130 | #ifndef exprtk_disable_string_capabilities 21131 | inline bool remove_stringvar(const std::string& string_name) 21132 | { 21133 | if (!valid()) 21134 | return false; 21135 | else 21136 | return local_data().stringvar_store.remove(string_name); 21137 | } 21138 | #endif 21139 | 21140 | inline bool remove_function(const std::string& function_name) 21141 | { 21142 | if (!valid()) 21143 | return false; 21144 | else 21145 | return local_data().function_store.remove(function_name); 21146 | } 21147 | 21148 | inline bool remove_vararg_function(const std::string& vararg_function_name) 21149 | { 21150 | if (!valid()) 21151 | return false; 21152 | else 21153 | return local_data().vararg_function_store.remove(vararg_function_name); 21154 | } 21155 | 21156 | inline bool remove_vector(const std::string& vector_name) 21157 | { 21158 | if (!valid()) 21159 | return false; 21160 | else 21161 | return local_data().vector_store.remove(vector_name); 21162 | } 21163 | 21164 | inline bool add_constants() 21165 | { 21166 | return add_pi () && 21167 | add_epsilon () && 21168 | add_infinity() ; 21169 | } 21170 | 21171 | inline bool add_pi() 21172 | { 21173 | const typename details::numeric::details::number_type<T>::type num_type; 21174 | static const T local_pi = details::numeric::details::const_pi_impl<T>(num_type); 21175 | return add_constant("pi",local_pi); 21176 | } 21177 | 21178 | inline bool add_epsilon() 21179 | { 21180 | static const T local_epsilon = details::numeric::details::epsilon_type<T>::value(); 21181 | return add_constant("epsilon",local_epsilon); 21182 | } 21183 | 21184 | inline bool add_infinity() 21185 | { 21186 | static const T local_infinity = std::numeric_limits<T>::infinity(); 21187 | return add_constant("inf",local_infinity); 21188 | } 21189 | 21190 | template <typename Package> 21191 | inline bool add_package(Package& package) 21192 | { 21193 | return package.register_package(*this); 21194 | } 21195 | 21196 | template <typename Allocator, 21197 | template <typename, typename> class Sequence> 21198 | inline std::size_t get_variable_list(Sequence<std::pair<std::string,T>,Allocator>& vlist) const 21199 | { 21200 | if (!valid()) 21201 | return 0; 21202 | else 21203 | return local_data().variable_store.get_list(vlist); 21204 | } 21205 | 21206 | template <typename Allocator, 21207 | template <typename, typename> class Sequence> 21208 | inline std::size_t get_variable_list(Sequence<std::string,Allocator>& vlist) const 21209 | { 21210 | if (!valid()) 21211 | return 0; 21212 | else 21213 | return local_data().variable_store.get_list(vlist); 21214 | } 21215 | 21216 | #ifndef exprtk_disable_string_capabilities 21217 | template <typename Allocator, 21218 | template <typename, typename> class Sequence> 21219 | inline std::size_t get_stringvar_list(Sequence<std::pair<std::string,std::string>,Allocator>& svlist) const 21220 | { 21221 | if (!valid()) 21222 | return 0; 21223 | else 21224 | return local_data().stringvar_store.get_list(svlist); 21225 | } 21226 | 21227 | template <typename Allocator, 21228 | template <typename, typename> class Sequence> 21229 | inline std::size_t get_stringvar_list(Sequence<std::string,Allocator>& svlist) const 21230 | { 21231 | if (!valid()) 21232 | return 0; 21233 | else 21234 | return local_data().stringvar_store.get_list(svlist); 21235 | } 21236 | #endif 21237 | 21238 | template <typename Allocator, 21239 | template <typename, typename> class Sequence> 21240 | inline std::size_t get_vector_list(Sequence<std::string,Allocator>& vec_list) const 21241 | { 21242 | if (!valid()) 21243 | return 0; 21244 | else 21245 | return local_data().vector_store.get_list(vec_list); 21246 | } 21247 | 21248 | template <typename Allocator, 21249 | template <typename, typename> class Sequence> 21250 | inline std::size_t get_function_list(Sequence<std::string,Allocator>& function_list) const 21251 | { 21252 | if (!valid()) 21253 | return 0; 21254 | 21255 | std::vector<std::string> function_names; 21256 | std::size_t count = 0; 21257 | 21258 | count += local_data().function_store .get_list(function_names); 21259 | count += local_data().vararg_function_store .get_list(function_names); 21260 | count += local_data().generic_function_store .get_list(function_names); 21261 | count += local_data().string_function_store .get_list(function_names); 21262 | count += local_data().overload_function_store.get_list(function_names); 21263 | 21264 | std::set<std::string> function_set; 21265 | 21266 | for (std::size_t i = 0; i < function_names.size(); ++i) 21267 | { 21268 | function_set.insert(function_names[i]); 21269 | } 21270 | 21271 | std::copy(function_set.begin(), function_set.end(), 21272 | std::back_inserter(function_list)); 21273 | 21274 | return count; 21275 | } 21276 | 21277 | inline std::vector<std::string> get_function_list() const 21278 | { 21279 | std::vector<std::string> result; 21280 | get_function_list(result); 21281 | return result; 21282 | } 21283 | 21284 | inline bool symbol_exists(const std::string& symbol_name, const bool check_reserved_symb = true) const 21285 | { 21286 | /* 21287 | Function will return true if symbol_name exists as either a 21288 | reserved symbol, variable, stringvar, vector or function name 21289 | in any of the type stores. 21290 | */ 21291 | if (!valid()) 21292 | return false; 21293 | else if (local_data().variable_store.symbol_exists(symbol_name)) 21294 | return true; 21295 | #ifndef exprtk_disable_string_capabilities 21296 | else if (local_data().stringvar_store.symbol_exists(symbol_name)) 21297 | return true; 21298 | #endif 21299 | else if (local_data().vector_store.symbol_exists(symbol_name)) 21300 | return true; 21301 | else if (local_data().function_store.symbol_exists(symbol_name)) 21302 | return true; 21303 | else if (check_reserved_symb && local_data().is_reserved_symbol(symbol_name)) 21304 | return true; 21305 | else 21306 | return false; 21307 | } 21308 | 21309 | inline bool is_variable(const std::string& variable_name) const 21310 | { 21311 | if (!valid()) 21312 | return false; 21313 | else 21314 | return local_data().variable_store.symbol_exists(variable_name); 21315 | } 21316 | 21317 | #ifndef exprtk_disable_string_capabilities 21318 | inline bool is_stringvar(const std::string& stringvar_name) const 21319 | { 21320 | if (!valid()) 21321 | return false; 21322 | else 21323 | return local_data().stringvar_store.symbol_exists(stringvar_name); 21324 | } 21325 | 21326 | inline bool is_conststr_stringvar(const std::string& symbol_name) const 21327 | { 21328 | if (!valid()) 21329 | return false; 21330 | else if (!valid_symbol(symbol_name)) 21331 | return false; 21332 | else if (!local_data().stringvar_store.symbol_exists(symbol_name)) 21333 | return false; 21334 | 21335 | return ( 21336 | local_data().stringvar_store.symbol_exists(symbol_name) || 21337 | local_data().stringvar_store.is_constant (symbol_name) 21338 | ); 21339 | } 21340 | #endif 21341 | 21342 | inline bool is_function(const std::string& function_name) const 21343 | { 21344 | if (!valid()) 21345 | return false; 21346 | else 21347 | return local_data().function_store.symbol_exists(function_name); 21348 | } 21349 | 21350 | inline bool is_vararg_function(const std::string& vararg_function_name) const 21351 | { 21352 | if (!valid()) 21353 | return false; 21354 | else 21355 | return local_data().vararg_function_store.symbol_exists(vararg_function_name); 21356 | } 21357 | 21358 | inline bool is_vector(const std::string& vector_name) const 21359 | { 21360 | if (!valid()) 21361 | return false; 21362 | else 21363 | return local_data().vector_store.symbol_exists(vector_name); 21364 | } 21365 | 21366 | inline std::string get_variable_name(const expression_ptr& ptr) const 21367 | { 21368 | return local_data().variable_store.entity_name(ptr); 21369 | } 21370 | 21371 | inline std::string get_vector_name(const vector_holder_ptr& ptr) const 21372 | { 21373 | return local_data().vector_store.entity_name(ptr); 21374 | } 21375 | 21376 | #ifndef exprtk_disable_string_capabilities 21377 | inline std::string get_stringvar_name(const expression_ptr& ptr) const 21378 | { 21379 | return local_data().stringvar_store.entity_name(ptr); 21380 | } 21381 | 21382 | inline std::string get_conststr_stringvar_name(const expression_ptr& ptr) const 21383 | { 21384 | return local_data().stringvar_store.entity_name(ptr); 21385 | } 21386 | #endif 21387 | 21388 | inline bool valid() const 21389 | { 21390 | // Symbol table sanity check. 21391 | return control_block_ && control_block_->data_; 21392 | } 21393 | 21394 | inline void load_from(const symbol_table<T>& st) 21395 | { 21396 | { 21397 | std::vector<std::string> name_list; 21398 | 21399 | st.local_data().function_store.get_list(name_list); 21400 | 21401 | if (!name_list.empty()) 21402 | { 21403 | for (std::size_t i = 0; i < name_list.size(); ++i) 21404 | { 21405 | exprtk::ifunction<T>& ifunc = *st.get_function(name_list[i]); 21406 | add_function(name_list[i],ifunc); 21407 | } 21408 | } 21409 | } 21410 | 21411 | { 21412 | std::vector<std::string> name_list; 21413 | 21414 | st.local_data().vararg_function_store.get_list(name_list); 21415 | 21416 | if (!name_list.empty()) 21417 | { 21418 | for (std::size_t i = 0; i < name_list.size(); ++i) 21419 | { 21420 | exprtk::ivararg_function<T>& ivafunc = *st.get_vararg_function(name_list[i]); 21421 | add_function(name_list[i],ivafunc); 21422 | } 21423 | } 21424 | } 21425 | 21426 | { 21427 | std::vector<std::string> name_list; 21428 | 21429 | st.local_data().generic_function_store.get_list(name_list); 21430 | 21431 | if (!name_list.empty()) 21432 | { 21433 | for (std::size_t i = 0; i < name_list.size(); ++i) 21434 | { 21435 | exprtk::igeneric_function<T>& ifunc = *st.get_generic_function(name_list[i]); 21436 | add_function(name_list[i],ifunc); 21437 | } 21438 | } 21439 | } 21440 | 21441 | { 21442 | std::vector<std::string> name_list; 21443 | 21444 | st.local_data().string_function_store.get_list(name_list); 21445 | 21446 | if (!name_list.empty()) 21447 | { 21448 | for (std::size_t i = 0; i < name_list.size(); ++i) 21449 | { 21450 | exprtk::igeneric_function<T>& ifunc = *st.get_string_function(name_list[i]); 21451 | add_function(name_list[i],ifunc); 21452 | } 21453 | } 21454 | } 21455 | 21456 | { 21457 | std::vector<std::string> name_list; 21458 | 21459 | st.local_data().overload_function_store.get_list(name_list); 21460 | 21461 | if (!name_list.empty()) 21462 | { 21463 | for (std::size_t i = 0; i < name_list.size(); ++i) 21464 | { 21465 | exprtk::igeneric_function<T>& ifunc = *st.get_overload_function(name_list[i]); 21466 | add_function(name_list[i],ifunc); 21467 | } 21468 | } 21469 | } 21470 | } 21471 | 21472 | inline void load_variables_from(const symbol_table<T>& st) 21473 | { 21474 | std::vector<std::string> name_list; 21475 | 21476 | st.local_data().variable_store.get_list(name_list); 21477 | 21478 | if (!name_list.empty()) 21479 | { 21480 | for (std::size_t i = 0; i < name_list.size(); ++i) 21481 | { 21482 | T& variable = st.get_variable(name_list[i])->ref(); 21483 | add_variable(name_list[i], variable); 21484 | } 21485 | } 21486 | } 21487 | 21488 | inline void load_vectors_from(const symbol_table<T>& st) 21489 | { 21490 | std::vector<std::string> name_list; 21491 | 21492 | st.local_data().vector_store.get_list(name_list); 21493 | 21494 | if (!name_list.empty()) 21495 | { 21496 | for (std::size_t i = 0; i < name_list.size(); ++i) 21497 | { 21498 | vector_holder_t& vecholder = *st.get_vector(name_list[i]); 21499 | add_vector(name_list[i], vecholder.data(), vecholder.size()); 21500 | } 21501 | } 21502 | } 21503 | 21504 | private: 21505 | 21506 | inline bool valid_symbol(const std::string& symbol, const bool check_reserved_symb = true) const 21507 | { 21508 | if (symbol.empty()) 21509 | return false; 21510 | else if (!details::is_letter(symbol[0])) 21511 | return false; 21512 | else if (symbol.size() > 1) 21513 | { 21514 | for (std::size_t i = 1; i < symbol.size(); ++i) 21515 | { 21516 | if ( 21517 | !details::is_letter_or_digit(symbol[i]) && 21518 | ('_' != symbol[i]) 21519 | ) 21520 | { 21521 | if ((i < (symbol.size() - 1)) && ('.' == symbol[i])) 21522 | continue; 21523 | else 21524 | return false; 21525 | } 21526 | } 21527 | } 21528 | 21529 | return (check_reserved_symb) ? (!local_data().is_reserved_symbol(symbol)) : true; 21530 | } 21531 | 21532 | inline bool valid_function(const std::string& symbol) const 21533 | { 21534 | if (symbol.empty()) 21535 | return false; 21536 | else if (!details::is_letter(symbol[0])) 21537 | return false; 21538 | else if (symbol.size() > 1) 21539 | { 21540 | for (std::size_t i = 1; i < symbol.size(); ++i) 21541 | { 21542 | if ( 21543 | !details::is_letter_or_digit(symbol[i]) && 21544 | ('_' != symbol[i]) 21545 | ) 21546 | { 21547 | if ((i < (symbol.size() - 1)) && ('.' == symbol[i])) 21548 | continue; 21549 | else 21550 | return false; 21551 | } 21552 | } 21553 | } 21554 | 21555 | return true; 21556 | } 21557 | 21558 | typedef typename control_block::st_data local_data_t; 21559 | 21560 | inline local_data_t& local_data() 21561 | { 21562 | return *(control_block_->data_); 21563 | } 21564 | 21565 | inline const local_data_t& local_data() const 21566 | { 21567 | return *(control_block_->data_); 21568 | } 21569 | 21570 | control_block* control_block_; 21571 | 21572 | friend class parser<T>; 21573 | }; // class symbol_table 21574 | 21575 | template <typename T> 21576 | class function_compositor; 21577 | 21578 | template <typename T> 21579 | class expression 21580 | { 21581 | private: 21582 | 21583 | typedef details::expression_node<T>* expression_ptr; 21584 | typedef details::vector_holder<T>* vector_holder_ptr; 21585 | typedef std::vector<symbol_table<T> > symtab_list_t; 21586 | 21587 | struct control_block 21588 | { 21589 | enum data_type 21590 | { 21591 | e_unknown , 21592 | e_expr , 21593 | e_vecholder, 21594 | e_data , 21595 | e_vecdata , 21596 | e_string 21597 | }; 21598 | 21599 | static std::string to_str(data_type dt) 21600 | { 21601 | switch (dt) 21602 | { 21603 | case e_unknown : return "e_unknown " 21604 | case e_expr : return "e_expr" ; 21605 | case e_vecholder : return "e_vecholder" 21606 | case e_data : return "e_data" ; 21607 | case e_vecdata : return "e_vecdata" ; 21608 | case e_string : return "e_string" ; 21609 | } 21610 | 21611 | return "" 21612 | } 21613 | 21614 | struct data_pack 21615 | { 21616 | data_pack() 21617 | : pointer(0) 21618 | , type(e_unknown) 21619 | , size(0) 21620 | {} 21621 | 21622 | data_pack(void* ptr, const data_type dt, const std::size_t sz = 0) 21623 | : pointer(ptr) 21624 | , type(dt) 21625 | , size(sz) 21626 | {} 21627 | 21628 | void* pointer; 21629 | data_type type; 21630 | std::size_t size; 21631 | }; 21632 | 21633 | typedef std::vector<data_pack> local_data_list_t; 21634 | typedef results_context<T> results_context_t; 21635 | typedef control_block* cntrl_blck_ptr_t; 21636 | 21637 | control_block() 21638 | : ref_count(0) 21639 | , expr (0) 21640 | , results (0) 21641 | , retinv_null(false) 21642 | , return_invoked(&retinv_null) 21643 | {} 21644 | 21645 | explicit control_block(expression_ptr e) 21646 | : ref_count(1) 21647 | , expr (e) 21648 | , results (0) 21649 | , retinv_null(false) 21650 | , return_invoked(&retinv_null) 21651 | {} 21652 | 21653 | ~control_block() 21654 | { 21655 | if (expr && details::branch_deletable(expr)) 21656 | { 21657 | destroy_node(expr); 21658 | } 21659 | 21660 | if (!local_data_list.empty()) 21661 | { 21662 | for (std::size_t i = 0; i < local_data_list.size(); ++i) 21663 | { 21664 | switch (local_data_list[i].type) 21665 | { 21666 | case e_expr : delete reinterpret_cast<expression_ptr>(local_data_list[i].pointer); 21667 | break; 21668 | 21669 | case e_vecholder : delete reinterpret_cast<vector_holder_ptr>(local_data_list[i].pointer); 21670 | break; 21671 | 21672 | case e_data : delete reinterpret_cast<T*>(local_data_list[i].pointer); 21673 | break; 21674 | 21675 | case e_vecdata : delete [] reinterpret_cast<T*>(local_data_list[i].pointer); 21676 | break; 21677 | 21678 | case e_string : delete reinterpret_cast<std::string*>(local_data_list[i].pointer); 21679 | break; 21680 | 21681 | default : break; 21682 | } 21683 | } 21684 | } 21685 | 21686 | if (results) 21687 | { 21688 | delete results; 21689 | } 21690 | } 21691 | 21692 | static inline cntrl_blck_ptr_t create(expression_ptr e) 21693 | { 21694 | return new control_block(e); 21695 | } 21696 | 21697 | static inline void destroy(cntrl_blck_ptr_t& cntrl_blck) 21698 | { 21699 | if (cntrl_blck) 21700 | { 21701 | if ( 21702 | (0 != cntrl_blck->ref_count) && 21703 | (0 == --cntrl_blck->ref_count) 21704 | ) 21705 | { 21706 | delete cntrl_blck; 21707 | } 21708 | 21709 | cntrl_blck = 0; 21710 | } 21711 | } 21712 | 21713 | std::size_t ref_count; 21714 | expression_ptr expr; 21715 | local_data_list_t local_data_list; 21716 | results_context_t* results; 21717 | bool retinv_null; 21718 | bool* return_invoked; 21719 | 21720 | friend class function_compositor<T>; 21721 | }; 21722 | 21723 | public: 21724 | 21725 | expression() 21726 | : control_block_(0) 21727 | { 21728 | set_expression(new details::null_node<T>()); 21729 | } 21730 | 21731 | expression(const expression<T>& e) 21732 | : control_block_ (e.control_block_ ) 21733 | , symbol_table_list_(e.symbol_table_list_) 21734 | { 21735 | control_block_->ref_count++; 21736 | } 21737 | 21738 | explicit expression(const symbol_table<T>& symbol_table) 21739 | : control_block_(0) 21740 | { 21741 | set_expression(new details::null_node<T>()); 21742 | symbol_table_list_.push_back(symbol_table); 21743 | } 21744 | 21745 | inline expression<T>& operator=(const expression<T>& e) 21746 | { 21747 | if (this != &e) 21748 | { 21749 | if (control_block_) 21750 | { 21751 | if ( 21752 | (0 != control_block_->ref_count) && 21753 | (0 == --control_block_->ref_count) 21754 | ) 21755 | { 21756 | delete control_block_; 21757 | } 21758 | 21759 | control_block_ = 0; 21760 | } 21761 | 21762 | control_block_ = e.control_block_; 21763 | control_block_->ref_count++; 21764 | symbol_table_list_ = e.symbol_table_list_; 21765 | } 21766 | 21767 | return *this; 21768 | } 21769 | 21770 | inline bool operator==(const expression<T>& e) const 21771 | { 21772 | return (this == &e); 21773 | } 21774 | 21775 | inline bool operator!() const 21776 | { 21777 | return ( 21778 | (0 == control_block_ ) || 21779 | (0 == control_block_->expr) 21780 | ); 21781 | } 21782 | 21783 | inline expression<T>& release() 21784 | { 21785 | exprtk::details::dump_ptr("expression::release", this); 21786 | control_block::destroy(control_block_); 21787 | 21788 | return (*this); 21789 | } 21790 | 21791 | ~expression() 21792 | { 21793 | control_block::destroy(control_block_); 21794 | } 21795 | 21796 | inline T value() const 21797 | { 21798 | assert(control_block_ ); 21799 | assert(control_block_->expr); 21800 | 21801 | return control_block_->expr->value(); 21802 | } 21803 | 21804 | inline T operator() () const 21805 | { 21806 | return value(); 21807 | } 21808 | 21809 | inline operator T() const 21810 | { 21811 | return value(); 21812 | } 21813 | 21814 | inline operator bool() const 21815 | { 21816 | return details::is_true(value()); 21817 | } 21818 | 21819 | inline bool register_symbol_table(symbol_table<T>& st) 21820 | { 21821 | for (std::size_t i = 0; i < symbol_table_list_.size(); ++i) 21822 | { 21823 | if (st == symbol_table_list_[i]) 21824 | { 21825 | return false; 21826 | } 21827 | } 21828 | 21829 | symbol_table_list_.push_back(st); 21830 | return true; 21831 | } 21832 | 21833 | inline const symbol_table<T>& get_symbol_table(const std::size_t& index = 0) const 21834 | { 21835 | return symbol_table_list_[index]; 21836 | } 21837 | 21838 | inline symbol_table<T>& get_symbol_table(const std::size_t& index = 0) 21839 | { 21840 | return symbol_table_list_[index]; 21841 | } 21842 | 21843 | std::size_t num_symbol_tables() const 21844 | { 21845 | return symbol_table_list_.size(); 21846 | } 21847 | 21848 | typedef results_context<T> results_context_t; 21849 | 21850 | inline const results_context_t& results() const 21851 | { 21852 | if (control_block_->results) 21853 | return (*control_block_->results); 21854 | else 21855 | { 21856 | static const results_context_t null_results; 21857 | return null_results; 21858 | } 21859 | } 21860 | 21861 | inline bool return_invoked() const 21862 | { 21863 | return (*control_block_->return_invoked); 21864 | } 21865 | 21866 | private: 21867 | 21868 | inline symtab_list_t get_symbol_table_list() const 21869 | { 21870 | return symbol_table_list_; 21871 | } 21872 | 21873 | inline void set_expression(const expression_ptr expr) 21874 | { 21875 | if (expr) 21876 | { 21877 | if (control_block_) 21878 | { 21879 | if (0 == --control_block_->ref_count) 21880 | { 21881 | delete control_block_; 21882 | } 21883 | } 21884 | 21885 | control_block_ = control_block::create(expr); 21886 | } 21887 | } 21888 | 21889 | inline void register_local_var(expression_ptr expr) 21890 | { 21891 | if (expr) 21892 | { 21893 | if (control_block_) 21894 | { 21895 | control_block_-> 21896 | local_data_list.push_back( 21897 | typename expression<T>::control_block:: 21898 | data_pack(reinterpret_cast<void*>(expr), 21899 | control_block::e_expr)); 21900 | } 21901 | } 21902 | } 21903 | 21904 | inline void register_local_var(vector_holder_ptr vec_holder) 21905 | { 21906 | if (vec_holder) 21907 | { 21908 | if (control_block_) 21909 | { 21910 | control_block_-> 21911 | local_data_list.push_back( 21912 | typename expression<T>::control_block:: 21913 | data_pack(reinterpret_cast<void*>(vec_holder), 21914 | control_block::e_vecholder)); 21915 | } 21916 | } 21917 | } 21918 | 21919 | inline void register_local_data(void* data, const std::size_t& size = 0, const std::size_t data_mode = 0) 21920 | { 21921 | if (data) 21922 | { 21923 | if (control_block_) 21924 | { 21925 | typename control_block::data_type dt = control_block::e_data; 21926 | 21927 | switch (data_mode) 21928 | { 21929 | case 0 : dt = control_block::e_data; break; 21930 | case 1 : dt = control_block::e_vecdata; break; 21931 | case 2 : dt = control_block::e_string; break; 21932 | } 21933 | 21934 | control_block_-> 21935 | local_data_list.push_back( 21936 | typename expression<T>::control_block:: 21937 | data_pack(reinterpret_cast<void*>(data), dt, size)); 21938 | } 21939 | } 21940 | } 21941 | 21942 | inline const typename control_block::local_data_list_t& local_data_list() 21943 | { 21944 | if (control_block_) 21945 | { 21946 | return control_block_->local_data_list; 21947 | } 21948 | else 21949 | { 21950 | static typename control_block::local_data_list_t null_local_data_list; 21951 | return null_local_data_list; 21952 | } 21953 | } 21954 | 21955 | inline void register_return_results(results_context_t* rc) 21956 | { 21957 | if (control_block_ && rc) 21958 | { 21959 | control_block_->results = rc; 21960 | } 21961 | } 21962 | 21963 | inline void set_retinvk(bool* retinvk_ptr) 21964 | { 21965 | if (control_block_) 21966 | { 21967 | control_block_->return_invoked = retinvk_ptr; 21968 | } 21969 | } 21970 | 21971 | control_block* control_block_; 21972 | symtab_list_t symbol_table_list_; 21973 | 21974 | friend class parser<T>; 21975 | friend class expression_helper<T>; 21976 | friend class function_compositor<T>; 21977 | template <typename TT> 21978 | friend bool is_valid(const expression<TT>& expr); 21979 | }; // class expression 21980 | 21981 | template <typename T> 21982 | class expression_helper 21983 | { 21984 | public: 21985 | 21986 | enum node_types 21987 | { 21988 | e_literal, 21989 | e_variable, 21990 | e_string, 21991 | e_unary, 21992 | e_binary, 21993 | e_function, 21994 | e_vararg, 21995 | e_null, 21996 | e_assert, 21997 | e_sf3ext, 21998 | e_sf4ext 21999 | }; 22000 | 22001 | static inline bool is_literal(const expression<T>& expr) 22002 | { 22003 | return expr.control_block_ && details::is_literal_node(expr.control_block_->expr); 22004 | } 22005 | 22006 | static inline bool is_variable(const expression<T>& expr) 22007 | { 22008 | return expr.control_block_ && details::is_variable_node(expr.control_block_->expr); 22009 | } 22010 | 22011 | static inline bool is_string(const expression<T>& expr) 22012 | { 22013 | return expr.control_block_ && details::is_generally_string_node(expr.control_block_->expr); 22014 | } 22015 | 22016 | static inline bool is_unary(const expression<T>& expr) 22017 | { 22018 | return expr.control_block_ && details::is_unary_node(expr.control_block_->expr); 22019 | } 22020 | 22021 | static inline bool is_binary(const expression<T>& expr) 22022 | { 22023 | return expr.control_block_ && details::is_binary_node(expr.control_block_->expr); 22024 | } 22025 | 22026 | static inline bool is_function(const expression<T>& expr) 22027 | { 22028 | return expr.control_block_ && details::is_function(expr.control_block_->expr); 22029 | } 22030 | 22031 | static inline bool is_vararg(const expression<T>& expr) 22032 | { 22033 | return expr.control_block_ && details::is_vararg_node(expr.control_block_->expr); 22034 | } 22035 | 22036 | static inline bool is_null(const expression<T>& expr) 22037 | { 22038 | return expr.control_block_ && details::is_null_node(expr.control_block_->expr); 22039 | } 22040 | 22041 | static inline bool is_assert(const expression<T>& expr) 22042 | { 22043 | return expr.control_block_ && details::is_assert_node(expr.control_block_->expr); 22044 | } 22045 | 22046 | static inline bool is_sf3ext(const expression<T>& expr) 22047 | { 22048 | return expr.control_block_ && details::is_sf3ext_node(expr.control_block_->expr); 22049 | } 22050 | 22051 | static inline bool is_sf4ext(const expression<T>& expr) 22052 | { 22053 | return expr.control_block_ && details::is_sf4ext_node(expr.control_block_->expr); 22054 | } 22055 | 22056 | static inline bool is_type(const expression<T>& expr, const node_types node_type) 22057 | { 22058 | if (0 == expr.control_block_) 22059 | { 22060 | return false; 22061 | } 22062 | 22063 | switch (node_type) 22064 | { 22065 | case e_literal : return is_literal_node(expr); 22066 | case e_variable : return is_variable (expr); 22067 | case e_string : return is_string (expr); 22068 | case e_unary : return is_unary (expr); 22069 | case e_binary : return is_binary (expr); 22070 | case e_function : return is_function (expr); 22071 | case e_null : return is_null (expr); 22072 | case e_assert : return is_assert (expr); 22073 | case e_sf3ext : return is_sf3ext (expr); 22074 | case e_sf4ext : return is_sf4ext (expr); 22075 | }; 22076 | 22077 | return false; 22078 | } 22079 | 22080 | static inline bool match_type_sequence(const expression<T>& expr, const std::vector<node_types>& type_seq) 22081 | { 22082 | if ((0 == expr.control_block_) || !is_vararg(expr)) 22083 | { 22084 | return false; 22085 | } 22086 | 22087 | typedef details::vararg_node<T, exprtk::details::vararg_multi_op<T> > mo_vararg_t; 22088 | 22089 | mo_vararg_t* vnode = dynamic_cast<mo_vararg_t*>(expr.control_block_->expr); 22090 | 22091 | if ( 22092 | (0 == vnode) || 22093 | type_seq.empty() || 22094 | (vnode->size() < type_seq.size()) 22095 | ) 22096 | { 22097 | return false; 22098 | } 22099 | 22100 | for (std::size_t i = 0; i < type_seq.size(); ++i) 22101 | { 22102 | assert((*vnode)[i]); 22103 | 22104 | switch (type_seq[i]) 22105 | { 22106 | case e_literal : { if (details::is_literal_node ((*vnode)[i])) continue; } break; 22107 | case e_variable : { if (details::is_variable_node ((*vnode)[i])) continue; } break; 22108 | case e_string : { if (details::is_generally_string_node((*vnode)[i])) continue; } break; 22109 | case e_unary : { if (details::is_unary_node ((*vnode)[i])) continue; } break; 22110 | case e_binary : { if (details::is_binary_node ((*vnode)[i])) continue; } break; 22111 | case e_function : { if (details::is_function ((*vnode)[i])) continue; } break; 22112 | case e_null : { if (details::is_null_node ((*vnode)[i])) continue; } break; 22113 | case e_assert : { if (details::is_assert_node ((*vnode)[i])) continue; } break; 22114 | case e_sf3ext : { if (details::is_sf3ext_node ((*vnode)[i])) continue; } break; 22115 | case e_sf4ext : { if (details::is_sf4ext_node ((*vnode)[i])) continue; } break; 22116 | case e_vararg : break; 22117 | } 22118 | 22119 | return false; 22120 | } 22121 | 22122 | return true; 22123 | } 22124 | }; 22125 | 22126 | template <typename T> 22127 | inline bool is_valid(const expression<T>& expr) 22128 | { 22129 | return expr.control_block_ && !expression_helper<T>::is_null(expr); 22130 | } 22131 | 22132 | namespace parser_error 22133 | { 22134 | enum error_mode 22135 | { 22136 | e_unknown = 0, 22137 | e_syntax = 1, 22138 | e_token = 2, 22139 | e_numeric = 4, 22140 | e_symtab = 5, 22141 | e_lexer = 6, 22142 | e_synthesis = 7, 22143 | e_helper = 8, 22144 | e_parser = 9 22145 | }; 22146 | 22147 | struct type 22148 | { 22149 | type() 22150 | : mode(parser_error::e_unknown) 22151 | , line_no (0) 22152 | , column_no(0) 22153 | {} 22154 | 22155 | lexer::token token; 22156 | error_mode mode; 22157 | std::string diagnostic; 22158 | std::string src_location; 22159 | std::string error_line; 22160 | std::size_t line_no; 22161 | std::size_t column_no; 22162 | }; 22163 | 22164 | inline type make_error(const error_mode mode, 22165 | const std::string& diagnostic = "", 22166 | const std::string& src_location = "") 22167 | { 22168 | type t; 22169 | t.mode = mode; 22170 | t.token.type = lexer::token::e_error; 22171 | t.diagnostic = diagnostic; 22172 | t.src_location = src_location; 22173 | exprtk_debug(("%s\n", diagnostic .c_str())); 22174 | return t; 22175 | } 22176 | 22177 | inline type make_error(const error_mode mode, 22178 | const lexer::token& tk, 22179 | const std::string& diagnostic = "", 22180 | const std::string& src_location = "") 22181 | { 22182 | type t; 22183 | t.mode = mode; 22184 | t.token = tk; 22185 | t.diagnostic = diagnostic; 22186 | t.src_location = src_location; 22187 | exprtk_debug(("%s\n", diagnostic .c_str())); 22188 | return t; 22189 | } 22190 | 22191 | inline std::string to_str(error_mode mode) 22192 | { 22193 | switch (mode) 22194 | { 22195 | case e_unknown : return std::string("Unknown Error"); 22196 | case e_syntax : return std::string("Syntax Error" ); 22197 | case e_token : return std::string("Token Error" ); 22198 | case e_numeric : return std::string("Numeric Error"); 22199 | case e_symtab : return std::string("Symbol Error" ); 22200 | case e_lexer : return std::string("Lexer Error" ); 22201 | case e_helper : return std::string("Helper Error" ); 22202 | case e_parser : return std::string("Parser Error" ); 22203 | default : return std::string("Unknown Error"); 22204 | } 22205 | } 22206 | 22207 | inline bool update_error(type& error, const std::string& expression) 22208 | { 22209 | if ( 22210 | expression.empty() || 22211 | (error.token.position > expression.size()) || 22212 | (std::numeric_limits<std::size_t>::max() == error.token.position) 22213 | ) 22214 | { 22215 | return false; 22216 | } 22217 | 22218 | std::size_t error_line_start = 0; 22219 | 22220 | for (std::size_t i = error.token.position; i > 0; --i) 22221 | { 22222 | const details::char_t c = expression[i]; 22223 | 22224 | if (('\n' == c) || ('\r' == c)) 22225 | { 22226 | error_line_start = i + 1; 22227 | break; 22228 | } 22229 | } 22230 | 22231 | std::size_t next_nl_position = std::min(expression.size(), 22232 | expression.find_first_of('\n',error.token.position + 1)); 22233 | 22234 | error.column_no = error.token.position - error_line_start; 22235 | error.error_line = expression.substr(error_line_start, 22236 | next_nl_position - error_line_start); 22237 | 22238 | error.line_no = 0; 22239 | 22240 | for (std::size_t i = 0; i < next_nl_position; ++i) 22241 | { 22242 | if ('\n' == expression[i]) 22243 | ++error.line_no; 22244 | } 22245 | 22246 | return true; 22247 | } 22248 | 22249 | inline void dump_error(const type& error) 22250 | { 22251 | printf("Position: %02d Type: [%s] Msg: %s\n", 22252 | static_cast<int>(error.token.position), 22253 | exprtk::parser_error::to_str(error.mode).c_str(), 22254 | error.diagnostic.c_str()); 22255 | } 22256 | } 22257 | 22258 | namespace details 22259 | { 22260 | template <typename Parser> 22261 | inline void disable_type_checking(Parser& p) 22262 | { 22263 | p.state_.type_check_enabled = false; 22264 | } 22265 | } 22266 | 22267 | template <typename T> 22268 | class parser : public lexer::parser_helper 22269 | { 22270 | private: 22271 | 22272 | enum precedence_level 22273 | { 22274 | e_level00, e_level01, e_level02, e_level03, e_level04, 22275 | e_level05, e_level06, e_level07, e_level08, e_level09, 22276 | e_level10, e_level11, e_level12, e_level13, e_level14 22277 | }; 22278 | 22279 | typedef const T& cref_t; 22280 | typedef const T const_t; 22281 | typedef ifunction<T> F; 22282 | typedef ivararg_function<T> VAF; 22283 | typedef igeneric_function<T> GF; 22284 | typedef ifunction<T> ifunction_t; 22285 | typedef ivararg_function<T> ivararg_function_t; 22286 | typedef igeneric_function<T> igeneric_function_t; 22287 | typedef details::expression_node<T> expression_node_t; 22288 | typedef details::literal_node<T> literal_node_t; 22289 | typedef details::unary_node<T> unary_node_t; 22290 | typedef details::binary_node<T> binary_node_t; 22291 | typedef details::trinary_node<T> trinary_node_t; 22292 | typedef details::quaternary_node<T> quaternary_node_t; 22293 | typedef details::conditional_node<T> conditional_node_t; 22294 | typedef details::cons_conditional_node<T> cons_conditional_node_t; 22295 | typedef details::while_loop_node<T> while_loop_node_t; 22296 | typedef details::repeat_until_loop_node<T> repeat_until_loop_node_t; 22297 | typedef details::for_loop_node<T> for_loop_node_t; 22298 | typedef details::while_loop_rtc_node<T> while_loop_rtc_node_t; 22299 | typedef details::repeat_until_loop_rtc_node<T> repeat_until_loop_rtc_node_t; 22300 | typedef details::for_loop_rtc_node<T> for_loop_rtc_node_t; 22301 | #ifndef exprtk_disable_break_continue 22302 | typedef details::while_loop_bc_node<T> while_loop_bc_node_t; 22303 | typedef details::repeat_until_loop_bc_node<T> repeat_until_loop_bc_node_t; 22304 | typedef details::for_loop_bc_node<T> for_loop_bc_node_t; 22305 | typedef details::while_loop_bc_rtc_node<T> while_loop_bc_rtc_node_t; 22306 | typedef details::repeat_until_loop_bc_rtc_node<T> repeat_until_loop_bc_rtc_node_t; 22307 | typedef details::for_loop_bc_rtc_node<T> for_loop_bc_rtc_node_t; 22308 | #endif 22309 | typedef details::switch_node<T> switch_node_t; 22310 | typedef details::variable_node<T> variable_node_t; 22311 | typedef details::vector_elem_node<T> vector_elem_node_t; 22312 | typedef details::vector_celem_node<T> vector_celem_node_t; 22313 | typedef details::vector_elem_rtc_node<T> vector_elem_rtc_node_t; 22314 | typedef details::vector_celem_rtc_node<T> vector_celem_rtc_node_t; 22315 | typedef details::rebasevector_elem_node<T> rebasevector_elem_node_t; 22316 | typedef details::rebasevector_celem_node<T> rebasevector_celem_node_t; 22317 | typedef details::rebasevector_elem_rtc_node<T> rebasevector_elem_rtc_node_t; 22318 | typedef details::rebasevector_celem_rtc_node<T> rebasevector_celem_rtc_node_t; 22319 | typedef details::vector_node<T> vector_node_t; 22320 | typedef details::vector_size_node<T> vector_size_node_t; 22321 | typedef details::range_pack<T> range_t; 22322 | #ifndef exprtk_disable_string_capabilities 22323 | typedef details::stringvar_node<T> stringvar_node_t; 22324 | typedef details::string_literal_node<T> string_literal_node_t; 22325 | typedef details::string_range_node<T> string_range_node_t; 22326 | typedef details::const_string_range_node<T> const_string_range_node_t; 22327 | typedef details::generic_string_range_node<T> generic_string_range_node_t; 22328 | typedef details::string_concat_node<T> string_concat_node_t; 22329 | typedef details::assignment_string_node<T> assignment_string_node_t; 22330 | typedef details::assignment_string_range_node<T> assignment_string_range_node_t; 22331 | typedef details::conditional_string_node<T> conditional_string_node_t; 22332 | typedef details::cons_conditional_str_node<T> cons_conditional_str_node_t; 22333 | #endif 22334 | typedef details::assignment_node<T> assignment_node_t; 22335 | typedef details::assignment_vec_elem_node<T> assignment_vec_elem_node_t; 22336 | typedef details::assignment_vec_elem_rtc_node<T> assignment_vec_elem_rtc_node_t; 22337 | typedef details::assignment_rebasevec_elem_node<T> assignment_rebasevec_elem_node_t; 22338 | typedef details::assignment_rebasevec_elem_rtc_node<T> assignment_rebasevec_elem_rtc_node_t; 22339 | typedef details::assignment_rebasevec_celem_node<T> assignment_rebasevec_celem_node_t; 22340 | typedef details::assignment_vec_node<T> assignment_vec_node_t; 22341 | typedef details::assignment_vecvec_node<T> assignment_vecvec_node_t; 22342 | typedef details::conditional_vector_node<T> conditional_vector_node_t; 22343 | typedef details::scand_node<T> scand_node_t; 22344 | typedef details::scor_node<T> scor_node_t; 22345 | typedef lexer::token token_t; 22346 | typedef expression_node_t* expression_node_ptr; 22347 | typedef expression<T> expression_t; 22348 | typedef symbol_table<T> symbol_table_t; 22349 | typedef typename expression<T>::symtab_list_t symbol_table_list_t; 22350 | typedef details::vector_holder<T> vector_holder_t; 22351 | typedef vector_holder_t* vector_holder_ptr; 22352 | 22353 | typedef typename details::functor_t<T> functor_t; 22354 | typedef typename functor_t::qfunc_t quaternary_functor_t; 22355 | typedef typename functor_t::tfunc_t trinary_functor_t; 22356 | typedef typename functor_t::bfunc_t binary_functor_t; 22357 | typedef typename functor_t::ufunc_t unary_functor_t; 22358 | 22359 | typedef details::operator_type operator_t; 22360 | 22361 | typedef std::map<operator_t, unary_functor_t > unary_op_map_t; 22362 | typedef std::map<operator_t, binary_functor_t > binary_op_map_t; 22363 | typedef std::map<operator_t, trinary_functor_t> trinary_op_map_t; 22364 | 22365 | typedef std::map<std::string,std::pair<trinary_functor_t ,operator_t> > sf3_map_t; 22366 | typedef std::map<std::string,std::pair<quaternary_functor_t,operator_t> > sf4_map_t; 22367 | 22368 | typedef std::map<binary_functor_t,operator_t> inv_binary_op_map_t; 22369 | typedef std::multimap<std::string,details::base_operation_t,details::ilesscompare> base_ops_map_t; 22370 | typedef std::set<std::string,details::ilesscompare> disabled_func_set_t; 22371 | 22372 | typedef details::T0oT1_define<T, cref_t , cref_t > vov_t; 22373 | typedef details::T0oT1_define<T, const_t, cref_t > cov_t; 22374 | typedef details::T0oT1_define<T, cref_t , const_t> voc_t; 22375 | 22376 | typedef details::T0oT1oT2_define<T, cref_t , cref_t , cref_t > vovov_t; 22377 | typedef details::T0oT1oT2_define<T, cref_t , cref_t , const_t> vovoc_t; 22378 | typedef details::T0oT1oT2_define<T, cref_t , const_t, cref_t > vocov_t; 22379 | typedef details::T0oT1oT2_define<T, const_t, cref_t , cref_t > covov_t; 22380 | typedef details::T0oT1oT2_define<T, const_t, cref_t , const_t> covoc_t; 22381 | typedef details::T0oT1oT2_define<T, const_t, const_t, cref_t > cocov_t; 22382 | typedef details::T0oT1oT2_define<T, cref_t , const_t, const_t> vococ_t; 22383 | 22384 | typedef details::T0oT1oT2oT3_define<T, cref_t , cref_t , cref_t , cref_t > vovovov_t; 22385 | typedef details::T0oT1oT2oT3_define<T, cref_t , cref_t , cref_t , const_t> vovovoc_t; 22386 | typedef details::T0oT1oT2oT3_define<T, cref_t , cref_t , const_t, cref_t > vovocov_t; 22387 | typedef details::T0oT1oT2oT3_define<T, cref_t , const_t, cref_t , cref_t > vocovov_t; 22388 | typedef details::T0oT1oT2oT3_define<T, const_t, cref_t , cref_t , cref_t > covovov_t; 22389 | 22390 | typedef details::T0oT1oT2oT3_define<T, const_t, cref_t , const_t, cref_t > covocov_t; 22391 | typedef details::T0oT1oT2oT3_define<T, cref_t , const_t, cref_t , const_t> vocovoc_t; 22392 | typedef details::T0oT1oT2oT3_define<T, const_t, cref_t , cref_t , const_t> covovoc_t; 22393 | typedef details::T0oT1oT2oT3_define<T, cref_t , const_t, const_t, cref_t > vococov_t; 22394 | 22395 | typedef results_context<T> results_context_t; 22396 | 22397 | typedef parser_helper prsrhlpr_t; 22398 | 22399 | struct scope_element 22400 | { 22401 | enum element_type 22402 | { 22403 | e_none , 22404 | e_literal , 22405 | e_variable, 22406 | e_vector , 22407 | e_vecelem , 22408 | e_string 22409 | }; 22410 | 22411 | typedef details::vector_holder<T> vector_holder_t; 22412 | typedef literal_node_t* literal_node_ptr; 22413 | typedef variable_node_t* variable_node_ptr; 22414 | typedef vector_holder_t* vector_holder_ptr; 22415 | typedef expression_node_t* expression_node_ptr; 22416 | #ifndef exprtk_disable_string_capabilities 22417 | typedef stringvar_node_t* stringvar_node_ptr; 22418 | #endif 22419 | 22420 | scope_element() 22421 | : name("???") 22422 | , size (std::numeric_limits<std::size_t>::max()) 22423 | , index(std::numeric_limits<std::size_t>::max()) 22424 | , depth(std::numeric_limits<std::size_t>::max()) 22425 | , ref_count(0) 22426 | , ip_index (0) 22427 | , type (e_none) 22428 | , active (false) 22429 | , data (0) 22430 | , var_node (0) 22431 | , vec_node (0) 22432 | #ifndef exprtk_disable_string_capabilities 22433 | , str_node(0) 22434 | #endif 22435 | {} 22436 | 22437 | bool operator < (const scope_element& se) const 22438 | { 22439 | if (ip_index < se.ip_index) 22440 | return true; 22441 | else if (ip_index > se.ip_index) 22442 | return false; 22443 | else if (depth < se.depth) 22444 | return true; 22445 | else if (depth > se.depth) 22446 | return false; 22447 | else if (index < se.index) 22448 | return true; 22449 | else if (index > se.index) 22450 | return false; 22451 | else 22452 | return (name < se.name); 22453 | } 22454 | 22455 | void clear() 22456 | { 22457 | name = "???" 22458 | size = std::numeric_limits<std::size_t>::max(); 22459 | index = std::numeric_limits<std::size_t>::max(); 22460 | depth = std::numeric_limits<std::size_t>::max(); 22461 | type = e_none; 22462 | active = false; 22463 | ref_count = 0; 22464 | ip_index = 0; 22465 | data = 0; 22466 | var_node = 0; 22467 | vec_node = 0; 22468 | #ifndef exprtk_disable_string_capabilities 22469 | str_node = 0; 22470 | #endif 22471 | } 22472 | 22473 | std::string name; 22474 | std::size_t size; 22475 | std::size_t index; 22476 | std::size_t depth; 22477 | std::size_t ref_count; 22478 | std::size_t ip_index; 22479 | element_type type; 22480 | bool active; 22481 | void* data; 22482 | expression_node_ptr var_node; 22483 | vector_holder_ptr vec_node; 22484 | #ifndef exprtk_disable_string_capabilities 22485 | stringvar_node_ptr str_node; 22486 | #endif 22487 | }; 22488 | 22489 | class scope_element_manager 22490 | { 22491 | public: 22492 | 22493 | typedef expression_node_t* expression_node_ptr; 22494 | typedef variable_node_t* variable_node_ptr; 22495 | typedef parser<T> parser_t; 22496 | 22497 | explicit scope_element_manager(parser<T>& p) 22498 | : parser_(p) 22499 | , input_param_cnt_(0) 22500 | , total_local_symb_size_bytes_(0) 22501 | {} 22502 | 22503 | inline std::size_t size() const 22504 | { 22505 | return element_.size(); 22506 | } 22507 | 22508 | inline bool empty() const 22509 | { 22510 | return element_.empty(); 22511 | } 22512 | 22513 | inline scope_element& get_element(const std::size_t& index) 22514 | { 22515 | if (index < element_.size()) 22516 | return element_[index]; 22517 | else 22518 | return null_element_; 22519 | } 22520 | 22521 | inline scope_element& get_element(const std::string& var_name, 22522 | const std::size_t index = std::numeric_limits<std::size_t>::max()) 22523 | { 22524 | const std::size_t current_depth = parser_.state_.scope_depth; 22525 | 22526 | for (std::size_t i = 0; i < element_.size(); ++i) 22527 | { 22528 | scope_element& se = element_[i]; 22529 | 22530 | if (se.depth > current_depth) 22531 | continue; 22532 | else if ( 22533 | details::imatch(se.name, var_name) && 22534 | (se.index == index) 22535 | ) 22536 | return se; 22537 | } 22538 | 22539 | return null_element_; 22540 | } 22541 | 22542 | inline scope_element& get_active_element(const std::string& var_name, 22543 | const std::size_t index = std::numeric_limits<std::size_t>::max()) 22544 | { 22545 | const std::size_t current_depth = parser_.state_.scope_depth; 22546 | 22547 | for (std::size_t i = 0; i < element_.size(); ++i) 22548 | { 22549 | scope_element& se = element_[i]; 22550 | 22551 | if (se.depth > current_depth) 22552 | continue; 22553 | else if ( 22554 | details::imatch(se.name, var_name) && 22555 | (se.index == index) && 22556 | (se.active) 22557 | ) 22558 | return se; 22559 | } 22560 | 22561 | return null_element_; 22562 | } 22563 | 22564 | inline bool add_element(const scope_element& se) 22565 | { 22566 | for (std::size_t i = 0; i < element_.size(); ++i) 22567 | { 22568 | scope_element& cse = element_[i]; 22569 | 22570 | if ( 22571 | details::imatch(cse.name, se.name) && 22572 | (cse.depth <= se.depth) && 22573 | (cse.index == se.index) && 22574 | (cse.size == se.size ) && 22575 | (cse.type == se.type ) && 22576 | (cse.active) 22577 | ) 22578 | return false; 22579 | } 22580 | 22581 | switch (se.type) 22582 | { 22583 | case scope_element::e_variable : total_local_symb_size_bytes_ += sizeof(T); 22584 | break; 22585 | 22586 | case scope_element::e_literal : total_local_symb_size_bytes_ += sizeof(T); 22587 | break; 22588 | 22589 | case scope_element::e_vector : total_local_symb_size_bytes_ += sizeof(T) * se.size; 22590 | break; 22591 | 22592 | default : break; 22593 | } 22594 | 22595 | element_.push_back(se); 22596 | std::sort(element_.begin(),element_.end()); 22597 | 22598 | return true; 22599 | } 22600 | 22601 | inline void deactivate(const std::size_t& scope_depth) 22602 | { 22603 | exprtk_debug(("deactivate() - Scope depth: %d\n", 22604 | static_cast<int>(parser_.state_.scope_depth))); 22605 | 22606 | for (std::size_t i = 0; i < element_.size(); ++i) 22607 | { 22608 | scope_element& se = element_[i]; 22609 | 22610 | if (se.active && (se.depth >= scope_depth)) 22611 | { 22612 | exprtk_debug(("deactivate() - element[%02d] '%s'\n", 22613 | static_cast<int>(i), 22614 | se.name.c_str())); 22615 | 22616 | se.active = false; 22617 | } 22618 | } 22619 | } 22620 | 22621 | inline void free_element(scope_element& se) 22622 | { 22623 | exprtk_debug(("free_element() - se[%s]\n", se.name.c_str())); 22624 | 22625 | switch (se.type) 22626 | { 22627 | case scope_element::e_literal : delete reinterpret_cast<T*>(se.data); 22628 | delete se.var_node; 22629 | break; 22630 | 22631 | case scope_element::e_variable : delete reinterpret_cast<T*>(se.data); 22632 | delete se.var_node; 22633 | break; 22634 | 22635 | case scope_element::e_vector : delete[] reinterpret_cast<T*>(se.data); 22636 | delete se.vec_node; 22637 | break; 22638 | 22639 | case scope_element::e_vecelem : delete se.var_node; 22640 | break; 22641 | 22642 | #ifndef exprtk_disable_string_capabilities 22643 | case scope_element::e_string : delete reinterpret_cast<std::string*>(se.data); 22644 | delete se.str_node; 22645 | break; 22646 | #endif 22647 | 22648 | default : return; 22649 | } 22650 | 22651 | se.clear(); 22652 | } 22653 | 22654 | inline void cleanup() 22655 | { 22656 | for (std::size_t i = 0; i < element_.size(); ++i) 22657 | { 22658 | free_element(element_[i]); 22659 | } 22660 | 22661 | element_.clear(); 22662 | 22663 | input_param_cnt_ = 0; 22664 | total_local_symb_size_bytes_ = 0; 22665 | } 22666 | 22667 | inline std::size_t total_local_symb_size_bytes() const 22668 | { 22669 | return total_local_symb_size_bytes_; 22670 | } 22671 | 22672 | inline std::size_t next_ip_index() 22673 | { 22674 | return ++input_param_cnt_; 22675 | } 22676 | 22677 | inline expression_node_ptr get_variable(const T& v) 22678 | { 22679 | for (std::size_t i = 0; i < element_.size(); ++i) 22680 | { 22681 | scope_element& se = element_[i]; 22682 | 22683 | if ( 22684 | se.active && 22685 | se.var_node && 22686 | details::is_variable_node(se.var_node) 22687 | ) 22688 | { 22689 | variable_node_ptr vn = reinterpret_cast<variable_node_ptr>(se.var_node); 22690 | 22691 | if (&(vn->ref()) == (&v)) 22692 | { 22693 | return se.var_node; 22694 | } 22695 | } 22696 | } 22697 | 22698 | return expression_node_ptr(0); 22699 | } 22700 | 22701 | inline std::string get_vector_name(const T* data) 22702 | { 22703 | for (std::size_t i = 0; i < element_.size(); ++i) 22704 | { 22705 | scope_element& se = element_[i]; 22706 | 22707 | if ( 22708 | se.active && 22709 | se.vec_node && 22710 | (se.vec_node->data() == data) 22711 | ) 22712 | { 22713 | return se.name; 22714 | } 22715 | } 22716 | 22717 | return "neo-vector" 22718 | } 22719 | 22720 | private: 22721 | 22722 | scope_element_manager(const scope_element_manager&) exprtk_delete; 22723 | scope_element_manager& operator=(const scope_element_manager&) exprtk_delete; 22724 | 22725 | parser_t& parser_; 22726 | std::vector<scope_element> element_; 22727 | scope_element null_element_; 22728 | std::size_t input_param_cnt_; 22729 | std::size_t total_local_symb_size_bytes_; 22730 | }; 22731 | 22732 | class scope_handler 22733 | { 22734 | public: 22735 | 22736 | typedef parser<T> parser_t; 22737 | 22738 | explicit scope_handler(parser<T>& p) 22739 | : parser_(p) 22740 | { 22741 | parser_.state_.scope_depth++; 22742 | #ifdef exprtk_enable_debugging 22743 | const std::string depth(2 * parser_.state_.scope_depth,'-'); 22744 | exprtk_debug(("%s> Scope Depth: %02d\n", 22745 | depth.c_str(), 22746 | static_cast<int>(parser_.state_.scope_depth))); 22747 | #endif 22748 | } 22749 | 22750 | ~scope_handler() 22751 | { 22752 | parser_.sem_.deactivate(parser_.state_.scope_depth); 22753 | parser_.state_.scope_depth--; 22754 | #ifdef exprtk_enable_debugging 22755 | const std::string depth(2 * parser_.state_.scope_depth,'-'); 22756 | exprtk_debug(("<%s Scope Depth: %02d\n", 22757 | depth.c_str(), 22758 | static_cast<int>(parser_.state_.scope_depth))); 22759 | #endif 22760 | } 22761 | 22762 | private: 22763 | 22764 | scope_handler(const scope_handler&) exprtk_delete; 22765 | scope_handler& operator=(const scope_handler&) exprtk_delete; 22766 | 22767 | parser_t& parser_; 22768 | }; 22769 | 22770 | template <typename T_> 22771 | struct halfopen_range_policy 22772 | { 22773 | static inline bool is_within(const T_& v, const T_& begin, const T_& end) 22774 | { 22775 | assert(begin <= end); 22776 | return (begin <= v) && (v < end); 22777 | } 22778 | 22779 | static inline bool is_less(const T_& v, const T_& begin) 22780 | { 22781 | return (v < begin); 22782 | } 22783 | 22784 | static inline bool is_greater(const T_& v, const T_& end) 22785 | { 22786 | return (end <= v); 22787 | } 22788 | 22789 | static inline bool end_inclusive() 22790 | { 22791 | return false; 22792 | } 22793 | }; 22794 | 22795 | template <typename T_> 22796 | struct closed_range_policy 22797 | { 22798 | static inline bool is_within(const T_& v, const T_& begin, const T_& end) 22799 | { 22800 | assert(begin <= end); 22801 | return (begin <= v) && (v <= end); 22802 | } 22803 | 22804 | static inline bool is_less(const T_& v, const T_& begin) 22805 | { 22806 | return (v < begin); 22807 | } 22808 | 22809 | static inline bool is_greater(const T_& v, const T_& end) 22810 | { 22811 | return (end < v); 22812 | } 22813 | 22814 | static inline bool end_inclusive() 22815 | { 22816 | return true; 22817 | } 22818 | }; 22819 | 22820 | template <typename IntervalPointType, 22821 | typename RangePolicy = halfopen_range_policy<IntervalPointType> > 22822 | class interval_container_t 22823 | { 22824 | public: 22825 | 22826 | typedef IntervalPointType interval_point_t; 22827 | typedef std::pair<interval_point_t, interval_point_t> interval_t; 22828 | typedef std::map<interval_point_t, interval_t> interval_map_t; 22829 | typedef typename interval_map_t::const_iterator interval_map_citr_t; 22830 | 22831 | std::size_t size() const 22832 | { 22833 | return interval_map_.size(); 22834 | } 22835 | 22836 | void reset() 22837 | { 22838 | interval_map_.clear(); 22839 | } 22840 | 22841 | bool in_interval(const interval_point_t point, interval_t& interval) const 22842 | { 22843 | interval_map_citr_t itr = RangePolicy::end_inclusive() ? 22844 | interval_map_.lower_bound(point): 22845 | interval_map_.upper_bound(point); 22846 | 22847 | for (; itr != interval_map_.end(); ++itr) 22848 | { 22849 | const interval_point_t& begin = itr->second.first; 22850 | const interval_point_t& end = itr->second.second; 22851 | 22852 | if (RangePolicy::is_within(point, begin, end)) 22853 | { 22854 | interval = interval_t(begin,end); 22855 | return true; 22856 | } 22857 | else if (RangePolicy::is_greater(point, end)) 22858 | { 22859 | break; 22860 | } 22861 | } 22862 | 22863 | return false; 22864 | } 22865 | 22866 | bool in_interval(const interval_point_t point) const 22867 | { 22868 | interval_t interval; 22869 | return in_interval(point,interval); 22870 | } 22871 | 22872 | bool add_interval(const interval_point_t begin, const interval_point_t end) 22873 | { 22874 | if ((end <= begin) || in_interval(begin) || in_interval(end)) 22875 | { 22876 | return false; 22877 | } 22878 | 22879 | interval_map_[end] = std::make_pair(begin, end); 22880 | 22881 | return true; 22882 | } 22883 | 22884 | bool add_interval(const interval_t interval) 22885 | { 22886 | return add_interval(interval.first, interval.second); 22887 | } 22888 | 22889 | private: 22890 | 22891 | interval_map_t interval_map_; 22892 | }; 22893 | 22894 | class stack_limit_handler 22895 | { 22896 | public: 22897 | 22898 | typedef parser<T> parser_t; 22899 | 22900 | explicit stack_limit_handler(parser<T>& p) 22901 | : parser_(p) 22902 | , limit_exceeded_(false) 22903 | { 22904 | if (++parser_.state_.stack_depth > parser_.settings_.max_stack_depth_) 22905 | { 22906 | limit_exceeded_ = true; 22907 | parser_.set_error(make_error( 22908 | parser_error::e_parser, 22909 | "ERR000 - Current stack depth " + details::to_str(parser_.state_.stack_depth) + 22910 | " exceeds maximum allowed stack depth of " + details::to_str(parser_.settings_.max_stack_depth_), 22911 | exprtk_error_location)); 22912 | } 22913 | } 22914 | 22915 | ~stack_limit_handler() 22916 | { 22917 | assert(parser_.state_.stack_depth > 0); 22918 | parser_.state_.stack_depth--; 22919 | } 22920 | 22921 | bool operator!() 22922 | { 22923 | return limit_exceeded_; 22924 | } 22925 | 22926 | private: 22927 | 22928 | stack_limit_handler(const stack_limit_handler&) exprtk_delete; 22929 | stack_limit_handler& operator=(const stack_limit_handler&) exprtk_delete; 22930 | 22931 | parser_t& parser_; 22932 | bool limit_exceeded_; 22933 | }; 22934 | 22935 | struct symtab_store 22936 | { 22937 | symbol_table_list_t symtab_list_; 22938 | 22939 | typedef typename symbol_table_t::local_data_t local_data_t; 22940 | typedef typename symbol_table_t::variable_ptr variable_ptr; 22941 | typedef typename symbol_table_t::function_ptr function_ptr; 22942 | #ifndef exprtk_disable_string_capabilities 22943 | typedef typename symbol_table_t::stringvar_ptr stringvar_ptr; 22944 | #endif 22945 | typedef typename symbol_table_t::vector_holder_ptr vector_holder_ptr; 22946 | typedef typename symbol_table_t::vararg_function_ptr vararg_function_ptr; 22947 | typedef typename symbol_table_t::generic_function_ptr generic_function_ptr; 22948 | 22949 | struct variable_context 22950 | { 22951 | variable_context() 22952 | : symbol_table(0) 22953 | , variable(0) 22954 | {} 22955 | 22956 | const symbol_table_t* symbol_table; 22957 | variable_ptr variable; 22958 | }; 22959 | 22960 | struct vector_context 22961 | { 22962 | vector_context() 22963 | : symbol_table(0) 22964 | , vector_holder(0) 22965 | {} 22966 | 22967 | const symbol_table_t* symbol_table; 22968 | vector_holder_ptr vector_holder; 22969 | }; 22970 | 22971 | #ifndef exprtk_disable_string_capabilities 22972 | struct string_context 22973 | { 22974 | string_context() 22975 | : symbol_table(0) 22976 | , str_var(0) 22977 | {} 22978 | 22979 | const symbol_table_t* symbol_table; 22980 | stringvar_ptr str_var; 22981 | }; 22982 | #endif 22983 | 22984 | inline bool empty() const 22985 | { 22986 | return symtab_list_.empty(); 22987 | } 22988 | 22989 | inline void clear() 22990 | { 22991 | symtab_list_.clear(); 22992 | } 22993 | 22994 | inline bool valid() const 22995 | { 22996 | if (!empty()) 22997 | { 22998 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 22999 | { 23000 | if (symtab_list_[i].valid()) 23001 | return true; 23002 | } 23003 | } 23004 | 23005 | return false; 23006 | } 23007 | 23008 | inline bool valid_symbol(const std::string& symbol) const 23009 | { 23010 | if (!symtab_list_.empty()) 23011 | return symtab_list_[0].valid_symbol(symbol); 23012 | else 23013 | return false; 23014 | } 23015 | 23016 | inline bool valid_function_name(const std::string& symbol) const 23017 | { 23018 | if (!symtab_list_.empty()) 23019 | return symtab_list_[0].valid_function(symbol); 23020 | else 23021 | return false; 23022 | } 23023 | 23024 | inline variable_context get_variable_context(const std::string& variable_name) const 23025 | { 23026 | variable_context result; 23027 | 23028 | if (valid_symbol(variable_name)) 23029 | { 23030 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23031 | { 23032 | if (!symtab_list_[i].valid()) 23033 | { 23034 | continue; 23035 | } 23036 | 23037 | result.variable = local_data(i) 23038 | .variable_store.get(variable_name); 23039 | if (result.variable) 23040 | { 23041 | result.symbol_table = &symtab_list_[i]; 23042 | break; 23043 | } 23044 | } 23045 | } 23046 | 23047 | return result; 23048 | } 23049 | 23050 | inline variable_ptr get_variable(const std::string& variable_name) const 23051 | { 23052 | if (!valid_symbol(variable_name)) 23053 | return reinterpret_cast<variable_ptr>(0); 23054 | 23055 | variable_ptr result = reinterpret_cast<variable_ptr>(0); 23056 | 23057 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23058 | { 23059 | if (!symtab_list_[i].valid()) 23060 | continue; 23061 | else 23062 | result = local_data(i) 23063 | .variable_store.get(variable_name); 23064 | 23065 | if (result) break; 23066 | } 23067 | 23068 | return result; 23069 | } 23070 | 23071 | inline variable_ptr get_variable(const T& var_ref) const 23072 | { 23073 | variable_ptr result = reinterpret_cast<variable_ptr>(0); 23074 | 23075 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23076 | { 23077 | if (!symtab_list_[i].valid()) 23078 | continue; 23079 | else 23080 | result = local_data(i).variable_store 23081 | .get_from_varptr(reinterpret_cast<const void*>(&var_ref)); 23082 | 23083 | if (result) break; 23084 | } 23085 | 23086 | return result; 23087 | } 23088 | 23089 | #ifndef exprtk_disable_string_capabilities 23090 | inline string_context get_string_context(const std::string& string_name) const 23091 | { 23092 | string_context result; 23093 | 23094 | if (!valid_symbol(string_name)) 23095 | return result; 23096 | 23097 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23098 | { 23099 | if (!symtab_list_[i].valid()) 23100 | { 23101 | continue; 23102 | } 23103 | 23104 | result.str_var = local_data(i).stringvar_store.get(string_name); 23105 | 23106 | if (result.str_var) 23107 | { 23108 | result.symbol_table = &symtab_list_[i]; 23109 | break; 23110 | } 23111 | } 23112 | 23113 | return result; 23114 | } 23115 | 23116 | inline stringvar_ptr get_stringvar(const std::string& string_name) const 23117 | { 23118 | if (!valid_symbol(string_name)) 23119 | return reinterpret_cast<stringvar_ptr>(0); 23120 | 23121 | stringvar_ptr result = reinterpret_cast<stringvar_ptr>(0); 23122 | 23123 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23124 | { 23125 | if (!symtab_list_[i].valid()) 23126 | continue; 23127 | else 23128 | result = local_data(i) 23129 | .stringvar_store.get(string_name); 23130 | 23131 | if (result) break; 23132 | } 23133 | 23134 | return result; 23135 | } 23136 | #endif 23137 | 23138 | inline function_ptr get_function(const std::string& function_name) const 23139 | { 23140 | if (!valid_function_name(function_name)) 23141 | return reinterpret_cast<function_ptr>(0); 23142 | 23143 | function_ptr result = reinterpret_cast<function_ptr>(0); 23144 | 23145 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23146 | { 23147 | if (!symtab_list_[i].valid()) 23148 | continue; 23149 | else 23150 | result = local_data(i) 23151 | .function_store.get(function_name); 23152 | 23153 | if (result) break; 23154 | } 23155 | 23156 | return result; 23157 | } 23158 | 23159 | inline vararg_function_ptr get_vararg_function(const std::string& vararg_function_name) const 23160 | { 23161 | if (!valid_function_name(vararg_function_name)) 23162 | return reinterpret_cast<vararg_function_ptr>(0); 23163 | 23164 | vararg_function_ptr result = reinterpret_cast<vararg_function_ptr>(0); 23165 | 23166 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23167 | { 23168 | if (!symtab_list_[i].valid()) 23169 | continue; 23170 | else 23171 | result = local_data(i) 23172 | .vararg_function_store.get(vararg_function_name); 23173 | 23174 | if (result) break; 23175 | } 23176 | 23177 | return result; 23178 | } 23179 | 23180 | inline generic_function_ptr get_generic_function(const std::string& function_name) const 23181 | { 23182 | if (!valid_function_name(function_name)) 23183 | return reinterpret_cast<generic_function_ptr>(0); 23184 | 23185 | generic_function_ptr result = reinterpret_cast<generic_function_ptr>(0); 23186 | 23187 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23188 | { 23189 | if (!symtab_list_[i].valid()) 23190 | continue; 23191 | else 23192 | result = local_data(i) 23193 | .generic_function_store.get(function_name); 23194 | 23195 | if (result) break; 23196 | } 23197 | 23198 | return result; 23199 | } 23200 | 23201 | inline generic_function_ptr get_string_function(const std::string& function_name) const 23202 | { 23203 | if (!valid_function_name(function_name)) 23204 | return reinterpret_cast<generic_function_ptr>(0); 23205 | 23206 | generic_function_ptr result = reinterpret_cast<generic_function_ptr>(0); 23207 | 23208 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23209 | { 23210 | if (!symtab_list_[i].valid()) 23211 | continue; 23212 | else 23213 | result = 23214 | local_data(i).string_function_store.get(function_name); 23215 | 23216 | if (result) break; 23217 | } 23218 | 23219 | return result; 23220 | } 23221 | 23222 | inline generic_function_ptr get_overload_function(const std::string& function_name) const 23223 | { 23224 | if (!valid_function_name(function_name)) 23225 | return reinterpret_cast<generic_function_ptr>(0); 23226 | 23227 | generic_function_ptr result = reinterpret_cast<generic_function_ptr>(0); 23228 | 23229 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23230 | { 23231 | if (!symtab_list_[i].valid()) 23232 | continue; 23233 | else 23234 | result = 23235 | local_data(i).overload_function_store.get(function_name); 23236 | 23237 | if (result) break; 23238 | } 23239 | 23240 | return result; 23241 | } 23242 | 23243 | inline vector_context get_vector_context(const std::string& vector_name) const 23244 | { 23245 | vector_context result; 23246 | if (!valid_symbol(vector_name)) 23247 | return result; 23248 | 23249 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23250 | { 23251 | if (!symtab_list_[i].valid()) 23252 | { 23253 | continue; 23254 | } 23255 | 23256 | result.vector_holder = local_data(i).vector_store.get(vector_name); 23257 | 23258 | if (result.vector_holder) 23259 | { 23260 | result.symbol_table = &symtab_list_[i]; 23261 | break; 23262 | } 23263 | } 23264 | 23265 | return result; 23266 | } 23267 | 23268 | inline vector_holder_ptr get_vector(const std::string& vector_name) const 23269 | { 23270 | if (!valid_symbol(vector_name)) 23271 | return reinterpret_cast<vector_holder_ptr>(0); 23272 | 23273 | vector_holder_ptr result = reinterpret_cast<vector_holder_ptr>(0); 23274 | 23275 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23276 | { 23277 | if (!symtab_list_[i].valid()) 23278 | { 23279 | continue; 23280 | } 23281 | 23282 | result = local_data(i).vector_store.get(vector_name); 23283 | 23284 | if (result) 23285 | { 23286 | break; 23287 | } 23288 | } 23289 | 23290 | return result; 23291 | } 23292 | 23293 | inline bool is_constant_node(const std::string& symbol_name) const 23294 | { 23295 | if (!valid_symbol(symbol_name)) 23296 | return false; 23297 | 23298 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23299 | { 23300 | if (!symtab_list_[i].valid()) 23301 | { 23302 | continue; 23303 | } 23304 | 23305 | if (local_data(i).variable_store.is_constant(symbol_name)) 23306 | { 23307 | return true; 23308 | } 23309 | } 23310 | 23311 | return false; 23312 | } 23313 | 23314 | #ifndef exprtk_disable_string_capabilities 23315 | inline bool is_constant_string(const std::string& symbol_name) const 23316 | { 23317 | if (!valid_symbol(symbol_name)) 23318 | return false; 23319 | 23320 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23321 | { 23322 | if (!symtab_list_[i].valid()) 23323 | continue; 23324 | else if (!local_data(i).stringvar_store.symbol_exists(symbol_name)) 23325 | continue; 23326 | else if (local_data(i).stringvar_store.is_constant(symbol_name)) 23327 | return true; 23328 | } 23329 | 23330 | return false; 23331 | } 23332 | #endif 23333 | 23334 | inline bool symbol_exists(const std::string& symbol) const 23335 | { 23336 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23337 | { 23338 | if (!symtab_list_[i].valid()) 23339 | { 23340 | continue; 23341 | } 23342 | 23343 | if (symtab_list_[i].symbol_exists(symbol)) 23344 | { 23345 | return true; 23346 | } 23347 | } 23348 | 23349 | return false; 23350 | } 23351 | 23352 | inline bool is_variable(const std::string& variable_name) const 23353 | { 23354 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23355 | { 23356 | if (!symtab_list_[i].valid()) 23357 | continue; 23358 | else if ( 23359 | symtab_list_[i].local_data().variable_store 23360 | .symbol_exists(variable_name) 23361 | ) 23362 | return true; 23363 | } 23364 | 23365 | return false; 23366 | } 23367 | 23368 | #ifndef exprtk_disable_string_capabilities 23369 | inline bool is_stringvar(const std::string& stringvar_name) const 23370 | { 23371 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23372 | { 23373 | if (!symtab_list_[i].valid()) 23374 | continue; 23375 | else if ( 23376 | symtab_list_[i].local_data().stringvar_store 23377 | .symbol_exists(stringvar_name) 23378 | ) 23379 | return true; 23380 | } 23381 | 23382 | return false; 23383 | } 23384 | 23385 | inline bool is_conststr_stringvar(const std::string& symbol_name) const 23386 | { 23387 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23388 | { 23389 | if (!symtab_list_[i].valid()) 23390 | continue; 23391 | else if ( 23392 | symtab_list_[i].local_data().stringvar_store 23393 | .symbol_exists(symbol_name) 23394 | ) 23395 | { 23396 | return ( 23397 | local_data(i).stringvar_store.symbol_exists(symbol_name) || 23398 | local_data(i).stringvar_store.is_constant (symbol_name) 23399 | ); 23400 | 23401 | } 23402 | } 23403 | 23404 | return false; 23405 | } 23406 | #endif 23407 | 23408 | inline bool is_function(const std::string& function_name) const 23409 | { 23410 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23411 | { 23412 | if (!symtab_list_[i].valid()) 23413 | continue; 23414 | else if ( 23415 | local_data(i).vararg_function_store 23416 | .symbol_exists(function_name) 23417 | ) 23418 | return true; 23419 | } 23420 | 23421 | return false; 23422 | } 23423 | 23424 | inline bool is_vararg_function(const std::string& vararg_function_name) const 23425 | { 23426 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23427 | { 23428 | if (!symtab_list_[i].valid()) 23429 | continue; 23430 | else if ( 23431 | local_data(i).vararg_function_store 23432 | .symbol_exists(vararg_function_name) 23433 | ) 23434 | return true; 23435 | } 23436 | 23437 | return false; 23438 | } 23439 | 23440 | inline bool is_vector(const std::string& vector_name) const 23441 | { 23442 | for (std::size_t i = 0; i < symtab_list_.size(); ++i) 23443 | { 23444 | if (!symtab_list_[i].valid()) 23445 | continue; 23446 | else if ( 23447 | local_data(i).vector_store 23448 | .symbol_exists(vector_name) 23449 | ) 23450 | return true; 23451 | } 23452 | 23453 | return false; 23454 | } 23455 | 23456 | inline std::string get_variable_name(const expression_node_ptr& ptr) const 23457 | { 23458 | return local_data().variable_store.entity_name(ptr); 23459 | } 23460 | 23461 | inline std::string get_vector_name(const vector_holder_ptr& ptr) const 23462 | { 23463 | return local_data().vector_store.entity_name(ptr); 23464 | } 23465 | 23466 | #ifndef exprtk_disable_string_capabilities 23467 | inline std::string get_stringvar_name(const expression_node_ptr& ptr) const 23468 | { 23469 | return local_data().stringvar_store.entity_name(ptr); 23470 | } 23471 | 23472 | inline std::string get_conststr_stringvar_name(const expression_node_ptr& ptr) const 23473 | { 23474 | return local_data().stringvar_store.entity_name(ptr); 23475 | } 23476 | #endif 23477 | 23478 | inline local_data_t& local_data(const std::size_t& index = 0) 23479 | { 23480 | return symtab_list_[index].local_data(); 23481 | } 23482 | 23483 | inline const local_data_t& local_data(const std::size_t& index = 0) const 23484 | { 23485 | return symtab_list_[index].local_data(); 23486 | } 23487 | 23488 | inline symbol_table_t& get_symbol_table(const std::size_t& index = 0) 23489 | { 23490 | return symtab_list_[index]; 23491 | } 23492 | }; 23493 | 23494 | struct parser_state 23495 | { 23496 | parser_state() 23497 | : type_check_enabled(true) 23498 | { 23499 | reset(); 23500 | } 23501 | 23502 | void reset() 23503 | { 23504 | parsing_return_stmt = false; 23505 | parsing_break_stmt = false; 23506 | parsing_assert_stmt = false; 23507 | return_stmt_present = false; 23508 | side_effect_present = false; 23509 | scope_depth = 0; 23510 | stack_depth = 0; 23511 | parsing_loop_stmt_count = 0; 23512 | } 23513 | 23514 | #ifndef exprtk_enable_debugging 23515 | void activate_side_effect(const std::string&) 23516 | #else 23517 | void activate_side_effect(const std::string& source) 23518 | #endif 23519 | { 23520 | if (!side_effect_present) 23521 | { 23522 | side_effect_present = true; 23523 | 23524 | exprtk_debug(("activate_side_effect() - caller: %s\n", source.c_str())); 23525 | } 23526 | } 23527 | 23528 | bool parsing_return_stmt; 23529 | bool parsing_break_stmt; 23530 | bool parsing_assert_stmt; 23531 | bool return_stmt_present; 23532 | bool side_effect_present; 23533 | bool type_check_enabled; 23534 | std::size_t scope_depth; 23535 | std::size_t stack_depth; 23536 | std::size_t parsing_loop_stmt_count; 23537 | }; 23538 | 23539 | public: 23540 | 23541 | struct unknown_symbol_resolver 23542 | { 23543 | 23544 | enum usr_symbol_type 23545 | { 23546 | e_usr_unknown_type = 0, 23547 | e_usr_variable_type = 1, 23548 | e_usr_constant_type = 2 23549 | }; 23550 | 23551 | enum usr_mode 23552 | { 23553 | e_usrmode_default = 0, 23554 | e_usrmode_extended = 1 23555 | }; 23556 | 23557 | usr_mode mode; 23558 | 23559 | explicit unknown_symbol_resolver(const usr_mode m = e_usrmode_default) 23560 | : mode(m) 23561 | {} 23562 | 23563 | virtual ~unknown_symbol_resolver() 23564 | {} 23565 | 23566 | virtual bool process(const std::string& /*unknown_symbol*/, 23567 | usr_symbol_type& st, 23568 | T& default_value, 23569 | std::string& error_message) 23570 | { 23571 | if (e_usrmode_default != mode) 23572 | return false; 23573 | 23574 | st = e_usr_variable_type; 23575 | default_value = T(0); 23576 | error_message.clear(); 23577 | 23578 | return true; 23579 | } 23580 | 23581 | virtual bool process(const std::string& /* unknown_symbol */, 23582 | symbol_table_t& /* symbol_table */, 23583 | std::string& /* error_message */) 23584 | { 23585 | return false; 23586 | } 23587 | }; 23588 | 23589 | enum collect_type 23590 | { 23591 | e_ct_none = 0, 23592 | e_ct_variables = 1, 23593 | e_ct_functions = 2, 23594 | e_ct_assignments = 4 23595 | }; 23596 | 23597 | enum symbol_type 23598 | { 23599 | e_st_unknown = 0, 23600 | e_st_variable = 1, 23601 | e_st_vector = 2, 23602 | e_st_vecelem = 3, 23603 | e_st_string = 4, 23604 | e_st_function = 5, 23605 | e_st_local_variable = 6, 23606 | e_st_local_vector = 7, 23607 | e_st_local_string = 8 23608 | }; 23609 | 23610 | class dependent_entity_collector 23611 | { 23612 | public: 23613 | 23614 | typedef std::pair<std::string,symbol_type> symbol_t; 23615 | typedef std::vector<symbol_t> symbol_list_t; 23616 | 23617 | explicit dependent_entity_collector(const std::size_t options = e_ct_none) 23618 | : options_(options) 23619 | , collect_variables_ ((options_ & e_ct_variables ) == e_ct_variables ) 23620 | , collect_functions_ ((options_ & e_ct_functions ) == e_ct_functions ) 23621 | , collect_assignments_((options_ & e_ct_assignments) == e_ct_assignments) 23622 | , return_present_ (false) 23623 | , final_stmt_return_(false) 23624 | {} 23625 | 23626 | template <typename Allocator, 23627 | template <typename, typename> class Sequence> 23628 | inline std::size_t symbols(Sequence<symbol_t,Allocator>& symbols_list) 23629 | { 23630 | if (!collect_variables_ && !collect_functions_) 23631 | return 0; 23632 | else if (symbol_name_list_.empty()) 23633 | return 0; 23634 | 23635 | for (std::size_t i = 0; i < symbol_name_list_.size(); ++i) 23636 | { 23637 | details::case_normalise(symbol_name_list_[i].first); 23638 | } 23639 | 23640 | std::sort(symbol_name_list_.begin(), symbol_name_list_.end()); 23641 | 23642 | std::unique_copy 23643 | ( 23644 | symbol_name_list_.begin(), 23645 | symbol_name_list_.end (), 23646 | std::back_inserter(symbols_list) 23647 | ); 23648 | 23649 | return symbols_list.size(); 23650 | } 23651 | 23652 | template <typename Allocator, 23653 | template <typename, typename> class Sequence> 23654 | inline std::size_t assignment_symbols(Sequence<symbol_t,Allocator>& assignment_list) 23655 | { 23656 | if (!collect_assignments_) 23657 | return 0; 23658 | else if (assignment_name_list_.empty()) 23659 | return 0; 23660 | 23661 | for (std::size_t i = 0; i < assignment_name_list_.size(); ++i) 23662 | { 23663 | details::case_normalise(assignment_name_list_[i].first); 23664 | } 23665 | 23666 | std::sort(assignment_name_list_.begin(),assignment_name_list_.end()); 23667 | 23668 | std::unique_copy 23669 | ( 23670 | assignment_name_list_.begin(), 23671 | assignment_name_list_.end (), 23672 | std::back_inserter(assignment_list) 23673 | ); 23674 | 23675 | return assignment_list.size(); 23676 | } 23677 | 23678 | void clear() 23679 | { 23680 | symbol_name_list_ .clear(); 23681 | assignment_name_list_.clear(); 23682 | retparam_list_ .clear(); 23683 | return_present_ = false; 23684 | final_stmt_return_ = false; 23685 | } 23686 | 23687 | bool& collect_variables() 23688 | { 23689 | return collect_variables_; 23690 | } 23691 | 23692 | bool& collect_functions() 23693 | { 23694 | return collect_functions_; 23695 | } 23696 | 23697 | bool& collect_assignments() 23698 | { 23699 | return collect_assignments_; 23700 | } 23701 | 23702 | bool return_present() const 23703 | { 23704 | return return_present_; 23705 | } 23706 | 23707 | bool final_stmt_return() const 23708 | { 23709 | return final_stmt_return_; 23710 | } 23711 | 23712 | typedef std::vector<std::string> retparam_list_t; 23713 | 23714 | retparam_list_t return_param_type_list() const 23715 | { 23716 | return retparam_list_; 23717 | } 23718 | 23719 | private: 23720 | 23721 | inline void add_symbol(const std::string& symbol, const symbol_type st) 23722 | { 23723 | switch (st) 23724 | { 23725 | case e_st_variable : 23726 | case e_st_vector : 23727 | case e_st_string : 23728 | case e_st_local_variable : 23729 | case e_st_local_vector : 23730 | case e_st_local_string : if (collect_variables_) 23731 | symbol_name_list_ 23732 | .push_back(std::make_pair(symbol, st)); 23733 | break; 23734 | 23735 | case e_st_function : if (collect_functions_) 23736 | symbol_name_list_ 23737 | .push_back(std::make_pair(symbol, st)); 23738 | break; 23739 | 23740 | default : return; 23741 | } 23742 | } 23743 | 23744 | inline void add_assignment(const std::string& symbol, const symbol_type st) 23745 | { 23746 | switch (st) 23747 | { 23748 | case e_st_variable : 23749 | case e_st_vector : 23750 | case e_st_string : if (collect_assignments_) 23751 | assignment_name_list_ 23752 | .push_back(std::make_pair(symbol, st)); 23753 | break; 23754 | 23755 | default : return; 23756 | } 23757 | } 23758 | 23759 | std::size_t options_; 23760 | bool collect_variables_; 23761 | bool collect_functions_; 23762 | bool collect_assignments_; 23763 | bool return_present_; 23764 | bool final_stmt_return_; 23765 | symbol_list_t symbol_name_list_; 23766 | symbol_list_t assignment_name_list_; 23767 | retparam_list_t retparam_list_; 23768 | 23769 | friend class parser<T>; 23770 | }; 23771 | 23772 | class settings_store 23773 | { 23774 | private: 23775 | 23776 | typedef std::set<std::string,details::ilesscompare> disabled_entity_set_t; 23777 | typedef disabled_entity_set_t::iterator des_itr_t; 23778 | 23779 | public: 23780 | 23781 | enum settings_compilation_options 23782 | { 23783 | e_unknown = 0, 23784 | e_replacer = 1, 23785 | e_joiner = 2, 23786 | e_numeric_check = 4, 23787 | e_bracket_check = 8, 23788 | e_sequence_check = 16, 23789 | e_commutative_check = 32, 23790 | e_strength_reduction = 64, 23791 | e_disable_vardef = 128, 23792 | e_collect_vars = 256, 23793 | e_collect_funcs = 512, 23794 | e_collect_assings = 1024, 23795 | e_disable_usr_on_rsrvd = 2048, 23796 | e_disable_zero_return = 4096 23797 | }; 23798 | 23799 | enum settings_base_funcs 23800 | { 23801 | e_bf_unknown = 0, 23802 | e_bf_abs , e_bf_acos , e_bf_acosh , e_bf_asin , 23803 | e_bf_asinh , e_bf_atan , e_bf_atan2 , e_bf_atanh , 23804 | e_bf_avg , e_bf_ceil , e_bf_clamp , e_bf_cos , 23805 | e_bf_cosh , e_bf_cot , e_bf_csc , e_bf_equal , 23806 | e_bf_erf , e_bf_erfc , e_bf_exp , e_bf_expm1 , 23807 | e_bf_floor , e_bf_frac , e_bf_hypot , e_bf_iclamp , 23808 | e_bf_like , e_bf_log , e_bf_log10 , e_bf_log1p , 23809 | e_bf_log2 , e_bf_logn , e_bf_mand , e_bf_max , 23810 | e_bf_min , e_bf_mod , e_bf_mor , e_bf_mul , 23811 | e_bf_ncdf , e_bf_pow , e_bf_root , e_bf_round , 23812 | e_bf_roundn , e_bf_sec , e_bf_sgn , e_bf_sin , 23813 | e_bf_sinc , e_bf_sinh , e_bf_sqrt , e_bf_sum , 23814 | e_bf_swap , e_bf_tan , e_bf_tanh , e_bf_trunc , 23815 | e_bf_not_equal , e_bf_inrange , e_bf_deg2grad , e_bf_deg2rad , 23816 | e_bf_rad2deg , e_bf_grad2deg 23817 | }; 23818 | 23819 | enum settings_control_structs 23820 | { 23821 | e_ctrl_unknown = 0, 23822 | e_ctrl_ifelse, 23823 | e_ctrl_switch, 23824 | e_ctrl_for_loop, 23825 | e_ctrl_while_loop, 23826 | e_ctrl_repeat_loop, 23827 | e_ctrl_return 23828 | }; 23829 | 23830 | enum settings_logic_opr 23831 | { 23832 | e_logic_unknown = 0, 23833 | e_logic_and, e_logic_nand , e_logic_nor , 23834 | e_logic_not, e_logic_or , e_logic_xnor, 23835 | e_logic_xor, e_logic_scand, e_logic_scor 23836 | }; 23837 | 23838 | enum settings_arithmetic_opr 23839 | { 23840 | e_arith_unknown = 0, 23841 | e_arith_add, e_arith_sub, e_arith_mul, 23842 | e_arith_div, e_arith_mod, e_arith_pow 23843 | }; 23844 | 23845 | enum settings_assignment_opr 23846 | { 23847 | e_assign_unknown = 0, 23848 | e_assign_assign, e_assign_addass, e_assign_subass, 23849 | e_assign_mulass, e_assign_divass, e_assign_modass 23850 | }; 23851 | 23852 | enum settings_inequality_opr 23853 | { 23854 | e_ineq_unknown = 0, 23855 | e_ineq_lt , e_ineq_lte, e_ineq_eq , 23856 | e_ineq_equal, e_ineq_ne , e_ineq_nequal, 23857 | e_ineq_gte , e_ineq_gt 23858 | }; 23859 | 23860 | static const std::size_t default_compile_all_opts = 23861 | e_replacer + 23862 | e_joiner + 23863 | e_numeric_check + 23864 | e_bracket_check + 23865 | e_sequence_check + 23866 | e_commutative_check + 23867 | e_strength_reduction; 23868 | 23869 | settings_store(const std::size_t compile_options = default_compile_all_opts) 23870 | : max_stack_depth_(400) 23871 | , max_node_depth_(10000) 23872 | , max_total_local_symbol_size_bytes_(2000000000) 23873 | , max_local_vector_size_(max_total_local_symbol_size_bytes_ / sizeof(T)) 23874 | { 23875 | load_compile_options(compile_options); 23876 | } 23877 | 23878 | settings_store& enable_all_base_functions() 23879 | { 23880 | disabled_func_set_.clear(); 23881 | return (*this); 23882 | } 23883 | 23884 | settings_store& enable_all_control_structures() 23885 | { 23886 | disabled_ctrl_set_.clear(); 23887 | return (*this); 23888 | } 23889 | 23890 | settings_store& enable_all_logic_ops() 23891 | { 23892 | disabled_logic_set_.clear(); 23893 | return (*this); 23894 | } 23895 | 23896 | settings_store& enable_all_arithmetic_ops() 23897 | { 23898 | disabled_arithmetic_set_.clear(); 23899 | return (*this); 23900 | } 23901 | 23902 | settings_store& enable_all_assignment_ops() 23903 | { 23904 | disabled_assignment_set_.clear(); 23905 | return (*this); 23906 | } 23907 | 23908 | settings_store& enable_all_inequality_ops() 23909 | { 23910 | disabled_inequality_set_.clear(); 23911 | return (*this); 23912 | } 23913 | 23914 | settings_store& enable_local_vardef() 23915 | { 23916 | disable_vardef_ = false; 23917 | return (*this); 23918 | } 23919 | 23920 | settings_store& enable_commutative_check() 23921 | { 23922 | enable_commutative_check_ = true; 23923 | return (*this); 23924 | } 23925 | 23926 | settings_store& enable_strength_reduction() 23927 | { 23928 | enable_strength_reduction_ = true; 23929 | return (*this); 23930 | } 23931 | 23932 | settings_store& disable_all_base_functions() 23933 | { 23934 | std::copy(details::base_function_list, 23935 | details::base_function_list + details::base_function_list_size, 23936 | std::insert_iterator<disabled_entity_set_t> 23937 | (disabled_func_set_, disabled_func_set_.begin())); 23938 | return (*this); 23939 | } 23940 | 23941 | settings_store& disable_all_control_structures() 23942 | { 23943 | std::copy(details::cntrl_struct_list, 23944 | details::cntrl_struct_list + details::cntrl_struct_list_size, 23945 | std::insert_iterator<disabled_entity_set_t> 23946 | (disabled_ctrl_set_, disabled_ctrl_set_.begin())); 23947 | return (*this); 23948 | } 23949 | 23950 | settings_store& disable_all_logic_ops() 23951 | { 23952 | std::copy(details::logic_ops_list, 23953 | details::logic_ops_list + details::logic_ops_list_size, 23954 | std::insert_iterator<disabled_entity_set_t> 23955 | (disabled_logic_set_, disabled_logic_set_.begin())); 23956 | return (*this); 23957 | } 23958 | 23959 | settings_store& disable_all_arithmetic_ops() 23960 | { 23961 | std::copy(details::arithmetic_ops_list, 23962 | details::arithmetic_ops_list + details::arithmetic_ops_list_size, 23963 | std::insert_iterator<disabled_entity_set_t> 23964 | (disabled_arithmetic_set_, disabled_arithmetic_set_.begin())); 23965 | return (*this); 23966 | } 23967 | 23968 | settings_store& disable_all_assignment_ops() 23969 | { 23970 | std::copy(details::assignment_ops_list, 23971 | details::assignment_ops_list + details::assignment_ops_list_size, 23972 | std::insert_iterator<disabled_entity_set_t> 23973 | (disabled_assignment_set_, disabled_assignment_set_.begin())); 23974 | return (*this); 23975 | } 23976 | 23977 | settings_store& disable_all_inequality_ops() 23978 | { 23979 | std::copy(details::inequality_ops_list, 23980 | details::inequality_ops_list + details::inequality_ops_list_size, 23981 | std::insert_iterator<disabled_entity_set_t> 23982 | (disabled_inequality_set_, disabled_inequality_set_.begin())); 23983 | return (*this); 23984 | } 23985 | 23986 | settings_store& disable_local_vardef() 23987 | { 23988 | disable_vardef_ = true; 23989 | return (*this); 23990 | } 23991 | 23992 | settings_store& disable_commutative_check() 23993 | { 23994 | enable_commutative_check_ = false; 23995 | return (*this); 23996 | } 23997 | 23998 | settings_store& disable_strength_reduction() 23999 | { 24000 | enable_strength_reduction_ = false; 24001 | return (*this); 24002 | } 24003 | 24004 | bool replacer_enabled () const { return enable_replacer_; } 24005 | bool commutative_check_enabled () const { return enable_commutative_check_; } 24006 | bool joiner_enabled () const { return enable_joiner_; } 24007 | bool numeric_check_enabled () const { return enable_numeric_check_; } 24008 | bool bracket_check_enabled () const { return enable_bracket_check_; } 24009 | bool sequence_check_enabled () const { return enable_sequence_check_; } 24010 | bool strength_reduction_enabled () const { return enable_strength_reduction_; } 24011 | bool collect_variables_enabled () const { return enable_collect_vars_; } 24012 | bool collect_functions_enabled () const { return enable_collect_funcs_; } 24013 | bool collect_assignments_enabled() const { return enable_collect_assings_; } 24014 | bool vardef_disabled () const { return disable_vardef_; } 24015 | bool rsrvd_sym_usr_disabled () const { return disable_rsrvd_sym_usr_; } 24016 | bool zero_return_disabled () const { return disable_zero_return_; } 24017 | 24018 | bool function_enabled(const std::string& function_name) const 24019 | { 24020 | if (disabled_func_set_.empty()) 24021 | return true; 24022 | else 24023 | return (disabled_func_set_.end() == disabled_func_set_.find(function_name)); 24024 | } 24025 | 24026 | bool control_struct_enabled(const std::string& control_struct) const 24027 | { 24028 | if (disabled_ctrl_set_.empty()) 24029 | return true; 24030 | else 24031 | return (disabled_ctrl_set_.end() == disabled_ctrl_set_.find(control_struct)); 24032 | } 24033 | 24034 | bool logic_enabled(const std::string& logic_operation) const 24035 | { 24036 | if (disabled_logic_set_.empty()) 24037 | return true; 24038 | else 24039 | return (disabled_logic_set_.end() == disabled_logic_set_.find(logic_operation)); 24040 | } 24041 | 24042 | bool arithmetic_enabled(const details::operator_type& arithmetic_operation) const 24043 | { 24044 | if (disabled_logic_set_.empty()) 24045 | return true; 24046 | else 24047 | return disabled_arithmetic_set_.end() == disabled_arithmetic_set_ 24048 | .find(arith_opr_to_string(arithmetic_operation)); 24049 | } 24050 | 24051 | bool assignment_enabled(const details::operator_type& assignment) const 24052 | { 24053 | if (disabled_assignment_set_.empty()) 24054 | return true; 24055 | else 24056 | return disabled_assignment_set_.end() == disabled_assignment_set_ 24057 | .find(assign_opr_to_string(assignment)); 24058 | } 24059 | 24060 | bool inequality_enabled(const details::operator_type& inequality) const 24061 | { 24062 | if (disabled_inequality_set_.empty()) 24063 | return true; 24064 | else 24065 | return disabled_inequality_set_.end() == disabled_inequality_set_ 24066 | .find(inequality_opr_to_string(inequality)); 24067 | } 24068 | 24069 | bool function_disabled(const std::string& function_name) const 24070 | { 24071 | if (disabled_func_set_.empty()) 24072 | return false; 24073 | else 24074 | return (disabled_func_set_.end() != disabled_func_set_.find(function_name)); 24075 | } 24076 | 24077 | bool control_struct_disabled(const std::string& control_struct) const 24078 | { 24079 | if (disabled_ctrl_set_.empty()) 24080 | return false; 24081 | else 24082 | return (disabled_ctrl_set_.end() != disabled_ctrl_set_.find(control_struct)); 24083 | } 24084 | 24085 | bool logic_disabled(const std::string& logic_operation) const 24086 | { 24087 | if (disabled_logic_set_.empty()) 24088 | return false; 24089 | else 24090 | return (disabled_logic_set_.end() != disabled_logic_set_.find(logic_operation)); 24091 | } 24092 | 24093 | bool assignment_disabled(const details::operator_type assignment_operation) const 24094 | { 24095 | if (disabled_assignment_set_.empty()) 24096 | return false; 24097 | else 24098 | return disabled_assignment_set_.end() != disabled_assignment_set_ 24099 | .find(assign_opr_to_string(assignment_operation)); 24100 | } 24101 | 24102 | bool logic_disabled(const details::operator_type logic_operation) const 24103 | { 24104 | if (disabled_logic_set_.empty()) 24105 | return false; 24106 | else 24107 | return disabled_logic_set_.end() != disabled_logic_set_ 24108 | .find(logic_opr_to_string(logic_operation)); 24109 | } 24110 | 24111 | bool arithmetic_disabled(const details::operator_type arithmetic_operation) const 24112 | { 24113 | if (disabled_arithmetic_set_.empty()) 24114 | return false; 24115 | else 24116 | return disabled_arithmetic_set_.end() != disabled_arithmetic_set_ 24117 | .find(arith_opr_to_string(arithmetic_operation)); 24118 | } 24119 | 24120 | bool inequality_disabled(const details::operator_type& inequality) const 24121 | { 24122 | if (disabled_inequality_set_.empty()) 24123 | return false; 24124 | else 24125 | return disabled_inequality_set_.end() != disabled_inequality_set_ 24126 | .find(inequality_opr_to_string(inequality)); 24127 | } 24128 | 24129 | settings_store& disable_base_function(const settings_base_funcs bf) 24130 | { 24131 | if ( 24132 | (e_bf_unknown != bf) && 24133 | (static_cast<std::size_t>(bf) < (details::base_function_list_size + 1)) 24134 | ) 24135 | { 24136 | disabled_func_set_.insert(details::base_function_list[bf - 1]); 24137 | } 24138 | 24139 | return (*this); 24140 | } 24141 | 24142 | settings_store& disable_control_structure(const settings_control_structs ctrl_struct) 24143 | { 24144 | if ( 24145 | (e_ctrl_unknown != ctrl_struct) && 24146 | (static_cast<std::size_t>(ctrl_struct) < (details::cntrl_struct_list_size + 1)) 24147 | ) 24148 | { 24149 | disabled_ctrl_set_.insert(details::cntrl_struct_list[ctrl_struct - 1]); 24150 | } 24151 | 24152 | return (*this); 24153 | } 24154 | 24155 | settings_store& disable_logic_operation(const settings_logic_opr logic) 24156 | { 24157 | if ( 24158 | (e_logic_unknown != logic) && 24159 | (static_cast<std::size_t>(logic) < (details::logic_ops_list_size + 1)) 24160 | ) 24161 | { 24162 | disabled_logic_set_.insert(details::logic_ops_list[logic - 1]); 24163 | } 24164 | 24165 | return (*this); 24166 | } 24167 | 24168 | settings_store& disable_arithmetic_operation(const settings_arithmetic_opr arithmetic) 24169 | { 24170 | if ( 24171 | (e_arith_unknown != arithmetic) && 24172 | (static_cast<std::size_t>(arithmetic) < (details::arithmetic_ops_list_size + 1)) 24173 | ) 24174 | { 24175 | disabled_arithmetic_set_.insert(details::arithmetic_ops_list[arithmetic - 1]); 24176 | } 24177 | 24178 | return (*this); 24179 | } 24180 | 24181 | settings_store& disable_assignment_operation(const settings_assignment_opr assignment) 24182 | { 24183 | if ( 24184 | (e_assign_unknown != assignment) && 24185 | (static_cast<std::size_t>(assignment) < (details::assignment_ops_list_size + 1)) 24186 | ) 24187 | { 24188 | disabled_assignment_set_.insert(details::assignment_ops_list[assignment - 1]); 24189 | } 24190 | 24191 | return (*this); 24192 | } 24193 | 24194 | settings_store& disable_inequality_operation(const settings_inequality_opr inequality) 24195 | { 24196 | if ( 24197 | (e_ineq_unknown != inequality) && 24198 | (static_cast<std::size_t>(inequality) < (details::inequality_ops_list_size + 1)) 24199 | ) 24200 | { 24201 | disabled_inequality_set_.insert(details::inequality_ops_list[inequality - 1]); 24202 | } 24203 | 24204 | return (*this); 24205 | } 24206 | 24207 | settings_store& enable_base_function(const settings_base_funcs bf) 24208 | { 24209 | if ( 24210 | (e_bf_unknown != bf) && 24211 | (static_cast<std::size_t>(bf) < (details::base_function_list_size + 1)) 24212 | ) 24213 | { 24214 | const des_itr_t itr = disabled_func_set_.find(details::base_function_list[bf - 1]); 24215 | 24216 | if (disabled_func_set_.end() != itr) 24217 | { 24218 | disabled_func_set_.erase(itr); 24219 | } 24220 | } 24221 | 24222 | return (*this); 24223 | } 24224 | 24225 | settings_store& enable_control_structure(const settings_control_structs ctrl_struct) 24226 | { 24227 | if ( 24228 | (e_ctrl_unknown != ctrl_struct) && 24229 | (static_cast<std::size_t>(ctrl_struct) < (details::cntrl_struct_list_size + 1)) 24230 | ) 24231 | { 24232 | const des_itr_t itr = disabled_ctrl_set_.find(details::cntrl_struct_list[ctrl_struct - 1]); 24233 | 24234 | if (disabled_ctrl_set_.end() != itr) 24235 | { 24236 | disabled_ctrl_set_.erase(itr); 24237 | } 24238 | } 24239 | 24240 | return (*this); 24241 | } 24242 | 24243 | settings_store& enable_logic_operation(const settings_logic_opr logic) 24244 | { 24245 | if ( 24246 | (e_logic_unknown != logic) && 24247 | (static_cast<std::size_t>(logic) < (details::logic_ops_list_size + 1)) 24248 | ) 24249 | { 24250 | const des_itr_t itr = disabled_logic_set_.find(details::logic_ops_list[logic - 1]); 24251 | 24252 | if (disabled_logic_set_.end() != itr) 24253 | { 24254 | disabled_logic_set_.erase(itr); 24255 | } 24256 | } 24257 | 24258 | return (*this); 24259 | } 24260 | 24261 | settings_store& enable_arithmetic_operation(const settings_arithmetic_opr arithmetic) 24262 | { 24263 | if ( 24264 | (e_arith_unknown != arithmetic) && 24265 | (static_cast<std::size_t>(arithmetic) < (details::arithmetic_ops_list_size + 1)) 24266 | ) 24267 | { 24268 | const des_itr_t itr = disabled_arithmetic_set_.find(details::arithmetic_ops_list[arithmetic - 1]); 24269 | 24270 | if (disabled_arithmetic_set_.end() != itr) 24271 | { 24272 | disabled_arithmetic_set_.erase(itr); 24273 | } 24274 | } 24275 | 24276 | return (*this); 24277 | } 24278 | 24279 | settings_store& enable_assignment_operation(const settings_assignment_opr assignment) 24280 | { 24281 | if ( 24282 | (e_assign_unknown != assignment) && 24283 | (static_cast<std::size_t>(assignment) < (details::assignment_ops_list_size + 1)) 24284 | ) 24285 | { 24286 | const des_itr_t itr = disabled_assignment_set_.find(details::assignment_ops_list[assignment - 1]); 24287 | 24288 | if (disabled_assignment_set_.end() != itr) 24289 | { 24290 | disabled_assignment_set_.erase(itr); 24291 | } 24292 | } 24293 | 24294 | return (*this); 24295 | } 24296 | 24297 | settings_store& enable_inequality_operation(const settings_inequality_opr inequality) 24298 | { 24299 | if ( 24300 | (e_ineq_unknown != inequality) && 24301 | (static_cast<std::size_t>(inequality) < (details::inequality_ops_list_size + 1)) 24302 | ) 24303 | { 24304 | const des_itr_t itr = disabled_inequality_set_.find(details::inequality_ops_list[inequality - 1]); 24305 | 24306 | if (disabled_inequality_set_.end() != itr) 24307 | { 24308 | disabled_inequality_set_.erase(itr); 24309 | } 24310 | } 24311 | 24312 | return (*this); 24313 | } 24314 | 24315 | void set_max_stack_depth(const std::size_t max_stack_depth) 24316 | { 24317 | max_stack_depth_ = max_stack_depth; 24318 | } 24319 | 24320 | void set_max_node_depth(const std::size_t max_node_depth) 24321 | { 24322 | max_node_depth_ = max_node_depth; 24323 | } 24324 | 24325 | void set_max_local_vector_size(const std::size_t max_local_vector_size) 24326 | { 24327 | max_local_vector_size_ = max_local_vector_size; 24328 | } 24329 | 24330 | void set_max_total_local_symbol_size_bytes(const std::size_t max_total_lcl_symb_size) 24331 | { 24332 | max_total_local_symbol_size_bytes_ = max_total_lcl_symb_size; 24333 | } 24334 | 24335 | std::size_t max_stack_depth() const 24336 | { 24337 | return max_stack_depth_; 24338 | } 24339 | 24340 | std::size_t max_node_depth() const 24341 | { 24342 | return max_node_depth_; 24343 | } 24344 | 24345 | std::size_t max_local_vector_size() const 24346 | { 24347 | return max_local_vector_size_; 24348 | } 24349 | 24350 | std::size_t max_total_local_symbol_size_bytes() const 24351 | { 24352 | return max_total_local_symbol_size_bytes_; 24353 | } 24354 | 24355 | private: 24356 | 24357 | void load_compile_options(const std::size_t compile_options) 24358 | { 24359 | enable_replacer_ = (compile_options & e_replacer ) == e_replacer; 24360 | enable_joiner_ = (compile_options & e_joiner ) == e_joiner; 24361 | enable_numeric_check_ = (compile_options & e_numeric_check ) == e_numeric_check; 24362 | enable_bracket_check_ = (compile_options & e_bracket_check ) == e_bracket_check; 24363 | enable_sequence_check_ = (compile_options & e_sequence_check ) == e_sequence_check; 24364 | enable_commutative_check_ = (compile_options & e_commutative_check ) == e_commutative_check; 24365 | enable_strength_reduction_ = (compile_options & e_strength_reduction ) == e_strength_reduction; 24366 | enable_collect_vars_ = (compile_options & e_collect_vars ) == e_collect_vars; 24367 | enable_collect_funcs_ = (compile_options & e_collect_funcs ) == e_collect_funcs; 24368 | enable_collect_assings_ = (compile_options & e_collect_assings ) == e_collect_assings; 24369 | disable_vardef_ = (compile_options & e_disable_vardef ) == e_disable_vardef; 24370 | disable_rsrvd_sym_usr_ = (compile_options & e_disable_usr_on_rsrvd) == e_disable_usr_on_rsrvd; 24371 | disable_zero_return_ = (compile_options & e_disable_zero_return ) == e_disable_zero_return; 24372 | } 24373 | 24374 | std::string assign_opr_to_string(details::operator_type opr) const 24375 | { 24376 | switch (opr) 24377 | { 24378 | case details::e_assign : return ":=" 24379 | case details::e_addass : return "+=" 24380 | case details::e_subass : return "-=" 24381 | case details::e_mulass : return "*=" 24382 | case details::e_divass : return "/=" 24383 | case details::e_modass : return "%=" 24384 | default : return "" ; 24385 | } 24386 | } 24387 | 24388 | std::string arith_opr_to_string(details::operator_type opr) const 24389 | { 24390 | switch (opr) 24391 | { 24392 | case details::e_add : return "+" 24393 | case details::e_sub : return "-" 24394 | case details::e_mul : return "*" 24395 | case details::e_div : return "/" 24396 | case details::e_mod : return "%" 24397 | case details::e_pow : return "^" 24398 | default : return "" ; 24399 | } 24400 | } 24401 | 24402 | std::string inequality_opr_to_string(details::operator_type opr) const 24403 | { 24404 | switch (opr) 24405 | { 24406 | case details::e_lt : return "<" ; 24407 | case details::e_lte : return "<=" 24408 | case details::e_eq : return "==" 24409 | case details::e_equal : return "=" ; 24410 | case details::e_ne : return "!=" 24411 | case details::e_nequal: return "<>" 24412 | case details::e_gte : return ">=" 24413 | case details::e_gt : return ">" ; 24414 | default : return "" ; 24415 | } 24416 | } 24417 | 24418 | std::string logic_opr_to_string(details::operator_type opr) const 24419 | { 24420 | switch (opr) 24421 | { 24422 | case details::e_and : return "and" ; 24423 | case details::e_or : return "or" ; 24424 | case details::e_xor : return "xor" ; 24425 | case details::e_nand : return "nand" 24426 | case details::e_nor : return "nor" ; 24427 | case details::e_xnor : return "xnor" 24428 | case details::e_notl : return "not" ; 24429 | default : return "" ; 24430 | } 24431 | } 24432 | 24433 | bool enable_replacer_; 24434 | bool enable_joiner_; 24435 | bool enable_numeric_check_; 24436 | bool enable_bracket_check_; 24437 | bool enable_sequence_check_; 24438 | bool enable_commutative_check_; 24439 | bool enable_strength_reduction_; 24440 | bool enable_collect_vars_; 24441 | bool enable_collect_funcs_; 24442 | bool enable_collect_assings_; 24443 | bool disable_vardef_; 24444 | bool disable_rsrvd_sym_usr_; 24445 | bool disable_zero_return_; 24446 | 24447 | disabled_entity_set_t disabled_func_set_ ; 24448 | disabled_entity_set_t disabled_ctrl_set_ ; 24449 | disabled_entity_set_t disabled_logic_set_; 24450 | disabled_entity_set_t disabled_arithmetic_set_; 24451 | disabled_entity_set_t disabled_assignment_set_; 24452 | disabled_entity_set_t disabled_inequality_set_; 24453 | 24454 | std::size_t max_stack_depth_; 24455 | std::size_t max_node_depth_; 24456 | std::size_t max_total_local_symbol_size_bytes_; 24457 | std::size_t max_local_vector_size_; 24458 | 24459 | friend class parser<T>; 24460 | }; 24461 | 24462 | typedef settings_store settings_t; 24463 | 24464 | explicit parser(const settings_t& settings = settings_t()) 24465 | : settings_(settings) 24466 | , resolve_unknown_symbol_(false) 24467 | , results_context_(0) 24468 | , unknown_symbol_resolver_(reinterpret_cast<unknown_symbol_resolver*>(0)) 24469 | #ifdef _MSC_VER 24470 | #pragma warning(push) 24471 | #pragma warning (disable:4355) 24472 | #endif 24473 | , sem_(*this) 24474 | #ifdef _MSC_VER 24475 | #pragma warning(pop) 24476 | #endif 24477 | , operator_joiner_2_(2) 24478 | , operator_joiner_3_(3) 24479 | , loop_runtime_check_(0) 24480 | , vector_access_runtime_check_(0) 24481 | , compilation_check_ptr_(0) 24482 | , assert_check_(0) 24483 | { 24484 | init_precompilation(); 24485 | 24486 | load_operations_map (base_ops_map_ ); 24487 | load_unary_operations_map (unary_op_map_ ); 24488 | load_binary_operations_map (binary_op_map_ ); 24489 | load_inv_binary_operations_map(inv_binary_op_map_); 24490 | load_sf3_map (sf3_map_ ); 24491 | load_sf4_map (sf4_map_ ); 24492 | 24493 | expression_generator_.init_synthesize_map(); 24494 | expression_generator_.set_parser(*this); 24495 | expression_generator_.set_uom (unary_op_map_ ); 24496 | expression_generator_.set_bom (binary_op_map_ ); 24497 | expression_generator_.set_ibom(inv_binary_op_map_); 24498 | expression_generator_.set_sf3m(sf3_map_ ); 24499 | expression_generator_.set_sf4m(sf4_map_ ); 24500 | expression_generator_.set_strength_reduction_state(settings_.strength_reduction_enabled()); 24501 | } 24502 | 24503 | ~parser() 24504 | {} 24505 | 24506 | inline void init_precompilation() 24507 | { 24508 | dec_.collect_variables() = 24509 | settings_.collect_variables_enabled(); 24510 | 24511 | dec_.collect_functions() = 24512 | settings_.collect_functions_enabled(); 24513 | 24514 | dec_.collect_assignments() = 24515 | settings_.collect_assignments_enabled(); 24516 | 24517 | if (settings_.replacer_enabled()) 24518 | { 24519 | symbol_replacer_.clear(); 24520 | symbol_replacer_.add_replace("true" , "1", lexer::token::e_number); 24521 | symbol_replacer_.add_replace("false", "0", lexer::token::e_number); 24522 | helper_assembly_.token_modifier_list.clear(); 24523 | helper_assembly_.register_modifier(&symbol_replacer_); 24524 | } 24525 | 24526 | if (settings_.commutative_check_enabled()) 24527 | { 24528 | for (std::size_t i = 0; i < details::reserved_words_size; ++i) 24529 | { 24530 | commutative_inserter_.ignore_symbol(details::reserved_words[i]); 24531 | } 24532 | 24533 | helper_assembly_.token_inserter_list.clear(); 24534 | helper_assembly_.register_inserter(&commutative_inserter_); 24535 | } 24536 | 24537 | if (settings_.joiner_enabled()) 24538 | { 24539 | helper_assembly_.token_joiner_list.clear(); 24540 | helper_assembly_.register_joiner(&operator_joiner_2_); 24541 | helper_assembly_.register_joiner(&operator_joiner_3_); 24542 | } 24543 | 24544 | if ( 24545 | settings_.numeric_check_enabled () || 24546 | settings_.bracket_check_enabled () || 24547 | settings_.sequence_check_enabled() 24548 | ) 24549 | { 24550 | helper_assembly_.token_scanner_list.clear(); 24551 | 24552 | if (settings_.numeric_check_enabled()) 24553 | { 24554 | helper_assembly_.register_scanner(&numeric_checker_); 24555 | } 24556 | 24557 | if (settings_.bracket_check_enabled()) 24558 | { 24559 | helper_assembly_.register_scanner(&bracket_checker_); 24560 | } 24561 | 24562 | if (settings_.sequence_check_enabled()) 24563 | { 24564 | helper_assembly_.register_scanner(&sequence_validator_ ); 24565 | helper_assembly_.register_scanner(&sequence_validator_3tkns_); 24566 | } 24567 | } 24568 | } 24569 | 24570 | inline bool compile(const std::string& expression_string, expression<T>& expr) 24571 | { 24572 | state_ .reset(); 24573 | error_list_ .clear(); 24574 | brkcnt_list_ .clear(); 24575 | synthesis_error_ .clear(); 24576 | immutable_memory_map_.reset(); 24577 | immutable_symtok_map_.clear(); 24578 | current_state_stack_ .clear(); 24579 | assert_ids_ .clear(); 24580 | sem_ .cleanup(); 24581 | 24582 | return_cleanup(); 24583 | 24584 | if (!valid_settings()) 24585 | { 24586 | return false; 24587 | } 24588 | 24589 | expression_generator_.set_allocator(node_allocator_); 24590 | 24591 | if (expression_string.empty()) 24592 | { 24593 | set_error(make_error( 24594 | parser_error::e_syntax, 24595 | "ERR001 - Empty expression!", 24596 | exprtk_error_location)); 24597 | 24598 | return false; 24599 | } 24600 | 24601 | if (!init(expression_string)) 24602 | { 24603 | process_lexer_errors(); 24604 | return false; 24605 | } 24606 | 24607 | if (lexer().empty()) 24608 | { 24609 | set_error(make_error( 24610 | parser_error::e_syntax, 24611 | "ERR002 - Empty expression!", 24612 | exprtk_error_location)); 24613 | 24614 | return false; 24615 | } 24616 | 24617 | if (halt_compilation_check()) 24618 | { 24619 | exprtk_debug(("halt_compilation_check() - compile checkpoint 0\n")); 24620 | sem_.cleanup(); 24621 | return false; 24622 | } 24623 | 24624 | if (!run_assemblies()) 24625 | { 24626 | sem_.cleanup(); 24627 | return false; 24628 | } 24629 | 24630 | if (halt_compilation_check()) 24631 | { 24632 | exprtk_debug(("halt_compilation_check() - compile checkpoint 1\n")); 24633 | sem_.cleanup(); 24634 | return false; 24635 | } 24636 | 24637 | symtab_store_.symtab_list_ = expr.get_symbol_table_list(); 24638 | dec_.clear(); 24639 | 24640 | lexer().begin(); 24641 | 24642 | next_token(); 24643 | 24644 | expression_node_ptr e = parse_corpus(); 24645 | 24646 | if ((0 != e) && (token_t::e_eof == current_token().type)) 24647 | { 24648 | bool* retinvk_ptr = 0; 24649 | 24650 | if (state_.return_stmt_present) 24651 | { 24652 | dec_.return_present_ = true; 24653 | 24654 | e = expression_generator_ 24655 | .return_envelope(e, results_context_, retinvk_ptr); 24656 | } 24657 | 24658 | expr.set_expression(e); 24659 | expr.set_retinvk(retinvk_ptr); 24660 | 24661 | register_local_vars(expr); 24662 | register_return_results(expr); 24663 | 24664 | return !(!expr); 24665 | } 24666 | else 24667 | { 24668 | if (error_list_.empty()) 24669 | { 24670 | set_error(make_error( 24671 | parser_error::e_syntax, 24672 | current_token(), 24673 | "ERR003 - Invalid expression encountered", 24674 | exprtk_error_location)); 24675 | } 24676 | 24677 | if ((0 != e) && branch_deletable(e)) 24678 | { 24679 | destroy_node(e); 24680 | } 24681 | 24682 | dec_.clear (); 24683 | sem_.cleanup (); 24684 | return_cleanup(); 24685 | 24686 | return false; 24687 | } 24688 | } 24689 | 24690 | inline expression_t compile(const std::string& expression_string, symbol_table_t& symtab) 24691 | { 24692 | expression_t expression; 24693 | expression.register_symbol_table(symtab); 24694 | compile(expression_string,expression); 24695 | return expression; 24696 | } 24697 | 24698 | void process_lexer_errors() 24699 | { 24700 | for (std::size_t i = 0; i < lexer().size(); ++i) 24701 | { 24702 | if (lexer()[i].is_error()) 24703 | { 24704 | std::string diagnostic = "ERR004 - " 24705 | 24706 | switch (lexer()[i].type) 24707 | { 24708 | case lexer::token::e_error : diagnostic += "General token error" 24709 | break; 24710 | 24711 | case lexer::token::e_err_symbol : diagnostic += "Symbol error" 24712 | break; 24713 | 24714 | case lexer::token::e_err_number : diagnostic += "Invalid numeric token" 24715 | break; 24716 | 24717 | case lexer::token::e_err_string : diagnostic += "Invalid string token" 24718 | break; 24719 | 24720 | case lexer::token::e_err_sfunc : diagnostic += "Invalid special function token" 24721 | break; 24722 | 24723 | default : diagnostic += "Unknown compiler error" 24724 | } 24725 | 24726 | set_error(make_error( 24727 | parser_error::e_lexer, 24728 | lexer()[i], 24729 | diagnostic + ": " + lexer()[i].value, 24730 | exprtk_error_location)); 24731 | } 24732 | } 24733 | } 24734 | 24735 | inline bool run_assemblies() 24736 | { 24737 | if (settings_.commutative_check_enabled()) 24738 | { 24739 | helper_assembly_.run_inserters(lexer()); 24740 | } 24741 | 24742 | if (settings_.joiner_enabled()) 24743 | { 24744 | helper_assembly_.run_joiners(lexer()); 24745 | } 24746 | 24747 | if (settings_.replacer_enabled()) 24748 | { 24749 | helper_assembly_.run_modifiers(lexer()); 24750 | } 24751 | 24752 | if ( 24753 | settings_.numeric_check_enabled () || 24754 | settings_.bracket_check_enabled () || 24755 | settings_.sequence_check_enabled() 24756 | ) 24757 | { 24758 | if (!helper_assembly_.run_scanners(lexer())) 24759 | { 24760 | if (helper_assembly_.error_token_scanner) 24761 | { 24762 | lexer::helper::bracket_checker* bracket_checker_ptr = 0; 24763 | lexer::helper::numeric_checker<T>* numeric_checker_ptr = 0; 24764 | lexer::helper::sequence_validator* sequence_validator_ptr = 0; 24765 | lexer::helper::sequence_validator_3tokens* sequence_validator3_ptr = 0; 24766 | 24767 | if (0 != (bracket_checker_ptr = dynamic_cast<lexer::helper::bracket_checker*>(helper_assembly_.error_token_scanner))) 24768 | { 24769 | set_error(make_error( 24770 | parser_error::e_token, 24771 | bracket_checker_ptr->error_token(), 24772 | "ERR005 - Mismatched brackets: '" + bracket_checker_ptr->error_token().value + "'", 24773 | exprtk_error_location)); 24774 | } 24775 | else if (0 != (numeric_checker_ptr = dynamic_cast<lexer::helper::numeric_checker<T>*>(helper_assembly_.error_token_scanner))) 24776 | { 24777 | for (std::size_t i = 0; i < numeric_checker_ptr->error_count(); ++i) 24778 | { 24779 | lexer::token error_token = lexer()[numeric_checker_ptr->error_index(i)]; 24780 | 24781 | set_error(make_error( 24782 | parser_error::e_token, 24783 | error_token, 24784 | "ERR006 - Invalid numeric token: '" + error_token.value + "'", 24785 | exprtk_error_location)); 24786 | } 24787 | 24788 | if (numeric_checker_ptr->error_count()) 24789 | { 24790 | numeric_checker_ptr->clear_errors(); 24791 | } 24792 | } 24793 | else if (0 != (sequence_validator_ptr = dynamic_cast<lexer::helper::sequence_validator*>(helper_assembly_.error_token_scanner))) 24794 | { 24795 | for (std::size_t i = 0; i < sequence_validator_ptr->error_count(); ++i) 24796 | { 24797 | std::pair<lexer::token,lexer::token> error_token = sequence_validator_ptr->error(i); 24798 | 24799 | set_error(make_error( 24800 | parser_error::e_token, 24801 | error_token.first, 24802 | "ERR007 - Invalid token sequence: '" + 24803 | error_token.first.value + "' and '" + 24804 | error_token.second.value + "'", 24805 | exprtk_error_location)); 24806 | } 24807 | 24808 | if (sequence_validator_ptr->error_count()) 24809 | { 24810 | sequence_validator_ptr->clear_errors(); 24811 | } 24812 | } 24813 | else if (0 != (sequence_validator3_ptr = dynamic_cast<lexer::helper::sequence_validator_3tokens*>(helper_assembly_.error_token_scanner))) 24814 | { 24815 | for (std::size_t i = 0; i < sequence_validator3_ptr->error_count(); ++i) 24816 | { 24817 | std::pair<lexer::token,lexer::token> error_token = sequence_validator3_ptr->error(i); 24818 | 24819 | set_error(make_error( 24820 | parser_error::e_token, 24821 | error_token.first, 24822 | "ERR008 - Invalid token sequence: '" + 24823 | error_token.first.value + "' and '" + 24824 | error_token.second.value + "'", 24825 | exprtk_error_location)); 24826 | } 24827 | 24828 | if (sequence_validator3_ptr->error_count()) 24829 | { 24830 | sequence_validator3_ptr->clear_errors(); 24831 | } 24832 | } 24833 | } 24834 | 24835 | return false; 24836 | } 24837 | } 24838 | 24839 | return true; 24840 | } 24841 | 24842 | inline settings_store& settings() 24843 | { 24844 | return settings_; 24845 | } 24846 | 24847 | inline parser_error::type get_error(const std::size_t& index) const 24848 | { 24849 | if (index < error_list_.size()) 24850 | { 24851 | return error_list_[index]; 24852 | } 24853 | 24854 | throw std::invalid_argument("parser::get_error() - Invalid error index specified"); 24855 | } 24856 | 24857 | inline std::string error() const 24858 | { 24859 | if (!error_list_.empty()) 24860 | { 24861 | return error_list_[0].diagnostic; 24862 | } 24863 | else 24864 | return std::string("No Error"); 24865 | } 24866 | 24867 | inline std::size_t error_count() const 24868 | { 24869 | return error_list_.size(); 24870 | } 24871 | 24872 | inline dependent_entity_collector& dec() 24873 | { 24874 | return dec_; 24875 | } 24876 | 24877 | inline std::size_t total_local_symbol_size_bytes() const 24878 | { 24879 | return sem_.total_local_symb_size_bytes(); 24880 | } 24881 | 24882 | inline bool replace_symbol(const std::string& old_symbol, const std::string& new_symbol) 24883 | { 24884 | if (!settings_.replacer_enabled()) 24885 | return false; 24886 | else if (details::is_reserved_word(old_symbol)) 24887 | return false; 24888 | else 24889 | return symbol_replacer_.add_replace(old_symbol,new_symbol,lexer::token::e_symbol); 24890 | } 24891 | 24892 | inline bool remove_replace_symbol(const std::string& symbol) 24893 | { 24894 | if (!settings_.replacer_enabled()) 24895 | return false; 24896 | else if (details::is_reserved_word(symbol)) 24897 | return false; 24898 | else 24899 | return symbol_replacer_.remove(symbol); 24900 | } 24901 | 24902 | inline void enable_unknown_symbol_resolver(unknown_symbol_resolver* usr = reinterpret_cast<unknown_symbol_resolver*>(0)) 24903 | { 24904 | resolve_unknown_symbol_ = true; 24905 | 24906 | if (usr) 24907 | unknown_symbol_resolver_ = usr; 24908 | else 24909 | unknown_symbol_resolver_ = &default_usr_; 24910 | } 24911 | 24912 | inline void enable_unknown_symbol_resolver(unknown_symbol_resolver& usr) 24913 | { 24914 | enable_unknown_symbol_resolver(&usr); 24915 | } 24916 | 24917 | inline void disable_unknown_symbol_resolver() 24918 | { 24919 | resolve_unknown_symbol_ = false; 24920 | unknown_symbol_resolver_ = &default_usr_; 24921 | } 24922 | 24923 | inline void register_loop_runtime_check(loop_runtime_check& lrtchk) 24924 | { 24925 | loop_runtime_check_ = &lrtchk; 24926 | } 24927 | 24928 | inline void register_vector_access_runtime_check(vector_access_runtime_check& vartchk) 24929 | { 24930 | vector_access_runtime_check_ = &vartchk; 24931 | } 24932 | 24933 | inline void register_compilation_timeout_check(compilation_check& compchk) 24934 | { 24935 | compilation_check_ptr_ = &compchk; 24936 | } 24937 | 24938 | inline void register_assert_check(assert_check& assrt_chck) 24939 | { 24940 | assert_check_ = &assrt_chck; 24941 | } 24942 | 24943 | inline void clear_loop_runtime_check() 24944 | { 24945 | loop_runtime_check_ = loop_runtime_check_ptr(0); 24946 | } 24947 | 24948 | inline void clear_vector_access_runtime_check() 24949 | { 24950 | vector_access_runtime_check_ = vector_access_runtime_check_ptr(0); 24951 | } 24952 | 24953 | inline void clear_compilation_timeout_check() 24954 | { 24955 | compilation_check_ptr_ = compilation_check_ptr(0); 24956 | } 24957 | 24958 | inline void clear_assert_check() 24959 | { 24960 | assert_check_ = assert_check_ptr(0); 24961 | } 24962 | 24963 | private: 24964 | 24965 | inline bool valid_base_operation(const std::string& symbol) const 24966 | { 24967 | const std::size_t length = symbol.size(); 24968 | 24969 | if ( 24970 | (length < 3) || // Shortest base op symbol length 24971 | (length > 9) // Longest base op symbol length 24972 | ) 24973 | return false; 24974 | else 24975 | return settings_.function_enabled(symbol) && 24976 | (base_ops_map_.end() != base_ops_map_.find(symbol)); 24977 | } 24978 | 24979 | inline bool valid_vararg_operation(const std::string& symbol) const 24980 | { 24981 | static const std::string s_sum = "sum" ; 24982 | static const std::string s_mul = "mul" ; 24983 | static const std::string s_avg = "avg" ; 24984 | static const std::string s_min = "min" ; 24985 | static const std::string s_max = "max" ; 24986 | static const std::string s_mand = "mand" 24987 | static const std::string s_mor = "mor" ; 24988 | static const std::string s_multi = "~" ; 24989 | static const std::string s_mswitch = "[*]" ; 24990 | 24991 | return 24992 | ( 24993 | details::imatch(symbol,s_sum ) || 24994 | details::imatch(symbol,s_mul ) || 24995 | details::imatch(symbol,s_avg ) || 24996 | details::imatch(symbol,s_min ) || 24997 | details::imatch(symbol,s_max ) || 24998 | details::imatch(symbol,s_mand ) || 24999 | details::imatch(symbol,s_mor ) || 25000 | details::imatch(symbol,s_multi ) || 25001 | details::imatch(symbol,s_mswitch) 25002 | ) && 25003 | settings_.function_enabled(symbol); 25004 | } 25005 | 25006 | bool is_invalid_logic_operation(const details::operator_type operation) const 25007 | { 25008 | return settings_.logic_disabled(operation); 25009 | } 25010 | 25011 | bool is_invalid_arithmetic_operation(const details::operator_type operation) const 25012 | { 25013 | return settings_.arithmetic_disabled(operation); 25014 | } 25015 | 25016 | bool is_invalid_assignment_operation(const details::operator_type operation) const 25017 | { 25018 | return settings_.assignment_disabled(operation); 25019 | } 25020 | 25021 | bool is_invalid_inequality_operation(const details::operator_type operation) const 25022 | { 25023 | return settings_.inequality_disabled(operation); 25024 | } 25025 | 25026 | #ifdef exprtk_enable_debugging 25027 | inline void next_token() 25028 | { 25029 | const std::string ct_str = current_token().value; 25030 | const std::size_t ct_pos = current_token().position; 25031 | parser_helper::next_token(); 25032 | const std::string depth(2 * state_.scope_depth,' '); 25033 | exprtk_debug(("%s" 25034 | "prev[%s | %04d] --> curr[%s | %04d] stack_level: %3d\n", 25035 | depth.c_str(), 25036 | ct_str.c_str(), 25037 | static_cast<unsigned int>(ct_pos), 25038 | current_token().value.c_str(), 25039 | static_cast<unsigned int>(current_token().position), 25040 | static_cast<unsigned int>(state_.stack_depth))); 25041 | } 25042 | #endif 25043 | 25044 | inline expression_node_ptr parse_corpus() 25045 | { 25046 | std::vector<expression_node_ptr> arg_list; 25047 | std::vector<bool> side_effect_list; 25048 | 25049 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 25050 | 25051 | lexer::token begin_token; 25052 | lexer::token end_token; 25053 | 25054 | for ( ; ; ) 25055 | { 25056 | state_.side_effect_present = false; 25057 | 25058 | begin_token = current_token(); 25059 | 25060 | expression_node_ptr arg = parse_expression(); 25061 | 25062 | if (0 == arg) 25063 | { 25064 | if (error_list_.empty()) 25065 | { 25066 | set_error(make_error( 25067 | parser_error::e_syntax, 25068 | current_token(), 25069 | "ERR009 - Invalid expression encountered", 25070 | exprtk_error_location)); 25071 | } 25072 | 25073 | return error_node(); 25074 | } 25075 | else 25076 | { 25077 | arg_list.push_back(arg); 25078 | 25079 | side_effect_list.push_back(state_.side_effect_present); 25080 | 25081 | end_token = current_token(); 25082 | 25083 | const std::string sub_expr = construct_subexpr(begin_token, end_token); 25084 | 25085 | exprtk_debug(("parse_corpus(%02d) Subexpr: %s\n", 25086 | static_cast<int>(arg_list.size() - 1), 25087 | sub_expr.c_str())); 25088 | 25089 | exprtk_debug(("parse_corpus(%02d) - Side effect present: %s\n", 25090 | static_cast<int>(arg_list.size() - 1), 25091 | state_.side_effect_present ? "true" : "false")); 25092 | 25093 | exprtk_debug(("-------------------------------------------------\n")); 25094 | } 25095 | 25096 | if (token_is(token_t::e_eof,prsrhlpr_t::e_hold)) 25097 | { 25098 | if (lexer().finished()) 25099 | break; 25100 | else 25101 | next_token(); 25102 | } 25103 | else if ( 25104 | !settings_.commutative_check_enabled() && 25105 | ( 25106 | current_token().type == token_t::e_symbol || 25107 | current_token().type == token_t::e_number || 25108 | current_token().type == token_t::e_string || 25109 | token_is_bracket(prsrhlpr_t::e_hold) 25110 | ) 25111 | ) 25112 | { 25113 | set_error(make_error( 25114 | parser_error::e_syntax, 25115 | current_token(), 25116 | "ERR010 - Invalid syntax '" + current_token().value + "' possible missing operator or context", 25117 | exprtk_error_location)); 25118 | 25119 | return error_node(); 25120 | } 25121 | } 25122 | 25123 | if ( 25124 | !arg_list.empty() && 25125 | is_return_node(arg_list.back()) 25126 | ) 25127 | { 25128 | dec_.final_stmt_return_ = true; 25129 | } 25130 | 25131 | const expression_node_ptr result = simplify(arg_list,side_effect_list); 25132 | 25133 | sdd.delete_ptr = (0 == result); 25134 | 25135 | return result; 25136 | } 25137 | 25138 | std::string construct_subexpr(lexer::token& begin_token, 25139 | lexer::token& end_token, 25140 | const bool cleanup_whitespace = true) 25141 | { 25142 | std::string result = lexer().substr(begin_token.position,end_token.position); 25143 | if (cleanup_whitespace) 25144 | { 25145 | for (std::size_t i = 0; i < result.size(); ++i) 25146 | { 25147 | if (details::is_whitespace(result[i])) result[i] = ' '; 25148 | } 25149 | } 25150 | 25151 | return result; 25152 | } 25153 | 25154 | static const precedence_level default_precedence = e_level00; 25155 | 25156 | struct state_t 25157 | { 25158 | inline void set(const precedence_level& l, 25159 | const precedence_level& r, 25160 | const details::operator_type& o, 25161 | const token_t tkn = token_t()) 25162 | { 25163 | left = l; 25164 | right = r; 25165 | operation = o; 25166 | token = tkn; 25167 | } 25168 | 25169 | inline void reset() 25170 | { 25171 | left = e_level00; 25172 | right = e_level00; 25173 | operation = details::e_default; 25174 | } 25175 | 25176 | precedence_level left; 25177 | precedence_level right; 25178 | details::operator_type operation; 25179 | token_t token; 25180 | }; 25181 | 25182 | inline void push_current_state(const state_t current_state) 25183 | { 25184 | current_state_stack_.push_back(current_state); 25185 | } 25186 | 25187 | inline void pop_current_state() 25188 | { 25189 | if (!current_state_stack_.empty()) 25190 | { 25191 | current_state_stack_.pop_back(); 25192 | } 25193 | } 25194 | 25195 | inline state_t current_state() const 25196 | { 25197 | return (!current_state_stack_.empty()) ? 25198 | current_state_stack_.back() : 25199 | state_t(); 25200 | } 25201 | 25202 | inline bool halt_compilation_check() 25203 | { 25204 | compilation_check::compilation_context context; 25205 | 25206 | if (compilation_check_ptr_ && !compilation_check_ptr_->continue_compilation(context)) 25207 | { 25208 | const std::string error_message = 25209 | !context.error_message.empty() ? " Details: " + context.error_message : "" 25210 | 25211 | set_error(make_error( 25212 | parser_error::e_parser, 25213 | token_t(), 25214 | "ERR011 - Internal compilation check failed." + error_message, 25215 | exprtk_error_location)); 25216 | 25217 | return true; 25218 | } 25219 | 25220 | return false; 25221 | } 25222 | 25223 | inline expression_node_ptr parse_expression(precedence_level precedence = e_level00) 25224 | { 25225 | if (halt_compilation_check()) 25226 | { 25227 | exprtk_debug(("halt_compilation_check() - parse_expression checkpoint 2\n")); 25228 | return error_node(); 25229 | } 25230 | 25231 | stack_limit_handler slh(*this); 25232 | 25233 | if (!slh) 25234 | { 25235 | return error_node(); 25236 | } 25237 | 25238 | expression_node_ptr expression = parse_branch(precedence); 25239 | 25240 | if (0 == expression) 25241 | { 25242 | return error_node(); 25243 | } 25244 | 25245 | if (token_is(token_t::e_eof,prsrhlpr_t::e_hold)) 25246 | { 25247 | return expression; 25248 | } 25249 | 25250 | bool break_loop = false; 25251 | 25252 | state_t current_state; 25253 | 25254 | for ( ; ; ) 25255 | { 25256 | current_state.reset(); 25257 | 25258 | switch (current_token().type) 25259 | { 25260 | case token_t::e_assign : current_state.set(e_level00, e_level00, details::e_assign, current_token()); break; 25261 | case token_t::e_addass : current_state.set(e_level00, e_level00, details::e_addass, current_token()); break; 25262 | case token_t::e_subass : current_state.set(e_level00, e_level00, details::e_subass, current_token()); break; 25263 | case token_t::e_mulass : current_state.set(e_level00, e_level00, details::e_mulass, current_token()); break; 25264 | case token_t::e_divass : current_state.set(e_level00, e_level00, details::e_divass, current_token()); break; 25265 | case token_t::e_modass : current_state.set(e_level00, e_level00, details::e_modass, current_token()); break; 25266 | case token_t::e_swap : current_state.set(e_level00, e_level00, details::e_swap , current_token()); break; 25267 | case token_t::e_lt : current_state.set(e_level05, e_level06, details::e_lt , current_token()); break; 25268 | case token_t::e_lte : current_state.set(e_level05, e_level06, details::e_lte , current_token()); break; 25269 | case token_t::e_eq : current_state.set(e_level05, e_level06, details::e_eq , current_token()); break; 25270 | case token_t::e_ne : current_state.set(e_level05, e_level06, details::e_ne , current_token()); break; 25271 | case token_t::e_gte : current_state.set(e_level05, e_level06, details::e_gte , current_token()); break; 25272 | case token_t::e_gt : current_state.set(e_level05, e_level06, details::e_gt , current_token()); break; 25273 | case token_t::e_add : current_state.set(e_level07, e_level08, details::e_add , current_token()); break; 25274 | case token_t::e_sub : current_state.set(e_level07, e_level08, details::e_sub , current_token()); break; 25275 | case token_t::e_div : current_state.set(e_level10, e_level11, details::e_div , current_token()); break; 25276 | case token_t::e_mul : current_state.set(e_level10, e_level11, details::e_mul , current_token()); break; 25277 | case token_t::e_mod : current_state.set(e_level10, e_level11, details::e_mod , current_token()); break; 25278 | case token_t::e_pow : current_state.set(e_level12, e_level12, details::e_pow , current_token()); break; 25279 | default : 25280 | if (token_t::e_symbol == current_token().type) 25281 | { 25282 | static const std::string s_and = "and" ; 25283 | static const std::string s_nand = "nand" ; 25284 | static const std::string s_or = "or" ; 25285 | static const std::string s_nor = "nor" ; 25286 | static const std::string s_xor = "xor" ; 25287 | static const std::string s_xnor = "xnor" ; 25288 | static const std::string s_in = "in" ; 25289 | static const std::string s_like = "like" ; 25290 | static const std::string s_ilike = "ilike" 25291 | static const std::string s_and1 = "&" ; 25292 | static const std::string s_or1 = "|" ; 25293 | static const std::string s_not = "not" ; 25294 | 25295 | if (details::imatch(current_token().value,s_and)) 25296 | { 25297 | current_state.set(e_level03, e_level04, details::e_and, current_token()); 25298 | break; 25299 | } 25300 | else if (details::imatch(current_token().value,s_and1)) 25301 | { 25302 | #ifndef exprtk_disable_sc_andor 25303 | current_state.set(e_level03, e_level04, details::e_scand, current_token()); 25304 | #else 25305 | current_state.set(e_level03, e_level04, details::e_and, current_token()); 25306 | #endif 25307 | break; 25308 | } 25309 | else if (details::imatch(current_token().value,s_nand)) 25310 | { 25311 | current_state.set(e_level03, e_level04, details::e_nand, current_token()); 25312 | break; 25313 | } 25314 | else if (details::imatch(current_token().value,s_or)) 25315 | { 25316 | current_state.set(e_level01, e_level02, details::e_or, current_token()); 25317 | break; 25318 | } 25319 | else if (details::imatch(current_token().value,s_or1)) 25320 | { 25321 | #ifndef exprtk_disable_sc_andor 25322 | current_state.set(e_level01, e_level02, details::e_scor, current_token()); 25323 | #else 25324 | current_state.set(e_level01, e_level02, details::e_or, current_token()); 25325 | #endif 25326 | break; 25327 | } 25328 | else if (details::imatch(current_token().value,s_nor)) 25329 | { 25330 | current_state.set(e_level01, e_level02, details::e_nor, current_token()); 25331 | break; 25332 | } 25333 | else if (details::imatch(current_token().value,s_xor)) 25334 | { 25335 | current_state.set(e_level01, e_level02, details::e_xor, current_token()); 25336 | break; 25337 | } 25338 | else if (details::imatch(current_token().value,s_xnor)) 25339 | { 25340 | current_state.set(e_level01, e_level02, details::e_xnor, current_token()); 25341 | break; 25342 | } 25343 | else if (details::imatch(current_token().value,s_in)) 25344 | { 25345 | current_state.set(e_level04, e_level04, details::e_in, current_token()); 25346 | break; 25347 | } 25348 | else if (details::imatch(current_token().value,s_like)) 25349 | { 25350 | current_state.set(e_level04, e_level04, details::e_like, current_token()); 25351 | break; 25352 | } 25353 | else if (details::imatch(current_token().value,s_ilike)) 25354 | { 25355 | current_state.set(e_level04, e_level04, details::e_ilike, current_token()); 25356 | break; 25357 | } 25358 | else if (details::imatch(current_token().value,s_not)) 25359 | { 25360 | break; 25361 | } 25362 | } 25363 | 25364 | break_loop = true; 25365 | } 25366 | 25367 | if (break_loop) 25368 | { 25369 | parse_pending_string_rangesize(expression); 25370 | break; 25371 | } 25372 | else if (current_state.left < precedence) 25373 | break; 25374 | 25375 | const lexer::token prev_token = current_token(); 25376 | 25377 | next_token(); 25378 | 25379 | expression_node_ptr right_branch = error_node(); 25380 | expression_node_ptr new_expression = error_node(); 25381 | 25382 | if (is_invalid_logic_operation(current_state.operation)) 25383 | { 25384 | free_node(node_allocator_, expression); 25385 | 25386 | set_error(make_error( 25387 | parser_error::e_syntax, 25388 | prev_token, 25389 | "ERR012 - Invalid or disabled logic operation '" + details::to_str(current_state.operation) + "'", 25390 | exprtk_error_location)); 25391 | 25392 | return error_node(); 25393 | } 25394 | else if (is_invalid_arithmetic_operation(current_state.operation)) 25395 | { 25396 | free_node(node_allocator_, expression); 25397 | 25398 | set_error(make_error( 25399 | parser_error::e_syntax, 25400 | prev_token, 25401 | "ERR013 - Invalid or disabled arithmetic operation '" + details::to_str(current_state.operation) + "'", 25402 | exprtk_error_location)); 25403 | 25404 | return error_node(); 25405 | } 25406 | else if (is_invalid_inequality_operation(current_state.operation)) 25407 | { 25408 | free_node(node_allocator_, expression); 25409 | 25410 | set_error(make_error( 25411 | parser_error::e_syntax, 25412 | prev_token, 25413 | "ERR014 - Invalid inequality operation '" + details::to_str(current_state.operation) + "'", 25414 | exprtk_error_location)); 25415 | 25416 | return error_node(); 25417 | } 25418 | else if (is_invalid_assignment_operation(current_state.operation)) 25419 | { 25420 | free_node(node_allocator_, expression); 25421 | 25422 | set_error(make_error( 25423 | parser_error::e_syntax, 25424 | prev_token, 25425 | "ERR015 - Invalid or disabled assignment operation '" + details::to_str(current_state.operation) + "'", 25426 | exprtk_error_location)); 25427 | 25428 | return error_node(); 25429 | } 25430 | 25431 | if (0 != (right_branch = parse_expression(current_state.right))) 25432 | { 25433 | if ( 25434 | details::is_return_node(expression ) || 25435 | details::is_return_node(right_branch) 25436 | ) 25437 | { 25438 | free_node(node_allocator_, expression ); 25439 | free_node(node_allocator_, right_branch); 25440 | 25441 | set_error(make_error( 25442 | parser_error::e_syntax, 25443 | prev_token, 25444 | "ERR016 - Return statements cannot be part of sub-expressions", 25445 | exprtk_error_location)); 25446 | 25447 | return error_node(); 25448 | } 25449 | 25450 | push_current_state(current_state); 25451 | 25452 | new_expression = expression_generator_ 25453 | ( 25454 | current_state.operation, 25455 | expression, 25456 | right_branch 25457 | ); 25458 | 25459 | pop_current_state(); 25460 | } 25461 | 25462 | if (0 == new_expression) 25463 | { 25464 | if (error_list_.empty()) 25465 | { 25466 | set_error(make_error( 25467 | parser_error::e_syntax, 25468 | prev_token, 25469 | !synthesis_error_.empty() ? 25470 | synthesis_error_ : 25471 | "ERR017 - General parsing error at token: '" + prev_token.value + "'", 25472 | exprtk_error_location)); 25473 | } 25474 | 25475 | free_node(node_allocator_, expression ); 25476 | free_node(node_allocator_, right_branch); 25477 | 25478 | return error_node(); 25479 | } 25480 | else 25481 | { 25482 | if ( 25483 | token_is(token_t::e_ternary,prsrhlpr_t::e_hold) && 25484 | (e_level00 == precedence) 25485 | ) 25486 | { 25487 | expression = parse_ternary_conditional_statement(new_expression); 25488 | } 25489 | else 25490 | expression = new_expression; 25491 | 25492 | parse_pending_string_rangesize(expression); 25493 | } 25494 | } 25495 | 25496 | if ((0 != expression) && (expression->node_depth() > settings_.max_node_depth_)) 25497 | { 25498 | set_error(make_error( 25499 | parser_error::e_syntax, 25500 | current_token(), 25501 | "ERR018 - Expression depth of " + details::to_str(static_cast<int>(expression->node_depth())) + 25502 | " exceeds maximum allowed expression depth of " + details::to_str(static_cast<int>(settings_.max_node_depth_)), 25503 | exprtk_error_location)); 25504 | 25505 | free_node(node_allocator_, expression); 25506 | 25507 | return error_node(); 25508 | } 25509 | else if ( 25510 | !settings_.commutative_check_enabled() && 25511 | !details::is_logic_opr(current_token().value) && 25512 | (current_state.operation == details::e_default) && 25513 | ( 25514 | current_token().type == token_t::e_symbol || 25515 | current_token().type == token_t::e_number || 25516 | current_token().type == token_t::e_string 25517 | ) 25518 | ) 25519 | { 25520 | set_error(make_error( 25521 | parser_error::e_syntax, 25522 | current_token(), 25523 | "ERR019 - Invalid syntax '" + current_token().value + "' possible missing operator or context", 25524 | exprtk_error_location)); 25525 | 25526 | free_node(node_allocator_, expression); 25527 | 25528 | return error_node(); 25529 | } 25530 | 25531 | return expression; 25532 | } 25533 | 25534 | bool simplify_unary_negation_branch(expression_node_ptr& node) 25535 | { 25536 | { 25537 | typedef details::unary_branch_node<T,details::neg_op<T> > ubn_t; 25538 | ubn_t* n = dynamic_cast<ubn_t*>(node); 25539 | 25540 | if (n) 25541 | { 25542 | expression_node_ptr un_r = n->branch(0); 25543 | n->release(); 25544 | free_node(node_allocator_, node); 25545 | node = un_r; 25546 | 25547 | return true; 25548 | } 25549 | } 25550 | 25551 | { 25552 | typedef details::unary_variable_node<T,details::neg_op<T> > uvn_t; 25553 | 25554 | uvn_t* n = dynamic_cast<uvn_t*>(node); 25555 | 25556 | if (n) 25557 | { 25558 | const T& v = n->v(); 25559 | expression_node_ptr return_node = error_node(); 25560 | 25561 | if ( 25562 | (0 != (return_node = symtab_store_.get_variable(v))) || 25563 | (0 != (return_node = sem_ .get_variable(v))) 25564 | ) 25565 | { 25566 | free_node(node_allocator_, node); 25567 | node = return_node; 25568 | 25569 | return true; 25570 | } 25571 | else 25572 | { 25573 | set_error(make_error( 25574 | parser_error::e_syntax, 25575 | current_token(), 25576 | "ERR020 - Failed to find variable node in symbol table", 25577 | exprtk_error_location)); 25578 | 25579 | free_node(node_allocator_, node); 25580 | 25581 | return false; 25582 | } 25583 | } 25584 | } 25585 | 25586 | return false; 25587 | } 25588 | 25589 | static inline expression_node_ptr error_node() 25590 | { 25591 | return reinterpret_cast<expression_node_ptr>(0); 25592 | } 25593 | 25594 | struct scoped_expression_delete 25595 | { 25596 | scoped_expression_delete(parser<T>& pr, expression_node_ptr& expression) 25597 | : delete_ptr(true) 25598 | , parser_(pr) 25599 | , expression_(expression) 25600 | {} 25601 | 25602 | ~scoped_expression_delete() 25603 | { 25604 | if (delete_ptr) 25605 | { 25606 | free_node(parser_.node_allocator_, expression_); 25607 | } 25608 | } 25609 | 25610 | bool delete_ptr; 25611 | parser<T>& parser_; 25612 | expression_node_ptr& expression_; 25613 | 25614 | private: 25615 | 25616 | scoped_expression_delete(const scoped_expression_delete&) exprtk_delete; 25617 | scoped_expression_delete& operator=(const scoped_expression_delete&) exprtk_delete; 25618 | }; 25619 | 25620 | template <typename Type, std::size_t N> 25621 | struct scoped_delete 25622 | { 25623 | typedef Type* ptr_t; 25624 | 25625 | scoped_delete(parser<T>& pr, ptr_t& p) 25626 | : delete_ptr(true) 25627 | , parser_(pr) 25628 | , p_(&p) 25629 | {} 25630 | 25631 | scoped_delete(parser<T>& pr, ptr_t (&p)[N]) 25632 | : delete_ptr(true) 25633 | , parser_(pr) 25634 | , p_(&p[0]) 25635 | {} 25636 | 25637 | ~scoped_delete() 25638 | { 25639 | if (delete_ptr) 25640 | { 25641 | for (std::size_t i = 0; i < N; ++i) 25642 | { 25643 | free_node(parser_.node_allocator_, p_[i]); 25644 | } 25645 | } 25646 | } 25647 | 25648 | bool delete_ptr; 25649 | parser<T>& parser_; 25650 | ptr_t* p_; 25651 | 25652 | private: 25653 | 25654 | scoped_delete(const scoped_delete<Type,N>&) exprtk_delete; 25655 | scoped_delete<Type,N>& operator=(const scoped_delete<Type,N>&) exprtk_delete; 25656 | }; 25657 | 25658 | template <typename Type> 25659 | struct scoped_deq_delete 25660 | { 25661 | typedef Type* ptr_t; 25662 | 25663 | scoped_deq_delete(parser<T>& pr, std::deque<ptr_t>& deq) 25664 | : delete_ptr(true) 25665 | , parser_(pr) 25666 | , deq_(deq) 25667 | {} 25668 | 25669 | ~scoped_deq_delete() 25670 | { 25671 | if (delete_ptr && !deq_.empty()) 25672 | { 25673 | for (std::size_t i = 0; i < deq_.size(); ++i) 25674 | { 25675 | exprtk_debug(("~scoped_deq_delete() - deleting node: %p\n", reinterpret_cast<void*>(deq_[i]))); 25676 | free_node(parser_.node_allocator_,deq_[i]); 25677 | } 25678 | 25679 | deq_.clear(); 25680 | } 25681 | } 25682 | 25683 | bool delete_ptr; 25684 | parser<T>& parser_; 25685 | std::deque<ptr_t>& deq_; 25686 | 25687 | private: 25688 | 25689 | scoped_deq_delete(const scoped_deq_delete<Type>&) exprtk_delete; 25690 | scoped_deq_delete<Type>& operator=(const scoped_deq_delete<Type>&) exprtk_delete; 25691 | }; 25692 | 25693 | template <typename Type> 25694 | struct scoped_vec_delete 25695 | { 25696 | typedef Type* ptr_t; 25697 | 25698 | scoped_vec_delete(parser<T>& pr, std::vector<ptr_t>& vec) 25699 | : delete_ptr(true) 25700 | , parser_(pr) 25701 | , vec_(vec) 25702 | {} 25703 | 25704 | ~scoped_vec_delete() 25705 | { 25706 | if (delete_ptr && !vec_.empty()) 25707 | { 25708 | for (std::size_t i = 0; i < vec_.size(); ++i) 25709 | { 25710 | exprtk_debug(("~scoped_vec_delete() - deleting node: %p\n", reinterpret_cast<void*>(vec_[i]))); 25711 | free_node(parser_.node_allocator_,vec_[i]); 25712 | } 25713 | 25714 | vec_.clear(); 25715 | } 25716 | } 25717 | 25718 | ptr_t operator[](const std::size_t index) 25719 | { 25720 | return vec_[index]; 25721 | } 25722 | 25723 | bool delete_ptr; 25724 | parser<T>& parser_; 25725 | std::vector<ptr_t>& vec_; 25726 | 25727 | private: 25728 | 25729 | scoped_vec_delete(const scoped_vec_delete<Type>&) exprtk_delete; 25730 | scoped_vec_delete<Type>& operator=(const scoped_vec_delete<Type>&) exprtk_delete; 25731 | }; 25732 | 25733 | struct scoped_bool_negator 25734 | { 25735 | explicit scoped_bool_negator(bool& bb) 25736 | : b(bb) 25737 | { b = !b; } 25738 | 25739 | ~scoped_bool_negator() 25740 | { b = !b; } 25741 | 25742 | bool& b; 25743 | }; 25744 | 25745 | struct scoped_bool_or_restorer 25746 | { 25747 | explicit scoped_bool_or_restorer(bool& bb) 25748 | : b(bb) 25749 | , original_value_(bb) 25750 | {} 25751 | 25752 | ~scoped_bool_or_restorer() 25753 | { 25754 | b = b || original_value_; 25755 | } 25756 | 25757 | bool& b; 25758 | bool original_value_; 25759 | }; 25760 | 25761 | struct scoped_inc_dec 25762 | { 25763 | explicit scoped_inc_dec(std::size_t& v) 25764 | : v_(v) 25765 | { ++v_; } 25766 | 25767 | ~scoped_inc_dec() 25768 | { 25769 | assert(v_ > 0); 25770 | --v_; 25771 | } 25772 | 25773 | std::size_t& v_; 25774 | }; 25775 | 25776 | inline expression_node_ptr parse_function_invocation(ifunction<T>* function, const std::string& function_name) 25777 | { 25778 | expression_node_ptr func_node = reinterpret_cast<expression_node_ptr>(0); 25779 | 25780 | switch (function->param_count) 25781 | { 25782 | case 0 : func_node = parse_function_call_0 (function,function_name); break; 25783 | case 1 : func_node = parse_function_call< 1>(function,function_name); break; 25784 | case 2 : func_node = parse_function_call< 2>(function,function_name); break; 25785 | case 3 : func_node = parse_function_call< 3>(function,function_name); break; 25786 | case 4 : func_node = parse_function_call< 4>(function,function_name); break; 25787 | case 5 : func_node = parse_function_call< 5>(function,function_name); break; 25788 | case 6 : func_node = parse_function_call< 6>(function,function_name); break; 25789 | case 7 : func_node = parse_function_call< 7>(function,function_name); break; 25790 | case 8 : func_node = parse_function_call< 8>(function,function_name); break; 25791 | case 9 : func_node = parse_function_call< 9>(function,function_name); break; 25792 | case 10 : func_node = parse_function_call<10>(function,function_name); break; 25793 | case 11 : func_node = parse_function_call<11>(function,function_name); break; 25794 | case 12 : func_node = parse_function_call<12>(function,function_name); break; 25795 | case 13 : func_node = parse_function_call<13>(function,function_name); break; 25796 | case 14 : func_node = parse_function_call<14>(function,function_name); break; 25797 | case 15 : func_node = parse_function_call<15>(function,function_name); break; 25798 | case 16 : func_node = parse_function_call<16>(function,function_name); break; 25799 | case 17 : func_node = parse_function_call<17>(function,function_name); break; 25800 | case 18 : func_node = parse_function_call<18>(function,function_name); break; 25801 | case 19 : func_node = parse_function_call<19>(function,function_name); break; 25802 | case 20 : func_node = parse_function_call<20>(function,function_name); break; 25803 | default : { 25804 | set_error(make_error( 25805 | parser_error::e_syntax, 25806 | current_token(), 25807 | "ERR021 - Invalid number of parameters for function: '" + function_name + "'", 25808 | exprtk_error_location)); 25809 | 25810 | return error_node(); 25811 | } 25812 | } 25813 | 25814 | if (func_node) 25815 | return func_node; 25816 | else 25817 | { 25818 | set_error(make_error( 25819 | parser_error::e_syntax, 25820 | current_token(), 25821 | "ERR022 - Failed to generate call to function: '" + function_name + "'", 25822 | exprtk_error_location)); 25823 | 25824 | return error_node(); 25825 | } 25826 | } 25827 | 25828 | template <std::size_t NumberofParameters> 25829 | inline expression_node_ptr parse_function_call(ifunction<T>* function, const std::string& function_name) 25830 | { 25831 | #ifdef _MSC_VER 25832 | #pragma warning(push) 25833 | #pragma warning(disable: 4127) 25834 | #endif 25835 | if (0 == NumberofParameters) 25836 | { 25837 | set_error(make_error( 25838 | parser_error::e_syntax, 25839 | current_token(), 25840 | "ERR023 - Expecting ifunction '" + function_name + "' to have non-zero parameter count", 25841 | exprtk_error_location)); 25842 | 25843 | return error_node(); 25844 | } 25845 | #ifdef _MSC_VER 25846 | #pragma warning(pop) 25847 | #endif 25848 | 25849 | expression_node_ptr branch[NumberofParameters]; 25850 | expression_node_ptr result = error_node(); 25851 | 25852 | std::fill_n(branch, NumberofParameters, reinterpret_cast<expression_node_ptr>(0)); 25853 | 25854 | scoped_delete<expression_node_t,NumberofParameters> sd((*this),branch); 25855 | 25856 | next_token(); 25857 | 25858 | if (!token_is(token_t::e_lbracket)) 25859 | { 25860 | set_error(make_error( 25861 | parser_error::e_syntax, 25862 | current_token(), 25863 | "ERR024 - Expecting argument list for function: '" + function_name + "'", 25864 | exprtk_error_location)); 25865 | 25866 | return error_node(); 25867 | } 25868 | 25869 | for (int i = 0; i < static_cast<int>(NumberofParameters); ++i) 25870 | { 25871 | branch[i] = parse_expression(); 25872 | 25873 | if (0 == branch[i]) 25874 | { 25875 | set_error(make_error( 25876 | parser_error::e_syntax, 25877 | current_token(), 25878 | "ERR025 - Failed to parse argument " + details::to_str(i) + " for function: '" + function_name + "'", 25879 | exprtk_error_location)); 25880 | 25881 | return error_node(); 25882 | } 25883 | else if (i < static_cast<int>(NumberofParameters - 1)) 25884 | { 25885 | if (!token_is(token_t::e_comma)) 25886 | { 25887 | set_error(make_error( 25888 | parser_error::e_syntax, 25889 | current_token(), 25890 | "ERR026 - Invalid number of arguments for function: '" + function_name + "'", 25891 | exprtk_error_location)); 25892 | 25893 | return error_node(); 25894 | } 25895 | } 25896 | } 25897 | 25898 | if (!token_is(token_t::e_rbracket)) 25899 | { 25900 | set_error(make_error( 25901 | parser_error::e_syntax, 25902 | current_token(), 25903 | "ERR027 - Invalid number of arguments for function: '" + function_name + "'", 25904 | exprtk_error_location)); 25905 | 25906 | return error_node(); 25907 | } 25908 | else 25909 | result = expression_generator_.function(function,branch); 25910 | 25911 | sd.delete_ptr = (0 == result); 25912 | 25913 | return result; 25914 | } 25915 | 25916 | inline expression_node_ptr parse_function_call_0(ifunction<T>* function, const std::string& function_name) 25917 | { 25918 | expression_node_ptr result = expression_generator_.function(function); 25919 | 25920 | state_.side_effect_present = function->has_side_effects(); 25921 | 25922 | next_token(); 25923 | 25924 | if ( 25925 | token_is(token_t::e_lbracket) && 25926 | !token_is(token_t::e_rbracket) 25927 | ) 25928 | { 25929 | set_error(make_error( 25930 | parser_error::e_syntax, 25931 | current_token(), 25932 | "ERR028 - Expecting '()' to proceed call to function: '" + function_name + "'", 25933 | exprtk_error_location)); 25934 | 25935 | free_node(node_allocator_, result); 25936 | 25937 | return error_node(); 25938 | } 25939 | else 25940 | return result; 25941 | } 25942 | 25943 | template <std::size_t MaxNumberofParameters> 25944 | inline std::size_t parse_base_function_call(expression_node_ptr (¶m_list)[MaxNumberofParameters], const std::string& function_name = "") 25945 | { 25946 | std::fill_n(param_list, MaxNumberofParameters, reinterpret_cast<expression_node_ptr>(0)); 25947 | 25948 | scoped_delete<expression_node_t,MaxNumberofParameters> sd((*this),param_list); 25949 | 25950 | next_token(); 25951 | 25952 | if (!token_is(token_t::e_lbracket)) 25953 | { 25954 | set_error(make_error( 25955 | parser_error::e_syntax, 25956 | current_token(), 25957 | "ERR029 - Expected a '(' at start of function call to '" + function_name + 25958 | "', instead got: '" + current_token().value + "'", 25959 | exprtk_error_location)); 25960 | 25961 | return 0; 25962 | } 25963 | 25964 | if (token_is(token_t::e_rbracket, e_hold)) 25965 | { 25966 | set_error(make_error( 25967 | parser_error::e_syntax, 25968 | current_token(), 25969 | "ERR030 - Expected at least one input parameter for function call '" + function_name + "'", 25970 | exprtk_error_location)); 25971 | 25972 | return 0; 25973 | } 25974 | 25975 | std::size_t param_index = 0; 25976 | 25977 | for (; param_index < MaxNumberofParameters; ++param_index) 25978 | { 25979 | param_list[param_index] = parse_expression(); 25980 | 25981 | if (0 == param_list[param_index]) 25982 | return 0; 25983 | else if (token_is(token_t::e_rbracket)) 25984 | { 25985 | sd.delete_ptr = false; 25986 | break; 25987 | } 25988 | else if (token_is(token_t::e_comma)) 25989 | continue; 25990 | else 25991 | { 25992 | set_error(make_error( 25993 | parser_error::e_syntax, 25994 | current_token(), 25995 | "ERR031 - Expected a ',' between function input parameters, instead got: '" + current_token().value + "'", 25996 | exprtk_error_location)); 25997 | 25998 | return 0; 25999 | } 26000 | } 26001 | 26002 | if (sd.delete_ptr) 26003 | { 26004 | set_error(make_error( 26005 | parser_error::e_syntax, 26006 | current_token(), 26007 | "ERR032 - Invalid number of input parameters passed to function '" + function_name + "'", 26008 | exprtk_error_location)); 26009 | 26010 | return 0; 26011 | } 26012 | 26013 | return (param_index + 1); 26014 | } 26015 | 26016 | inline expression_node_ptr parse_base_operation() 26017 | { 26018 | typedef std::pair<base_ops_map_t::iterator,base_ops_map_t::iterator> map_range_t; 26019 | 26020 | const std::string operation_name = current_token().value; 26021 | const token_t diagnostic_token = current_token(); 26022 | 26023 | map_range_t itr_range = base_ops_map_.equal_range(operation_name); 26024 | 26025 | if (0 == std::distance(itr_range.first,itr_range.second)) 26026 | { 26027 | set_error(make_error( 26028 | parser_error::e_syntax, 26029 | diagnostic_token, 26030 | "ERR033 - No entry found for base operation: " + operation_name, 26031 | exprtk_error_location)); 26032 | 26033 | return error_node(); 26034 | } 26035 | 26036 | static const std::size_t MaxNumberofParameters = 4; 26037 | expression_node_ptr param_list[MaxNumberofParameters] = {0}; 26038 | 26039 | const std::size_t parameter_count = parse_base_function_call(param_list, operation_name); 26040 | 26041 | if ((parameter_count > 0) && (parameter_count <= MaxNumberofParameters)) 26042 | { 26043 | for (base_ops_map_t::iterator itr = itr_range.first; itr != itr_range.second; ++itr) 26044 | { 26045 | const details::base_operation_t& operation = itr->second; 26046 | 26047 | if (operation.num_params == parameter_count) 26048 | { 26049 | switch (parameter_count) 26050 | { 26051 | #define base_opr_case(N) \ 26052 | case N : { \ 26053 | expression_node_ptr pl##N[N] = {0}; \ 26054 | std::copy(param_list, param_list + N, pl##N); \ 26055 | lodge_symbol(operation_name, e_st_function); \ 26056 | return expression_generator_(operation.type, pl##N); \ 26057 | } \ 26058 | 26059 | base_opr_case(1) 26060 | base_opr_case(2) 26061 | base_opr_case(3) 26062 | base_opr_case(4) 26063 | #undef base_opr_case 26064 | } 26065 | } 26066 | } 26067 | } 26068 | 26069 | for (std::size_t i = 0; i < MaxNumberofParameters; ++i) 26070 | { 26071 | free_node(node_allocator_, param_list[i]); 26072 | } 26073 | 26074 | set_error(make_error( 26075 | parser_error::e_syntax, 26076 | diagnostic_token, 26077 | "ERR034 - Invalid number of input parameters for call to function: '" + operation_name + "'", 26078 | exprtk_error_location)); 26079 | 26080 | return error_node(); 26081 | } 26082 | 26083 | inline expression_node_ptr parse_conditional_statement_01(expression_node_ptr condition) 26084 | { 26085 | // Parse: [if][(][condition][,][consequent][,][alternative][)] 26086 | 26087 | expression_node_ptr consequent = error_node(); 26088 | expression_node_ptr alternative = error_node(); 26089 | 26090 | bool result = true; 26091 | 26092 | if (!token_is(token_t::e_comma)) 26093 | { 26094 | set_error(make_error( 26095 | parser_error::e_syntax, 26096 | current_token(), 26097 | "ERR035 - Expected ',' between if-statement condition and consequent", 26098 | exprtk_error_location)); 26099 | 26100 | result = false; 26101 | } 26102 | else if (0 == (consequent = parse_expression())) 26103 | { 26104 | set_error(make_error( 26105 | parser_error::e_syntax, 26106 | current_token(), 26107 | "ERR036 - Failed to parse consequent for if-statement", 26108 | exprtk_error_location)); 26109 | 26110 | result = false; 26111 | } 26112 | else if (!token_is(token_t::e_comma)) 26113 | { 26114 | set_error(make_error( 26115 | parser_error::e_syntax, 26116 | current_token(), 26117 | "ERR037 - Expected ',' between if-statement consequent and alternative", 26118 | exprtk_error_location)); 26119 | 26120 | result = false; 26121 | } 26122 | else if (0 == (alternative = parse_expression())) 26123 | { 26124 | set_error(make_error( 26125 | parser_error::e_syntax, 26126 | current_token(), 26127 | "ERR038 - Failed to parse alternative for if-statement", 26128 | exprtk_error_location)); 26129 | 26130 | result = false; 26131 | } 26132 | else if (!token_is(token_t::e_rbracket)) 26133 | { 26134 | set_error(make_error( 26135 | parser_error::e_syntax, 26136 | current_token(), 26137 | "ERR039 - Expected ')' at the end of if-statement", 26138 | exprtk_error_location)); 26139 | 26140 | result = false; 26141 | } 26142 | 26143 | #ifndef exprtk_disable_string_capabilities 26144 | if (result) 26145 | { 26146 | const bool consq_is_str = is_generally_string_node(consequent ); 26147 | const bool alter_is_str = is_generally_string_node(alternative); 26148 | 26149 | if (consq_is_str || alter_is_str) 26150 | { 26151 | if (consq_is_str && alter_is_str) 26152 | { 26153 | expression_node_ptr result_node = 26154 | expression_generator_ 26155 | .conditional_string(condition, consequent, alternative); 26156 | 26157 | if (result_node && result_node->valid()) 26158 | { 26159 | return result_node; 26160 | } 26161 | 26162 | set_error(make_error( 26163 | parser_error::e_synthesis, 26164 | current_token(), 26165 | "ERR040 - Failed to synthesize node: conditional_string", 26166 | exprtk_error_location)); 26167 | 26168 | free_node(node_allocator_, result_node); 26169 | return error_node(); 26170 | } 26171 | 26172 | set_error(make_error( 26173 | parser_error::e_syntax, 26174 | current_token(), 26175 | "ERR041 - Return types of if-statement differ: string/non-string", 26176 | exprtk_error_location)); 26177 | 26178 | result = false; 26179 | } 26180 | } 26181 | #endif 26182 | 26183 | if (result) 26184 | { 26185 | const bool consq_is_vec = is_ivector_node(consequent ); 26186 | const bool alter_is_vec = is_ivector_node(alternative); 26187 | 26188 | if (consq_is_vec || alter_is_vec) 26189 | { 26190 | if (consq_is_vec && alter_is_vec) 26191 | { 26192 | return expression_generator_ 26193 | .conditional_vector(condition, consequent, alternative); 26194 | } 26195 | 26196 | set_error(make_error( 26197 | parser_error::e_syntax, 26198 | current_token(), 26199 | "ERR042 - Return types of if-statement differ: vector/non-vector", 26200 | exprtk_error_location)); 26201 | 26202 | result = false; 26203 | } 26204 | } 26205 | 26206 | if (!result) 26207 | { 26208 | free_node(node_allocator_, condition ); 26209 | free_node(node_allocator_, consequent ); 26210 | free_node(node_allocator_, alternative); 26211 | 26212 | return error_node(); 26213 | } 26214 | else 26215 | return expression_generator_ 26216 | .conditional(condition, consequent, alternative); 26217 | } 26218 | 26219 | inline expression_node_ptr parse_conditional_statement_02(expression_node_ptr condition) 26220 | { 26221 | expression_node_ptr consequent = error_node(); 26222 | expression_node_ptr alternative = error_node(); 26223 | 26224 | bool result = true; 26225 | 26226 | if (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) 26227 | { 26228 | if (0 == (consequent = parse_multi_sequence("if-statement-01"))) 26229 | { 26230 | set_error(make_error( 26231 | parser_error::e_syntax, 26232 | current_token(), 26233 | "ERR043 - Failed to parse body of consequent for if-statement", 26234 | exprtk_error_location)); 26235 | 26236 | result = false; 26237 | } 26238 | else if 26239 | ( 26240 | !settings_.commutative_check_enabled() && 26241 | !token_is("else",prsrhlpr_t::e_hold) && 26242 | !token_is_loop(prsrhlpr_t::e_hold) && 26243 | !token_is_arithmetic_opr(prsrhlpr_t::e_hold) && 26244 | !token_is_right_bracket (prsrhlpr_t::e_hold) && 26245 | !token_is_ineq_opr (prsrhlpr_t::e_hold) && 26246 | !token_is(token_t::e_ternary,prsrhlpr_t::e_hold) && 26247 | !token_is(token_t::e_eof ,prsrhlpr_t::e_hold) 26248 | ) 26249 | { 26250 | set_error(make_error( 26251 | parser_error::e_syntax, 26252 | current_token(), 26253 | "ERR044 - Expected ';' at the end of the consequent for if-statement (1)", 26254 | exprtk_error_location)); 26255 | 26256 | result = false; 26257 | } 26258 | } 26259 | else 26260 | { 26261 | if ( 26262 | settings_.commutative_check_enabled() && 26263 | token_is(token_t::e_mul,prsrhlpr_t::e_hold) 26264 | ) 26265 | { 26266 | next_token(); 26267 | } 26268 | 26269 | if (0 != (consequent = parse_expression())) 26270 | { 26271 | if (!token_is(token_t::e_eof, prsrhlpr_t::e_hold)) 26272 | { 26273 | set_error(make_error( 26274 | parser_error::e_syntax, 26275 | current_token(), 26276 | "ERR045 - Expected ';' at the end of the consequent for if-statement (2)", 26277 | exprtk_error_location)); 26278 | 26279 | result = false; 26280 | } 26281 | } 26282 | else 26283 | { 26284 | set_error(make_error( 26285 | parser_error::e_syntax, 26286 | current_token(), 26287 | "ERR046 - Failed to parse body of consequent for if-statement", 26288 | exprtk_error_location)); 26289 | 26290 | result = false; 26291 | } 26292 | } 26293 | 26294 | if (result) 26295 | { 26296 | if ( 26297 | details::imatch(current_token().value,"else") || 26298 | (token_is(token_t::e_eof, prsrhlpr_t::e_hold) && peek_token_is("else")) 26299 | ) 26300 | { 26301 | next_token(); 26302 | 26303 | if (details::imatch(current_token().value,"else")) 26304 | { 26305 | next_token(); 26306 | } 26307 | 26308 | if (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) 26309 | { 26310 | if (0 == (alternative = parse_multi_sequence("else-statement-01"))) 26311 | { 26312 | set_error(make_error( 26313 | parser_error::e_syntax, 26314 | current_token(), 26315 | "ERR047 - Failed to parse body of the 'else' for if-statement", 26316 | exprtk_error_location)); 26317 | 26318 | result = false; 26319 | } 26320 | } 26321 | else if (details::imatch(current_token().value,"if")) 26322 | { 26323 | if (0 == (alternative = parse_conditional_statement())) 26324 | { 26325 | set_error(make_error( 26326 | parser_error::e_syntax, 26327 | current_token(), 26328 | "ERR048 - Failed to parse body of if-else statement", 26329 | exprtk_error_location)); 26330 | 26331 | result = false; 26332 | } 26333 | } 26334 | else if (0 != (alternative = parse_expression())) 26335 | { 26336 | if ( 26337 | !token_is(token_t::e_ternary , prsrhlpr_t::e_hold) && 26338 | !token_is(token_t::e_rcrlbracket, prsrhlpr_t::e_hold) && 26339 | !token_is(token_t::e_eof) 26340 | ) 26341 | { 26342 | set_error(make_error( 26343 | parser_error::e_syntax, 26344 | current_token(), 26345 | "ERR049 - Expected ';' at the end of the 'else-if' for the if-statement", 26346 | exprtk_error_location)); 26347 | 26348 | result = false; 26349 | } 26350 | } 26351 | else 26352 | { 26353 | set_error(make_error( 26354 | parser_error::e_syntax, 26355 | current_token(), 26356 | "ERR050 - Failed to parse body of the 'else' for if-statement", 26357 | exprtk_error_location)); 26358 | 26359 | result = false; 26360 | } 26361 | } 26362 | } 26363 | 26364 | #ifndef exprtk_disable_string_capabilities 26365 | if (result) 26366 | { 26367 | const bool consq_is_str = is_generally_string_node(consequent ); 26368 | const bool alter_is_str = is_generally_string_node(alternative); 26369 | 26370 | if (consq_is_str || alter_is_str) 26371 | { 26372 | if (consq_is_str && alter_is_str) 26373 | { 26374 | return expression_generator_ 26375 | .conditional_string(condition, consequent, alternative); 26376 | } 26377 | 26378 | set_error(make_error( 26379 | parser_error::e_syntax, 26380 | current_token(), 26381 | "ERR051 - Return types of if-statement differ: string/non-string", 26382 | exprtk_error_location)); 26383 | 26384 | result = false; 26385 | } 26386 | } 26387 | #endif 26388 | 26389 | if (result) 26390 | { 26391 | const bool consq_is_vec = is_ivector_node(consequent ); 26392 | const bool alter_is_vec = is_ivector_node(alternative); 26393 | 26394 | if (consq_is_vec || alter_is_vec) 26395 | { 26396 | if (consq_is_vec && alter_is_vec) 26397 | { 26398 | return expression_generator_ 26399 | .conditional_vector(condition, consequent, alternative); 26400 | } 26401 | 26402 | set_error(make_error( 26403 | parser_error::e_syntax, 26404 | current_token(), 26405 | "ERR052 - Return types of if-statement differ: vector/non-vector", 26406 | exprtk_error_location)); 26407 | 26408 | result = false; 26409 | } 26410 | } 26411 | 26412 | if (!result) 26413 | { 26414 | free_node(node_allocator_, condition ); 26415 | free_node(node_allocator_, consequent ); 26416 | free_node(node_allocator_, alternative); 26417 | 26418 | return error_node(); 26419 | } 26420 | else 26421 | return expression_generator_ 26422 | .conditional(condition, consequent, alternative); 26423 | } 26424 | 26425 | inline expression_node_ptr parse_conditional_statement() 26426 | { 26427 | expression_node_ptr condition = error_node(); 26428 | 26429 | next_token(); 26430 | 26431 | if (!token_is(token_t::e_lbracket)) 26432 | { 26433 | set_error(make_error( 26434 | parser_error::e_syntax, 26435 | current_token(), 26436 | "ERR053 - Expected '(' at start of if-statement, instead got: '" + current_token().value + "'", 26437 | exprtk_error_location)); 26438 | 26439 | return error_node(); 26440 | } 26441 | else if (0 == (condition = parse_expression())) 26442 | { 26443 | set_error(make_error( 26444 | parser_error::e_syntax, 26445 | current_token(), 26446 | "ERR054 - Failed to parse condition for if-statement", 26447 | exprtk_error_location)); 26448 | 26449 | return error_node(); 26450 | } 26451 | else if (token_is(token_t::e_comma,prsrhlpr_t::e_hold)) 26452 | { 26453 | // if (x,y,z) 26454 | return parse_conditional_statement_01(condition); 26455 | } 26456 | else if (token_is(token_t::e_rbracket)) 26457 | { 26458 | /* 26459 | 00. if (x) y; 26460 | 01. if (x) y; else z; 26461 | 02. if (x) y; else {z0; ... zn;} 26462 | 03. if (x) y; else if (z) w; 26463 | 04. if (x) y; else if (z) w; else u; 26464 | 05. if (x) y; else if (z) w; else {u0; ... un;} 26465 | 06. if (x) y; else if (z) {w0; ... wn;} 26466 | 07. if (x) {y0; ... yn;} 26467 | 08. if (x) {y0; ... yn;} else z; 26468 | 09. if (x) {y0; ... yn;} else {z0; ... zn;}; 26469 | 10. if (x) {y0; ... yn;} else if (z) w; 26470 | 11. if (x) {y0; ... yn;} else if (z) w; else u; 26471 | 12. if (x) {y0; ... nex;} else if (z) w; else {u0 ... un;} 26472 | 13. if (x) {y0; ... yn;} else if (z) {w0; ... wn;} 26473 | */ 26474 | return parse_conditional_statement_02(condition); 26475 | } 26476 | 26477 | set_error(make_error( 26478 | parser_error::e_syntax, 26479 | current_token(), 26480 | "ERR055 - Invalid if-statement", 26481 | exprtk_error_location)); 26482 | 26483 | free_node(node_allocator_, condition); 26484 | 26485 | return error_node(); 26486 | } 26487 | 26488 | inline expression_node_ptr parse_ternary_conditional_statement(expression_node_ptr condition) 26489 | { 26490 | // Parse: [condition][?][consequent][:][alternative] 26491 | expression_node_ptr consequent = error_node(); 26492 | expression_node_ptr alternative = error_node(); 26493 | 26494 | bool result = true; 26495 | 26496 | if (0 == condition) 26497 | { 26498 | set_error(make_error( 26499 | parser_error::e_syntax, 26500 | current_token(), 26501 | "ERR056 - Encountered invalid condition branch for ternary if-statement", 26502 | exprtk_error_location)); 26503 | 26504 | return error_node(); 26505 | } 26506 | else if (!token_is(token_t::e_ternary)) 26507 | { 26508 | set_error(make_error( 26509 | parser_error::e_syntax, 26510 | current_token(), 26511 | "ERR057 - Expected '?' after condition of ternary if-statement", 26512 | exprtk_error_location)); 26513 | 26514 | result = false; 26515 | } 26516 | else if (0 == (consequent = parse_expression())) 26517 | { 26518 | set_error(make_error( 26519 | parser_error::e_syntax, 26520 | current_token(), 26521 | "ERR058 - Failed to parse consequent for ternary if-statement", 26522 | exprtk_error_location)); 26523 | 26524 | result = false; 26525 | } 26526 | else if (!token_is(token_t::e_colon)) 26527 | { 26528 | set_error(make_error( 26529 | parser_error::e_syntax, 26530 | current_token(), 26531 | "ERR059 - Expected ':' between ternary if-statement consequent and alternative", 26532 | exprtk_error_location)); 26533 | 26534 | result = false; 26535 | } 26536 | else if (0 == (alternative = parse_expression())) 26537 | { 26538 | set_error(make_error( 26539 | parser_error::e_syntax, 26540 | current_token(), 26541 | "ERR060 - Failed to parse alternative for ternary if-statement", 26542 | exprtk_error_location)); 26543 | 26544 | result = false; 26545 | } 26546 | 26547 | #ifndef exprtk_disable_string_capabilities 26548 | if (result) 26549 | { 26550 | const bool consq_is_str = is_generally_string_node(consequent ); 26551 | const bool alter_is_str = is_generally_string_node(alternative); 26552 | 26553 | if (consq_is_str || alter_is_str) 26554 | { 26555 | if (consq_is_str && alter_is_str) 26556 | { 26557 | return expression_generator_ 26558 | .conditional_string(condition, consequent, alternative); 26559 | } 26560 | 26561 | set_error(make_error( 26562 | parser_error::e_syntax, 26563 | current_token(), 26564 | "ERR061 - Return types of ternary differ: string/non-string", 26565 | exprtk_error_location)); 26566 | 26567 | result = false; 26568 | } 26569 | } 26570 | #endif 26571 | 26572 | if (result) 26573 | { 26574 | const bool consq_is_vec = is_ivector_node(consequent ); 26575 | const bool alter_is_vec = is_ivector_node(alternative); 26576 | 26577 | if (consq_is_vec || alter_is_vec) 26578 | { 26579 | if (consq_is_vec && alter_is_vec) 26580 | { 26581 | return expression_generator_ 26582 | .conditional_vector(condition, consequent, alternative); 26583 | } 26584 | 26585 | set_error(make_error( 26586 | parser_error::e_syntax, 26587 | current_token(), 26588 | "ERR062 - Return types of ternary differ: vector/non-vector", 26589 | exprtk_error_location)); 26590 | 26591 | result = false; 26592 | } 26593 | } 26594 | 26595 | if (!result) 26596 | { 26597 | free_node(node_allocator_, condition ); 26598 | free_node(node_allocator_, consequent ); 26599 | free_node(node_allocator_, alternative); 26600 | 26601 | return error_node(); 26602 | } 26603 | else 26604 | return expression_generator_ 26605 | .conditional(condition, consequent, alternative); 26606 | } 26607 | 26608 | inline expression_node_ptr parse_not_statement() 26609 | { 26610 | if (settings_.logic_disabled("not")) 26611 | { 26612 | set_error(make_error( 26613 | parser_error::e_syntax, 26614 | current_token(), 26615 | "ERR063 - Invalid or disabled logic operation 'not'", 26616 | exprtk_error_location)); 26617 | 26618 | return error_node(); 26619 | } 26620 | 26621 | return parse_base_operation(); 26622 | } 26623 | 26624 | void handle_brkcnt_scope_exit() 26625 | { 26626 | assert(!brkcnt_list_.empty()); 26627 | brkcnt_list_.pop_front(); 26628 | } 26629 | 26630 | inline expression_node_ptr parse_while_loop() 26631 | { 26632 | // Parse: [while][(][test expr][)][{][expression][}] 26633 | expression_node_ptr condition = error_node(); 26634 | expression_node_ptr branch = error_node(); 26635 | expression_node_ptr result_node = error_node(); 26636 | 26637 | bool result = true; 26638 | 26639 | next_token(); 26640 | 26641 | if (!token_is(token_t::e_lbracket)) 26642 | { 26643 | set_error(make_error( 26644 | parser_error::e_syntax, 26645 | current_token(), 26646 | "ERR064 - Expected '(' at start of while-loop condition statement", 26647 | exprtk_error_location)); 26648 | 26649 | return error_node(); 26650 | } 26651 | else if (0 == (condition = parse_expression())) 26652 | { 26653 | set_error(make_error( 26654 | parser_error::e_syntax, 26655 | current_token(), 26656 | "ERR065 - Failed to parse condition for while-loop", 26657 | exprtk_error_location)); 26658 | 26659 | return error_node(); 26660 | } 26661 | else if (!token_is(token_t::e_rbracket)) 26662 | { 26663 | set_error(make_error( 26664 | parser_error::e_syntax, 26665 | current_token(), 26666 | "ERR066 - Expected ')' at end of while-loop condition statement", 26667 | exprtk_error_location)); 26668 | 26669 | result = false; 26670 | } 26671 | 26672 | brkcnt_list_.push_front(false); 26673 | 26674 | if (result) 26675 | { 26676 | scoped_inc_dec sid(state_.parsing_loop_stmt_count); 26677 | 26678 | if (0 == (branch = parse_multi_sequence("while-loop", true))) 26679 | { 26680 | set_error(make_error( 26681 | parser_error::e_syntax, 26682 | current_token(), 26683 | "ERR067 - Failed to parse body of while-loop")); 26684 | result = false; 26685 | } 26686 | else if (0 == (result_node = expression_generator_.while_loop(condition, 26687 | branch, 26688 | brkcnt_list_.front()))) 26689 | { 26690 | set_error(make_error( 26691 | parser_error::e_syntax, 26692 | current_token(), 26693 | "ERR068 - Failed to synthesize while-loop", 26694 | exprtk_error_location)); 26695 | 26696 | result = false; 26697 | } 26698 | } 26699 | 26700 | handle_brkcnt_scope_exit(); 26701 | 26702 | if (!result) 26703 | { 26704 | free_node(node_allocator_, branch ); 26705 | free_node(node_allocator_, condition ); 26706 | free_node(node_allocator_, result_node); 26707 | 26708 | return error_node(); 26709 | } 26710 | 26711 | if (result_node && result_node->valid()) 26712 | { 26713 | return result_node; 26714 | } 26715 | 26716 | set_error(make_error( 26717 | parser_error::e_synthesis, 26718 | current_token(), 26719 | "ERR069 - Failed to synthesize 'valid' while-loop", 26720 | exprtk_error_location)); 26721 | 26722 | free_node(node_allocator_, result_node); 26723 | 26724 | return error_node(); 26725 | } 26726 | 26727 | inline expression_node_ptr parse_repeat_until_loop() 26728 | { 26729 | // Parse: [repeat][{][expression][}][until][(][test expr][)] 26730 | expression_node_ptr condition = error_node(); 26731 | expression_node_ptr branch = error_node(); 26732 | next_token(); 26733 | 26734 | std::vector<expression_node_ptr> arg_list; 26735 | std::vector<bool> side_effect_list; 26736 | 26737 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 26738 | 26739 | brkcnt_list_.push_front(false); 26740 | 26741 | if (details::imatch(current_token().value,"until")) 26742 | { 26743 | next_token(); 26744 | branch = node_allocator_.allocate<details::null_node<T> >(); 26745 | } 26746 | else 26747 | { 26748 | const token_t::token_type separator = token_t::e_eof; 26749 | 26750 | scope_handler sh(*this); 26751 | 26752 | scoped_bool_or_restorer sbr(state_.side_effect_present); 26753 | 26754 | scoped_inc_dec sid(state_.parsing_loop_stmt_count); 26755 | 26756 | for ( ; ; ) 26757 | { 26758 | state_.side_effect_present = false; 26759 | 26760 | expression_node_ptr arg = parse_expression(); 26761 | 26762 | if (0 == arg) 26763 | return error_node(); 26764 | else 26765 | { 26766 | arg_list.push_back(arg); 26767 | side_effect_list.push_back(state_.side_effect_present); 26768 | } 26769 | 26770 | if (details::imatch(current_token().value,"until")) 26771 | { 26772 | next_token(); 26773 | break; 26774 | } 26775 | 26776 | const bool is_next_until = peek_token_is(token_t::e_symbol) && 26777 | peek_token_is("until"); 26778 | 26779 | if (!token_is(separator) && is_next_until) 26780 | { 26781 | set_error(make_error( 26782 | parser_error::e_syntax, 26783 | current_token(), 26784 | "ERR070 - Expected '" + token_t::to_str(separator) + "' in body of repeat until loop", 26785 | exprtk_error_location)); 26786 | 26787 | return error_node(); 26788 | } 26789 | 26790 | if (details::imatch(current_token().value,"until")) 26791 | { 26792 | next_token(); 26793 | break; 26794 | } 26795 | } 26796 | 26797 | branch = simplify(arg_list,side_effect_list); 26798 | 26799 | sdd.delete_ptr = (0 == branch); 26800 | 26801 | if (sdd.delete_ptr) 26802 | { 26803 | set_error(make_error( 26804 | parser_error::e_syntax, 26805 | current_token(), 26806 | "ERR071 - Failed to parse body of repeat until loop", 26807 | exprtk_error_location)); 26808 | 26809 | return error_node(); 26810 | } 26811 | } 26812 | 26813 | if (!token_is(token_t::e_lbracket)) 26814 | { 26815 | set_error(make_error( 26816 | parser_error::e_syntax, 26817 | current_token(), 26818 | "ERR072 - Expected '(' before condition statement of repeat until loop", 26819 | exprtk_error_location)); 26820 | 26821 | free_node(node_allocator_, branch); 26822 | return error_node(); 26823 | } 26824 | else if (0 == (condition = parse_expression())) 26825 | { 26826 | set_error(make_error( 26827 | parser_error::e_syntax, 26828 | current_token(), 26829 | "ERR073 - Failed to parse condition for repeat until loop", 26830 | exprtk_error_location)); 26831 | 26832 | free_node(node_allocator_, branch); 26833 | return error_node(); 26834 | } 26835 | else if (!token_is(token_t::e_rbracket)) 26836 | { 26837 | set_error(make_error( 26838 | parser_error::e_syntax, 26839 | current_token(), 26840 | "ERR074 - Expected ')' after condition of repeat until loop", 26841 | exprtk_error_location)); 26842 | 26843 | free_node(node_allocator_, branch ); 26844 | free_node(node_allocator_, condition); 26845 | 26846 | return error_node(); 26847 | } 26848 | 26849 | expression_node_ptr result_node = 26850 | expression_generator_ 26851 | .repeat_until_loop( 26852 | condition, 26853 | branch, 26854 | brkcnt_list_.front()); 26855 | 26856 | if (0 == result_node) 26857 | { 26858 | set_error(make_error( 26859 | parser_error::e_syntax, 26860 | current_token(), 26861 | "ERR075 - Failed to synthesize repeat until loop", 26862 | exprtk_error_location)); 26863 | 26864 | free_node(node_allocator_, condition); 26865 | 26866 | return error_node(); 26867 | } 26868 | 26869 | handle_brkcnt_scope_exit(); 26870 | 26871 | if (result_node && result_node->valid()) 26872 | { 26873 | return result_node; 26874 | } 26875 | 26876 | set_error(make_error( 26877 | parser_error::e_synthesis, 26878 | current_token(), 26879 | "ERR076 - Failed to synthesize 'valid' repeat until loop", 26880 | exprtk_error_location)); 26881 | 26882 | free_node(node_allocator_, result_node); 26883 | 26884 | return error_node(); 26885 | } 26886 | 26887 | inline expression_node_ptr parse_for_loop() 26888 | { 26889 | expression_node_ptr initialiser = error_node(); 26890 | expression_node_ptr condition = error_node(); 26891 | expression_node_ptr incrementor = error_node(); 26892 | expression_node_ptr loop_body = error_node(); 26893 | 26894 | scope_element* se = 0; 26895 | bool result = true; 26896 | 26897 | next_token(); 26898 | 26899 | scope_handler sh(*this); 26900 | 26901 | if (!token_is(token_t::e_lbracket)) 26902 | { 26903 | set_error(make_error( 26904 | parser_error::e_syntax, 26905 | current_token(), 26906 | "ERR077 - Expected '(' at start of for-loop", 26907 | exprtk_error_location)); 26908 | 26909 | return error_node(); 26910 | } 26911 | 26912 | if (!token_is(token_t::e_eof)) 26913 | { 26914 | if ( 26915 | !token_is(token_t::e_symbol,prsrhlpr_t::e_hold) && 26916 | details::imatch(current_token().value,"var") 26917 | ) 26918 | { 26919 | next_token(); 26920 | 26921 | if (!token_is(token_t::e_symbol,prsrhlpr_t::e_hold)) 26922 | { 26923 | set_error(make_error( 26924 | parser_error::e_syntax, 26925 | current_token(), 26926 | "ERR078 - Expected a variable at the start of initialiser section of for-loop", 26927 | exprtk_error_location)); 26928 | 26929 | return error_node(); 26930 | } 26931 | else if (!peek_token_is(token_t::e_assign)) 26932 | { 26933 | set_error(make_error( 26934 | parser_error::e_syntax, 26935 | current_token(), 26936 | "ERR079 - Expected variable assignment of initialiser section of for-loop", 26937 | exprtk_error_location)); 26938 | 26939 | return error_node(); 26940 | } 26941 | 26942 | const std::string loop_counter_symbol = current_token().value; 26943 | 26944 | se = &sem_.get_element(loop_counter_symbol); 26945 | 26946 | if ((se->name == loop_counter_symbol) && se->active) 26947 | { 26948 | set_error(make_error( 26949 | parser_error::e_syntax, 26950 | current_token(), 26951 | "ERR080 - For-loop variable '" + loop_counter_symbol+ "' is being shadowed by a previous declaration", 26952 | exprtk_error_location)); 26953 | 26954 | return error_node(); 26955 | } 26956 | else if (!symtab_store_.is_variable(loop_counter_symbol)) 26957 | { 26958 | if ( 26959 | !se->active && 26960 | (se->name == loop_counter_symbol) && 26961 | (se->type == scope_element::e_variable) 26962 | ) 26963 | { 26964 | se->active = true; 26965 | se->ref_count++; 26966 | } 26967 | else 26968 | { 26969 | scope_element nse; 26970 | nse.name = loop_counter_symbol; 26971 | nse.active = true; 26972 | nse.ref_count = 1; 26973 | nse.type = scope_element::e_variable; 26974 | nse.depth = state_.scope_depth; 26975 | nse.data = new T(T(0)); 26976 | nse.var_node = node_allocator_.allocate<variable_node_t>(*reinterpret_cast<T*>(nse.data)); 26977 | 26978 | if (!sem_.add_element(nse)) 26979 | { 26980 | set_error(make_error( 26981 | parser_error::e_syntax, 26982 | current_token(), 26983 | "ERR081 - Failed to add new local variable '" + loop_counter_symbol + "' to SEM", 26984 | exprtk_error_location)); 26985 | 26986 | sem_.free_element(nse); 26987 | 26988 | result = false; 26989 | } 26990 | else 26991 | { 26992 | exprtk_debug(("parse_for_loop() - INFO - Added new local variable: %s\n", nse.name.c_str())); 26993 | 26994 | state_.activate_side_effect("parse_for_loop()"); 26995 | } 26996 | } 26997 | } 26998 | } 26999 | 27000 | if (0 == (initialiser = parse_expression())) 27001 | { 27002 | set_error(make_error( 27003 | parser_error::e_syntax, 27004 | current_token(), 27005 | "ERR082 - Failed to parse initialiser of for-loop", 27006 | exprtk_error_location)); 27007 | 27008 | result = false; 27009 | } 27010 | else if (!token_is(token_t::e_eof)) 27011 | { 27012 | set_error(make_error( 27013 | parser_error::e_syntax, 27014 | current_token(), 27015 | "ERR083 - Expected ';' after initialiser of for-loop", 27016 | exprtk_error_location)); 27017 | 27018 | result = false; 27019 | } 27020 | } 27021 | 27022 | if (!token_is(token_t::e_eof)) 27023 | { 27024 | if (0 == (condition = parse_expression())) 27025 | { 27026 | set_error(make_error( 27027 | parser_error::e_syntax, 27028 | current_token(), 27029 | "ERR084 - Failed to parse condition of for-loop", 27030 | exprtk_error_location)); 27031 | 27032 | result = false; 27033 | } 27034 | else if (!token_is(token_t::e_eof)) 27035 | { 27036 | set_error(make_error( 27037 | parser_error::e_syntax, 27038 | current_token(), 27039 | "ERR085 - Expected ';' after condition section of for-loop", 27040 | exprtk_error_location)); 27041 | 27042 | result = false; 27043 | } 27044 | } 27045 | 27046 | if (!token_is(token_t::e_rbracket)) 27047 | { 27048 | if (0 == (incrementor = parse_expression())) 27049 | { 27050 | set_error(make_error( 27051 | parser_error::e_syntax, 27052 | current_token(), 27053 | "ERR086 - Failed to parse incrementor of for-loop", 27054 | exprtk_error_location)); 27055 | 27056 | result = false; 27057 | } 27058 | else if (!token_is(token_t::e_rbracket)) 27059 | { 27060 | set_error(make_error( 27061 | parser_error::e_syntax, 27062 | current_token(), 27063 | "ERR087 - Expected ')' after incrementor section of for-loop", 27064 | exprtk_error_location)); 27065 | 27066 | result = false; 27067 | } 27068 | } 27069 | 27070 | if (result) 27071 | { 27072 | brkcnt_list_.push_front(false); 27073 | 27074 | scoped_inc_dec sid(state_.parsing_loop_stmt_count); 27075 | 27076 | if (0 == (loop_body = parse_multi_sequence("for-loop", true))) 27077 | { 27078 | set_error(make_error( 27079 | parser_error::e_syntax, 27080 | current_token(), 27081 | "ERR088 - Failed to parse body of for-loop", 27082 | exprtk_error_location)); 27083 | 27084 | result = false; 27085 | } 27086 | } 27087 | 27088 | if (!result) 27089 | { 27090 | if (se) 27091 | { 27092 | se->ref_count--; 27093 | } 27094 | 27095 | free_node(node_allocator_, initialiser); 27096 | free_node(node_allocator_, condition ); 27097 | free_node(node_allocator_, incrementor); 27098 | free_node(node_allocator_, loop_body ); 27099 | return error_node(); 27100 | } 27101 | 27102 | expression_node_ptr result_node = 27103 | expression_generator_.for_loop(initialiser, 27104 | condition, 27105 | incrementor, 27106 | loop_body, 27107 | brkcnt_list_.front()); 27108 | handle_brkcnt_scope_exit(); 27109 | 27110 | if (result_node && result_node->valid()) 27111 | { 27112 | return result_node; 27113 | } 27114 | 27115 | set_error(make_error( 27116 | parser_error::e_synthesis, 27117 | current_token(), 27118 | "ERR089 - Failed to synthesize 'valid' for-loop", 27119 | exprtk_error_location)); 27120 | 27121 | free_node(node_allocator_, result_node); 27122 | 27123 | return error_node(); 27124 | } 27125 | 27126 | inline expression_node_ptr parse_switch_statement() 27127 | { 27128 | std::vector<expression_node_ptr> arg_list; 27129 | 27130 | if (!details::imatch(current_token().value,"switch")) 27131 | { 27132 | set_error(make_error( 27133 | parser_error::e_syntax, 27134 | current_token(), 27135 | "ERR090 - Expected keyword 'switch'", 27136 | exprtk_error_location)); 27137 | 27138 | return error_node(); 27139 | } 27140 | 27141 | scoped_vec_delete<expression_node_t> svd((*this),arg_list); 27142 | 27143 | next_token(); 27144 | 27145 | if (!token_is(token_t::e_lcrlbracket)) 27146 | { 27147 | set_error(make_error( 27148 | parser_error::e_syntax, 27149 | current_token(), 27150 | "ERR091 - Expected '{' for call to switch statement", 27151 | exprtk_error_location)); 27152 | 27153 | return error_node(); 27154 | } 27155 | 27156 | expression_node_ptr default_statement = error_node(); 27157 | 27158 | scoped_expression_delete defstmt_delete((*this), default_statement); 27159 | 27160 | for ( ; ; ) 27161 | { 27162 | if (details::imatch("case",current_token().value)) 27163 | { 27164 | next_token(); 27165 | 27166 | expression_node_ptr condition = parse_expression(); 27167 | 27168 | if (0 == condition) 27169 | return error_node(); 27170 | else if (!token_is(token_t::e_colon)) 27171 | { 27172 | set_error(make_error( 27173 | parser_error::e_syntax, 27174 | current_token(), 27175 | "ERR092 - Expected ':' for case of switch statement", 27176 | exprtk_error_location)); 27177 | 27178 | free_node(node_allocator_, condition); 27179 | 27180 | return error_node(); 27181 | } 27182 | 27183 | expression_node_ptr consequent = 27184 | (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) ? 27185 | parse_multi_sequence("switch-consequent") : 27186 | parse_expression(); 27187 | 27188 | if (0 == consequent) 27189 | { 27190 | free_node(node_allocator_, condition); 27191 | 27192 | return error_node(); 27193 | } 27194 | else if (!token_is(token_t::e_eof)) 27195 | { 27196 | set_error(make_error( 27197 | parser_error::e_syntax, 27198 | current_token(), 27199 | "ERR093 - Expected ';' at end of case for switch statement", 27200 | exprtk_error_location)); 27201 | 27202 | free_node(node_allocator_, condition ); 27203 | free_node(node_allocator_, consequent); 27204 | 27205 | return error_node(); 27206 | } 27207 | 27208 | // Can we optimise away the case statement? 27209 | if (is_constant_node(condition) && is_false(condition)) 27210 | { 27211 | free_node(node_allocator_, condition ); 27212 | free_node(node_allocator_, consequent); 27213 | } 27214 | else 27215 | { 27216 | arg_list.push_back(condition ); 27217 | arg_list.push_back(consequent); 27218 | } 27219 | 27220 | } 27221 | else if (details::imatch("default",current_token().value)) 27222 | { 27223 | if (0 != default_statement) 27224 | { 27225 | set_error(make_error( 27226 | parser_error::e_syntax, 27227 | current_token(), 27228 | "ERR094 - Multiple default cases for switch statement", 27229 | exprtk_error_location)); 27230 | 27231 | return error_node(); 27232 | } 27233 | 27234 | next_token(); 27235 | 27236 | if (!token_is(token_t::e_colon)) 27237 | { 27238 | set_error(make_error( 27239 | parser_error::e_syntax, 27240 | current_token(), 27241 | "ERR095 - Expected ':' for default of switch statement", 27242 | exprtk_error_location)); 27243 | 27244 | return error_node(); 27245 | } 27246 | 27247 | default_statement = 27248 | (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) ? 27249 | parse_multi_sequence("switch-default"): 27250 | parse_expression(); 27251 | 27252 | if (0 == default_statement) 27253 | return error_node(); 27254 | else if (!token_is(token_t::e_eof)) 27255 | { 27256 | set_error(make_error( 27257 | parser_error::e_syntax, 27258 | current_token(), 27259 | "ERR096 - Expected ';' at end of default for switch statement", 27260 | exprtk_error_location)); 27261 | 27262 | return error_node(); 27263 | } 27264 | } 27265 | else if (token_is(token_t::e_rcrlbracket)) 27266 | break; 27267 | else 27268 | { 27269 | set_error(make_error( 27270 | parser_error::e_syntax, 27271 | current_token(), 27272 | "ERR097 - Expected '}' at end of switch statement", 27273 | exprtk_error_location)); 27274 | 27275 | return error_node(); 27276 | } 27277 | } 27278 | 27279 | const bool default_statement_present = (0 != default_statement); 27280 | 27281 | if (default_statement_present) 27282 | { 27283 | arg_list.push_back(default_statement); 27284 | } 27285 | else 27286 | { 27287 | arg_list.push_back(node_allocator_.allocate_c<literal_node_t>(std::numeric_limits<T>::quiet_NaN())); 27288 | } 27289 | 27290 | expression_node_ptr result = expression_generator_.switch_statement(arg_list, (0 != default_statement)); 27291 | 27292 | svd.delete_ptr = (0 == result); 27293 | defstmt_delete.delete_ptr = (0 == result); 27294 | 27295 | return result; 27296 | } 27297 | 27298 | inline expression_node_ptr parse_multi_switch_statement() 27299 | { 27300 | std::vector<expression_node_ptr> arg_list; 27301 | 27302 | if (!details::imatch(current_token().value,"[*]")) 27303 | { 27304 | set_error(make_error( 27305 | parser_error::e_syntax, 27306 | current_token(), 27307 | "ERR098 - Expected token '[*]'", 27308 | exprtk_error_location)); 27309 | 27310 | return error_node(); 27311 | } 27312 | 27313 | scoped_vec_delete<expression_node_t> svd((*this),arg_list); 27314 | 27315 | next_token(); 27316 | 27317 | if (!token_is(token_t::e_lcrlbracket)) 27318 | { 27319 | set_error(make_error( 27320 | parser_error::e_syntax, 27321 | current_token(), 27322 | "ERR099 - Expected '{' for call to [*] statement", 27323 | exprtk_error_location)); 27324 | 27325 | return error_node(); 27326 | } 27327 | 27328 | for ( ; ; ) 27329 | { 27330 | if (!details::imatch("case",current_token().value)) 27331 | { 27332 | set_error(make_error( 27333 | parser_error::e_syntax, 27334 | current_token(), 27335 | "ERR100 - Expected a 'case' statement for multi-switch", 27336 | exprtk_error_location)); 27337 | 27338 | return error_node(); 27339 | } 27340 | 27341 | next_token(); 27342 | 27343 | expression_node_ptr condition = parse_expression(); 27344 | 27345 | if (0 == condition) 27346 | return error_node(); 27347 | 27348 | if (!token_is(token_t::e_colon)) 27349 | { 27350 | set_error(make_error( 27351 | parser_error::e_syntax, 27352 | current_token(), 27353 | "ERR101 - Expected ':' for case of [*] statement", 27354 | exprtk_error_location)); 27355 | 27356 | return error_node(); 27357 | } 27358 | 27359 | expression_node_ptr consequent = 27360 | (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) ? 27361 | parse_multi_sequence("multi-switch-consequent") : 27362 | parse_expression(); 27363 | 27364 | if (0 == consequent) 27365 | return error_node(); 27366 | 27367 | if (!token_is(token_t::e_eof)) 27368 | { 27369 | set_error(make_error( 27370 | parser_error::e_syntax, 27371 | current_token(), 27372 | "ERR102 - Expected ';' at end of case for [*] statement", 27373 | exprtk_error_location)); 27374 | 27375 | return error_node(); 27376 | } 27377 | 27378 | // Can we optimise away the case statement? 27379 | if (is_constant_node(condition) && is_false(condition)) 27380 | { 27381 | free_node(node_allocator_, condition ); 27382 | free_node(node_allocator_, consequent); 27383 | } 27384 | else 27385 | { 27386 | arg_list.push_back(condition ); 27387 | arg_list.push_back(consequent); 27388 | } 27389 | 27390 | if (token_is(token_t::e_rcrlbracket,prsrhlpr_t::e_hold)) 27391 | { 27392 | break; 27393 | } 27394 | } 27395 | 27396 | if (!token_is(token_t::e_rcrlbracket)) 27397 | { 27398 | set_error(make_error( 27399 | parser_error::e_syntax, 27400 | current_token(), 27401 | "ERR103 - Expected '}' at end of [*] statement", 27402 | exprtk_error_location)); 27403 | 27404 | return error_node(); 27405 | } 27406 | 27407 | const expression_node_ptr result = expression_generator_.multi_switch_statement(arg_list); 27408 | 27409 | svd.delete_ptr = (0 == result); 27410 | 27411 | return result; 27412 | } 27413 | 27414 | inline expression_node_ptr parse_vararg_function() 27415 | { 27416 | std::vector<expression_node_ptr> arg_list; 27417 | 27418 | details::operator_type opt_type = details::e_default; 27419 | const std::string symbol = current_token().value; 27420 | 27421 | if (details::imatch(symbol,"~")) 27422 | { 27423 | next_token(); 27424 | return check_block_statement_closure(parse_multi_sequence()); 27425 | } 27426 | else if (details::imatch(symbol,"[*]")) 27427 | { 27428 | return check_block_statement_closure(parse_multi_switch_statement()); 27429 | } 27430 | else if (details::imatch(symbol, "avg" )) opt_type = details::e_avg ; 27431 | else if (details::imatch(symbol, "mand")) opt_type = details::e_mand; 27432 | else if (details::imatch(symbol, "max" )) opt_type = details::e_max ; 27433 | else if (details::imatch(symbol, "min" )) opt_type = details::e_min ; 27434 | else if (details::imatch(symbol, "mor" )) opt_type = details::e_mor ; 27435 | else if (details::imatch(symbol, "mul" )) opt_type = details::e_prod; 27436 | else if (details::imatch(symbol, "sum" )) opt_type = details::e_sum ; 27437 | else 27438 | { 27439 | set_error(make_error( 27440 | parser_error::e_syntax, 27441 | current_token(), 27442 | "ERR104 - Unsupported built-in vararg function: " + symbol, 27443 | exprtk_error_location)); 27444 | 27445 | return error_node(); 27446 | } 27447 | 27448 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 27449 | 27450 | lodge_symbol(symbol, e_st_function); 27451 | 27452 | next_token(); 27453 | 27454 | if (!token_is(token_t::e_lbracket)) 27455 | { 27456 | set_error(make_error( 27457 | parser_error::e_syntax, 27458 | current_token(), 27459 | "ERR105 - Expected '(' for call to vararg function: " + symbol, 27460 | exprtk_error_location)); 27461 | 27462 | return error_node(); 27463 | } 27464 | 27465 | if (token_is(token_t::e_rbracket)) 27466 | { 27467 | set_error(make_error( 27468 | parser_error::e_syntax, 27469 | current_token(), 27470 | "ERR106 - vararg function: " + symbol + 27471 | " requires at least one input parameter", 27472 | exprtk_error_location)); 27473 | 27474 | return error_node(); 27475 | } 27476 | 27477 | for ( ; ; ) 27478 | { 27479 | expression_node_ptr arg = parse_expression(); 27480 | 27481 | if (0 == arg) 27482 | return error_node(); 27483 | else 27484 | arg_list.push_back(arg); 27485 | 27486 | if (token_is(token_t::e_rbracket)) 27487 | break; 27488 | else if (!token_is(token_t::e_comma)) 27489 | { 27490 | set_error(make_error( 27491 | parser_error::e_syntax, 27492 | current_token(), 27493 | "ERR107 - Expected ',' for call to vararg function: " + symbol, 27494 | exprtk_error_location)); 27495 | 27496 | return error_node(); 27497 | } 27498 | } 27499 | 27500 | const expression_node_ptr result = expression_generator_.vararg_function(opt_type,arg_list); 27501 | 27502 | sdd.delete_ptr = (0 == result); 27503 | return result; 27504 | } 27505 | 27506 | #ifndef exprtk_disable_string_capabilities 27507 | inline expression_node_ptr parse_string_range_statement(expression_node_ptr& expression) 27508 | { 27509 | if (!token_is(token_t::e_lsqrbracket)) 27510 | { 27511 | set_error(make_error( 27512 | parser_error::e_syntax, 27513 | current_token(), 27514 | "ERR108 - Expected '[' as start of string range definition", 27515 | exprtk_error_location)); 27516 | 27517 | free_node(node_allocator_, expression); 27518 | 27519 | return error_node(); 27520 | } 27521 | else if (token_is(token_t::e_rsqrbracket)) 27522 | { 27523 | return node_allocator_.allocate<details::string_size_node<T> >(expression); 27524 | } 27525 | 27526 | range_t rp; 27527 | 27528 | if (!parse_range(rp,true)) 27529 | { 27530 | free_node(node_allocator_, expression); 27531 | 27532 | return error_node(); 27533 | } 27534 | 27535 | expression_node_ptr result = expression_generator_(expression,rp); 27536 | 27537 | if (0 == result) 27538 | { 27539 | set_error(make_error( 27540 | parser_error::e_syntax, 27541 | current_token(), 27542 | "ERR109 - Failed to generate string range node", 27543 | exprtk_error_location)); 27544 | 27545 | free_node(node_allocator_, expression); 27546 | rp.free(); 27547 | } 27548 | 27549 | rp.clear(); 27550 | 27551 | if (result && result->valid()) 27552 | { 27553 | return result; 27554 | } 27555 | 27556 | set_error(make_error( 27557 | parser_error::e_synthesis, 27558 | current_token(), 27559 | "ERR110 - Failed to synthesize node: string_range_node", 27560 | exprtk_error_location)); 27561 | 27562 | free_node(node_allocator_, result); 27563 | rp.free(); 27564 | return error_node(); 27565 | } 27566 | #else 27567 | inline expression_node_ptr parse_string_range_statement(expression_node_ptr&) 27568 | { 27569 | return error_node(); 27570 | } 27571 | #endif 27572 | 27573 | inline bool parse_pending_string_rangesize(expression_node_ptr& expression) 27574 | { 27575 | // Allow no more than 100 range calls, eg: s[][][]...[][] 27576 | const std::size_t max_rangesize_parses = 100; 27577 | 27578 | std::size_t i = 0; 27579 | 27580 | while 27581 | ( 27582 | (0 != expression) && 27583 | (i++ < max_rangesize_parses) && 27584 | error_list_.empty() && 27585 | is_generally_string_node(expression) && 27586 | token_is(token_t::e_lsqrbracket,prsrhlpr_t::e_hold) 27587 | ) 27588 | { 27589 | expression = parse_string_range_statement(expression); 27590 | } 27591 | 27592 | return (i > 1); 27593 | } 27594 | 27595 | inline void parse_pending_vector_index_operator(expression_node_ptr& expression) 27596 | { 27597 | if 27598 | ( 27599 | (0 != expression) && 27600 | error_list_.empty() && 27601 | is_ivector_node(expression) 27602 | ) 27603 | { 27604 | if ( 27605 | settings_.commutative_check_enabled() && 27606 | token_is(token_t::e_mul,prsrhlpr_t::e_hold) && 27607 | peek_token_is(token_t::e_lsqrbracket) 27608 | ) 27609 | { 27610 | token_is(token_t::e_mul); 27611 | token_is(token_t::e_lsqrbracket); 27612 | } 27613 | else if (token_is(token_t::e_lsqrbracket,prsrhlpr_t::e_hold)) 27614 | { 27615 | token_is(token_t::e_lsqrbracket); 27616 | } 27617 | else if ( 27618 | token_is(token_t::e_rbracket,prsrhlpr_t::e_hold) && 27619 | peek_token_is(token_t::e_lsqrbracket) 27620 | ) 27621 | { 27622 | token_is(token_t::e_rbracket ); 27623 | token_is(token_t::e_lsqrbracket); 27624 | } 27625 | else 27626 | return; 27627 | 27628 | details::vector_interface<T>* vi = dynamic_cast<details::vector_interface<T>*>(expression); 27629 | 27630 | if (vi) 27631 | { 27632 | details::vector_holder<T>& vec = vi->vec()->vec_holder(); 27633 | const std::string vector_name = sem_.get_vector_name(vec.data()); 27634 | expression_node_ptr index = parse_vector_index(vector_name); 27635 | 27636 | if (index) 27637 | { 27638 | expression = synthesize_vector_element(vector_name, &vec, expression, index); 27639 | return; 27640 | } 27641 | } 27642 | 27643 | free_node(node_allocator_, expression); 27644 | expression = error_node(); 27645 | } 27646 | } 27647 | 27648 | template <typename Allocator1, 27649 | typename Allocator2, 27650 | template <typename, typename> class Sequence> 27651 | inline expression_node_ptr simplify(Sequence<expression_node_ptr,Allocator1>& expression_list, 27652 | Sequence<bool,Allocator2>& side_effect_list, 27653 | const bool specialise_on_final_type = false) 27654 | { 27655 | if (expression_list.empty()) 27656 | return error_node(); 27657 | else if (1 == expression_list.size()) 27658 | return expression_list[0]; 27659 | 27660 | Sequence<expression_node_ptr,Allocator1> tmp_expression_list; 27661 | 27662 | exprtk_debug(("simplify() - expression_list.size: %d side_effect_list.size(): %d\n", 27663 | static_cast<int>(expression_list .size()), 27664 | static_cast<int>(side_effect_list.size()))); 27665 | 27666 | bool return_node_present = false; 27667 | 27668 | for (std::size_t i = 0; i < (expression_list.size() - 1); ++i) 27669 | { 27670 | if (is_variable_node(expression_list[i])) 27671 | continue; 27672 | else if ( 27673 | is_return_node (expression_list[i]) || 27674 | is_break_node (expression_list[i]) || 27675 | is_continue_node(expression_list[i]) 27676 | ) 27677 | { 27678 | tmp_expression_list.push_back(expression_list[i]); 27679 | 27680 | // Remove all subexpressions after first short-circuit 27681 | // node has been encountered. 27682 | 27683 | for (std::size_t j = i + 1; j < expression_list.size(); ++j) 27684 | { 27685 | free_node(node_allocator_, expression_list[j]); 27686 | } 27687 | 27688 | return_node_present = true; 27689 | 27690 | break; 27691 | } 27692 | else if ( 27693 | is_constant_node(expression_list[i]) || 27694 | is_null_node (expression_list[i]) || 27695 | !side_effect_list[i] 27696 | ) 27697 | { 27698 | free_node(node_allocator_, expression_list[i]); 27699 | continue; 27700 | } 27701 | else 27702 | tmp_expression_list.push_back(expression_list[i]); 27703 | } 27704 | 27705 | if (!return_node_present) 27706 | { 27707 | tmp_expression_list.push_back(expression_list.back()); 27708 | } 27709 | 27710 | expression_list.swap(tmp_expression_list); 27711 | 27712 | if (tmp_expression_list.size() > expression_list.size()) 27713 | { 27714 | exprtk_debug(("simplify() - Reduced subexpressions from %d to %d\n", 27715 | static_cast<int>(tmp_expression_list.size()), 27716 | static_cast<int>(expression_list .size()))); 27717 | } 27718 | 27719 | if ( 27720 | return_node_present || 27721 | side_effect_list.back() || 27722 | (expression_list.size() > 1) 27723 | ) 27724 | state_.activate_side_effect("simplify()"); 27725 | 27726 | if (1 == expression_list.size()) 27727 | return expression_list[0]; 27728 | else if (specialise_on_final_type && is_generally_string_node(expression_list.back())) 27729 | return expression_generator_.vararg_function(details::e_smulti,expression_list); 27730 | else 27731 | return expression_generator_.vararg_function(details::e_multi,expression_list); 27732 | } 27733 | 27734 | inline expression_node_ptr parse_multi_sequence(const std::string& source = "", 27735 | const bool enforce_crlbrackets = false) 27736 | { 27737 | token_t::token_type open_bracket = token_t::e_lcrlbracket; 27738 | token_t::token_type close_bracket = token_t::e_rcrlbracket; 27739 | token_t::token_type separator = token_t::e_eof; 27740 | 27741 | if (!token_is(open_bracket)) 27742 | { 27743 | if (!enforce_crlbrackets && token_is(token_t::e_lbracket)) 27744 | { 27745 | open_bracket = token_t::e_lbracket; 27746 | close_bracket = token_t::e_rbracket; 27747 | separator = token_t::e_comma; 27748 | } 27749 | else 27750 | { 27751 | set_error(make_error( 27752 | parser_error::e_syntax, 27753 | current_token(), 27754 | "ERR111 - Expected '" + token_t::to_str(open_bracket) + "' for call to multi-sequence" + 27755 | ((!source.empty()) ? std::string(" section of " + source): ""), 27756 | exprtk_error_location)); 27757 | 27758 | return error_node(); 27759 | } 27760 | } 27761 | else if (token_is(close_bracket)) 27762 | { 27763 | return node_allocator_.allocate<details::null_node<T> >(); 27764 | } 27765 | 27766 | std::vector<expression_node_ptr> arg_list; 27767 | std::vector<bool> side_effect_list; 27768 | 27769 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 27770 | 27771 | scope_handler sh(*this); 27772 | 27773 | scoped_bool_or_restorer sbr(state_.side_effect_present); 27774 | 27775 | for ( ; ; ) 27776 | { 27777 | state_.side_effect_present = false; 27778 | 27779 | expression_node_ptr arg = parse_expression(); 27780 | 27781 | if (0 == arg) 27782 | return error_node(); 27783 | else 27784 | { 27785 | arg_list.push_back(arg); 27786 | side_effect_list.push_back(state_.side_effect_present); 27787 | } 27788 | 27789 | if (token_is(close_bracket)) 27790 | break; 27791 | 27792 | const bool is_next_close = peek_token_is(close_bracket); 27793 | 27794 | if (!token_is(separator) && is_next_close) 27795 | { 27796 | set_error(make_error( 27797 | parser_error::e_syntax, 27798 | current_token(), 27799 | "ERR112 - Expected '" + lexer::token::seperator_to_str(separator) + "' for call to multi-sequence section of " + source, 27800 | exprtk_error_location)); 27801 | 27802 | return error_node(); 27803 | } 27804 | 27805 | if (token_is(close_bracket)) 27806 | break; 27807 | } 27808 | 27809 | expression_node_ptr result = simplify(arg_list, side_effect_list, source.empty()); 27810 | 27811 | sdd.delete_ptr = (0 == result); 27812 | return result; 27813 | } 27814 | 27815 | inline bool parse_range(range_t& rp, const bool skip_lsqr = false) 27816 | { 27817 | // Examples of valid ranges: 27818 | // 1. [1:5] -> [1,5) 27819 | // 2. [ :5] -> [0,5) 27820 | // 3. [1: ] -> [1,end) 27821 | // 4. [x:y] -> [x,y) where x <= y 27822 | // 5. [x+1:y/2] -> [x+1,y/2) where x+1 <= y/2 27823 | // 6. [ :y] -> [0,y) where 0 <= y 27824 | // 7. [x: ] -> [x,end) where x <= end 27825 | 27826 | rp.clear(); 27827 | 27828 | if (!skip_lsqr && !token_is(token_t::e_lsqrbracket)) 27829 | { 27830 | set_error(make_error( 27831 | parser_error::e_syntax, 27832 | current_token(), 27833 | "ERR113 - Expected '[' for start of range", 27834 | exprtk_error_location)); 27835 | 27836 | return false; 27837 | } 27838 | 27839 | if (token_is(token_t::e_colon)) 27840 | { 27841 | rp.n0_c.first = true; 27842 | rp.n0_c.second = 0; 27843 | rp.cache.first = 0; 27844 | } 27845 | else 27846 | { 27847 | expression_node_ptr r0 = parse_expression(); 27848 | 27849 | if (0 == r0) 27850 | { 27851 | set_error(make_error( 27852 | parser_error::e_syntax, 27853 | current_token(), 27854 | "ERR114 - Failed parse begin section of range", 27855 | exprtk_error_location)); 27856 | 27857 | return false; 27858 | } 27859 | else if (is_constant_node(r0)) 27860 | { 27861 | const T r0_value = r0->value(); 27862 | 27863 | if (r0_value >= T(0)) 27864 | { 27865 | rp.n0_c.first = true; 27866 | rp.n0_c.second = static_cast<std::size_t>(details::numeric::to_int64(r0_value)); 27867 | rp.cache.first = rp.n0_c.second; 27868 | } 27869 | 27870 | free_node(node_allocator_, r0); 27871 | 27872 | if (r0_value < T(0)) 27873 | { 27874 | set_error(make_error( 27875 | parser_error::e_syntax, 27876 | current_token(), 27877 | "ERR115 - Range lower bound less than zero! Constraint: r0 >= 0", 27878 | exprtk_error_location)); 27879 | 27880 | return false; 27881 | } 27882 | } 27883 | else 27884 | { 27885 | rp.n0_e.first = true; 27886 | rp.n0_e.second = r0; 27887 | } 27888 | 27889 | if (!token_is(token_t::e_colon)) 27890 | { 27891 | set_error(make_error( 27892 | parser_error::e_syntax, 27893 | current_token(), 27894 | "ERR116 - Expected ':' for break in range", 27895 | exprtk_error_location)); 27896 | 27897 | rp.free(); 27898 | 27899 | return false; 27900 | } 27901 | } 27902 | 27903 | if (token_is(token_t::e_rsqrbracket)) 27904 | { 27905 | rp.n1_c.first = true; 27906 | rp.n1_c.second = std::numeric_limits<std::size_t>::max(); 27907 | } 27908 | else 27909 | { 27910 | expression_node_ptr r1 = parse_expression(); 27911 | 27912 | if (0 == r1) 27913 | { 27914 | set_error(make_error( 27915 | parser_error::e_syntax, 27916 | current_token(), 27917 | "ERR117 - Failed parse end section of range", 27918 | exprtk_error_location)); 27919 | 27920 | rp.free(); 27921 | 27922 | return false; 27923 | } 27924 | else if (is_constant_node(r1)) 27925 | { 27926 | const T r1_value = r1->value(); 27927 | 27928 | if (r1_value >= T(0)) 27929 | { 27930 | rp.n1_c.first = true; 27931 | rp.n1_c.second = static_cast<std::size_t>(details::numeric::to_int64(r1_value)); 27932 | rp.cache.second = rp.n1_c.second; 27933 | } 27934 | 27935 | free_node(node_allocator_, r1); 27936 | 27937 | if (r1_value < T(0)) 27938 | { 27939 | set_error(make_error( 27940 | parser_error::e_syntax, 27941 | current_token(), 27942 | "ERR118 - Range upper bound less than zero! Constraint: r1 >= 0", 27943 | exprtk_error_location)); 27944 | 27945 | rp.free(); 27946 | 27947 | return false; 27948 | } 27949 | } 27950 | else 27951 | { 27952 | rp.n1_e.first = true; 27953 | rp.n1_e.second = r1; 27954 | } 27955 | 27956 | if (!token_is(token_t::e_rsqrbracket)) 27957 | { 27958 | set_error(make_error( 27959 | parser_error::e_syntax, 27960 | current_token(), 27961 | "ERR119 - Expected ']' for start of range", 27962 | exprtk_error_location)); 27963 | 27964 | rp.free(); 27965 | 27966 | return false; 27967 | } 27968 | } 27969 | 27970 | if (rp.const_range()) 27971 | { 27972 | std::size_t r0 = 0; 27973 | std::size_t r1 = 0; 27974 | 27975 | bool rp_result = false; 27976 | 27977 | try 27978 | { 27979 | rp_result = rp(r0, r1); 27980 | } 27981 | catch (std::runtime_error&) 27982 | {} 27983 | 27984 | if (!rp_result || (r0 > r1)) 27985 | { 27986 | set_error(make_error( 27987 | parser_error::e_syntax, 27988 | current_token(), 27989 | "ERR120 - Invalid range, Constraint: r0 <= r1", 27990 | exprtk_error_location)); 27991 | 27992 | return false; 27993 | } 27994 | } 27995 | 27996 | return true; 27997 | } 27998 | 27999 | inline void lodge_symbol(const std::string& symbol, 28000 | const symbol_type st) 28001 | { 28002 | dec_.add_symbol(symbol,st); 28003 | } 28004 | 28005 | #ifndef exprtk_disable_string_capabilities 28006 | inline expression_node_ptr parse_string() 28007 | { 28008 | const std::string symbol = current_token().value; 28009 | 28010 | typedef details::stringvar_node<T>* strvar_node_t; 28011 | 28012 | expression_node_ptr result = error_node(); 28013 | strvar_node_t const_str_node = static_cast<strvar_node_t>(0); 28014 | 28015 | scope_element& se = sem_.get_active_element(symbol); 28016 | 28017 | if (scope_element::e_string == se.type) 28018 | { 28019 | se.active = true; 28020 | result = se.str_node; 28021 | lodge_symbol(symbol, e_st_local_string); 28022 | } 28023 | else 28024 | { 28025 | typedef typename symtab_store::string_context str_ctxt_t; 28026 | str_ctxt_t str_ctx = symtab_store_.get_string_context(symbol); 28027 | 28028 | if ((0 == str_ctx.str_var) || !symtab_store_.is_conststr_stringvar(symbol)) 28029 | { 28030 | set_error(make_error( 28031 | parser_error::e_syntax, 28032 | current_token(), 28033 | "ERR121 - Unknown string symbol", 28034 | exprtk_error_location)); 28035 | 28036 | return error_node(); 28037 | } 28038 | 28039 | assert(str_ctx.str_var != 0); 28040 | assert(str_ctx.symbol_table != 0); 28041 | 28042 | result = str_ctx.str_var; 28043 | 28044 | if (symtab_store_.is_constant_string(symbol)) 28045 | { 28046 | const_str_node = static_cast<strvar_node_t>(result); 28047 | result = expression_generator_(const_str_node->str()); 28048 | } 28049 | else if (symbol_table_t::e_immutable == str_ctx.symbol_table->mutability()) 28050 | { 28051 | lodge_immutable_symbol( 28052 | current_token(), 28053 | make_memory_range(str_ctx.str_var->base(), str_ctx.str_var->size())); 28054 | } 28055 | 28056 | lodge_symbol(symbol, e_st_string); 28057 | } 28058 | 28059 | if (peek_token_is(token_t::e_lsqrbracket)) 28060 | { 28061 | next_token(); 28062 | 28063 | if (peek_token_is(token_t::e_rsqrbracket)) 28064 | { 28065 | next_token(); 28066 | next_token(); 28067 | 28068 | if (const_str_node) 28069 | { 28070 | free_node(node_allocator_, result); 28071 | 28072 | return expression_generator_(T(const_str_node->size())); 28073 | } 28074 | else 28075 | return node_allocator_.allocate<details::stringvar_size_node<T> > 28076 | (static_cast<details::stringvar_node<T>*>(result)->ref()); 28077 | } 28078 | 28079 | range_t rp; 28080 | 28081 | if (!parse_range(rp)) 28082 | { 28083 | free_node(node_allocator_, result); 28084 | 28085 | return error_node(); 28086 | } 28087 | else if (const_str_node) 28088 | { 28089 | free_node(node_allocator_, result); 28090 | result = expression_generator_(const_str_node->ref(),rp); 28091 | } 28092 | else 28093 | result = expression_generator_(static_cast<details::stringvar_node<T>*> 28094 | (result)->ref(), rp); 28095 | 28096 | if (result) 28097 | rp.clear(); 28098 | } 28099 | else 28100 | next_token(); 28101 | 28102 | return result; 28103 | } 28104 | #else 28105 | inline expression_node_ptr parse_string() 28106 | { 28107 | return error_node(); 28108 | } 28109 | #endif 28110 | 28111 | #ifndef exprtk_disable_string_capabilities 28112 | inline expression_node_ptr parse_const_string() 28113 | { 28114 | const std::string const_str = current_token().value; 28115 | expression_node_ptr result = expression_generator_(const_str); 28116 | 28117 | if (peek_token_is(token_t::e_lsqrbracket)) 28118 | { 28119 | next_token(); 28120 | 28121 | if (peek_token_is(token_t::e_rsqrbracket)) 28122 | { 28123 | next_token(); 28124 | next_token(); 28125 | 28126 | free_node(node_allocator_, result); 28127 | 28128 | return expression_generator_(T(const_str.size())); 28129 | } 28130 | 28131 | range_t rp; 28132 | 28133 | if (!parse_range(rp)) 28134 | { 28135 | free_node(node_allocator_, result); 28136 | rp.free(); 28137 | 28138 | return error_node(); 28139 | } 28140 | 28141 | free_node(node_allocator_, result); 28142 | 28143 | if (rp.n1_c.first && (rp.n1_c.second == std::numeric_limits<std::size_t>::max())) 28144 | { 28145 | rp.n1_c.second = const_str.size() - 1; 28146 | rp.cache.second = rp.n1_c.second; 28147 | } 28148 | 28149 | if ( 28150 | (rp.n0_c.first && (rp.n0_c.second >= const_str.size())) || 28151 | (rp.n1_c.first && (rp.n1_c.second >= const_str.size())) 28152 | ) 28153 | { 28154 | set_error(make_error( 28155 | parser_error::e_syntax, 28156 | current_token(), 28157 | "ERR122 - Overflow in range for string: '" + const_str + "'[" + 28158 | (rp.n0_c.first ? details::to_str(static_cast<int>(rp.n0_c.second)) : "?") + ":" + 28159 | (rp.n1_c.first ? details::to_str(static_cast<int>(rp.n1_c.second)) : "?") + "]", 28160 | exprtk_error_location)); 28161 | 28162 | rp.free(); 28163 | 28164 | return error_node(); 28165 | } 28166 | 28167 | result = expression_generator_(const_str,rp); 28168 | 28169 | if (result) 28170 | rp.clear(); 28171 | } 28172 | else 28173 | next_token(); 28174 | 28175 | return result; 28176 | } 28177 | #else 28178 | inline expression_node_ptr parse_const_string() 28179 | { 28180 | return error_node(); 28181 | } 28182 | #endif 28183 | 28184 | inline expression_node_ptr parse_vector_index(const std::string& vector_name = "") 28185 | { 28186 | expression_node_ptr index_expr = error_node(); 28187 | 28188 | if (0 == (index_expr = parse_expression())) 28189 | { 28190 | set_error(make_error( 28191 | parser_error::e_syntax, 28192 | current_token(), 28193 | "ERR123 - Failed to parse index for vector: '" + vector_name + "'", 28194 | exprtk_error_location)); 28195 | 28196 | return error_node(); 28197 | } 28198 | else if (!token_is(token_t::e_rsqrbracket)) 28199 | { 28200 | set_error(make_error( 28201 | parser_error::e_syntax, 28202 | current_token(), 28203 | "ERR124 - Expected ']' for index of vector: '" + vector_name + "'", 28204 | exprtk_error_location)); 28205 | 28206 | free_node(node_allocator_, index_expr); 28207 | 28208 | return error_node(); 28209 | } 28210 | 28211 | return index_expr; 28212 | } 28213 | 28214 | inline expression_node_ptr parse_vector() 28215 | { 28216 | const std::string vector_name = current_token().value; 28217 | 28218 | vector_holder_ptr vec = vector_holder_ptr(0); 28219 | 28220 | const scope_element& se = sem_.get_active_element(vector_name); 28221 | 28222 | if ( 28223 | !details::imatch(se.name, vector_name) || 28224 | (se.depth > state_.scope_depth) || 28225 | (scope_element::e_vector != se.type) 28226 | ) 28227 | { 28228 | typedef typename symtab_store::vector_context vec_ctxt_t; 28229 | vec_ctxt_t vec_ctx = symtab_store_.get_vector_context(vector_name); 28230 | 28231 | if (0 == vec_ctx.vector_holder) 28232 | { 28233 | set_error(make_error( 28234 | parser_error::e_syntax, 28235 | current_token(), 28236 | "ERR125 - Symbol '" + vector_name + " not a vector", 28237 | exprtk_error_location)); 28238 | 28239 | return error_node(); 28240 | } 28241 | 28242 | assert(0 != vec_ctx.vector_holder); 28243 | assert(0 != vec_ctx.symbol_table ); 28244 | 28245 | vec = vec_ctx.vector_holder; 28246 | 28247 | if (symbol_table_t::e_immutable == vec_ctx.symbol_table->mutability()) 28248 | { 28249 | lodge_immutable_symbol( 28250 | current_token(), 28251 | make_memory_range(vec->data(), vec->size())); 28252 | } 28253 | } 28254 | else 28255 | { 28256 | vec = se.vec_node; 28257 | } 28258 | 28259 | assert(0 != vec); 28260 | 28261 | next_token(); 28262 | 28263 | if (!token_is(token_t::e_lsqrbracket)) 28264 | { 28265 | return node_allocator_.allocate<vector_node_t>(vec); 28266 | } 28267 | else if (token_is(token_t::e_rsqrbracket)) 28268 | { 28269 | return (vec->rebaseable()) ? 28270 | node_allocator_.allocate<vector_size_node_t>(vec) : 28271 | expression_generator_(T(vec->size())); 28272 | } 28273 | 28274 | expression_node_ptr index_expr = parse_vector_index(vector_name); 28275 | 28276 | if (index_expr) 28277 | { 28278 | expression_node_ptr vec_node = node_allocator_.allocate<vector_node_t>(vec); 28279 | 28280 | return synthesize_vector_element(vector_name, vec, vec_node, index_expr); 28281 | } 28282 | 28283 | return error_node(); 28284 | } 28285 | 28286 | inline expression_node_ptr synthesize_vector_element(const std::string& vector_name, 28287 | vector_holder_ptr vec, 28288 | expression_node_ptr vec_node, 28289 | expression_node_ptr index_expr) 28290 | { 28291 | // Perform compile-time range check 28292 | if (details::is_constant_node(index_expr)) 28293 | { 28294 | const std::size_t index = static_cast<std::size_t>(details::numeric::to_int32(index_expr->value())); 28295 | const std::size_t vec_size = vec->size(); 28296 | 28297 | if (index >= vec_size) 28298 | { 28299 | set_error(make_error( 28300 | parser_error::e_syntax, 28301 | current_token(), 28302 | "ERR126 - Index of " + details::to_str(index) + " out of range for " 28303 | "vector '" + vector_name + "' of size " + details::to_str(vec_size), 28304 | exprtk_error_location)); 28305 | 28306 | free_node(node_allocator_, vec_node ); 28307 | free_node(node_allocator_, index_expr); 28308 | 28309 | return error_node(); 28310 | } 28311 | } 28312 | 28313 | return expression_generator_.vector_element(vector_name, vec, vec_node, index_expr); 28314 | } 28315 | 28316 | inline expression_node_ptr parse_vararg_function_call(ivararg_function<T>* vararg_function, const std::string& vararg_function_name) 28317 | { 28318 | std::vector<expression_node_ptr> arg_list; 28319 | 28320 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 28321 | 28322 | next_token(); 28323 | 28324 | if (token_is(token_t::e_lbracket)) 28325 | { 28326 | if (token_is(token_t::e_rbracket)) 28327 | { 28328 | if (!vararg_function->allow_zero_parameters()) 28329 | { 28330 | set_error(make_error( 28331 | parser_error::e_syntax, 28332 | current_token(), 28333 | "ERR127 - Zero parameter call to vararg function: " 28334 | + vararg_function_name + " not allowed", 28335 | exprtk_error_location)); 28336 | 28337 | return error_node(); 28338 | } 28339 | } 28340 | else 28341 | { 28342 | for ( ; ; ) 28343 | { 28344 | expression_node_ptr arg = parse_expression(); 28345 | 28346 | if (0 == arg) 28347 | return error_node(); 28348 | else 28349 | arg_list.push_back(arg); 28350 | 28351 | if (token_is(token_t::e_rbracket)) 28352 | break; 28353 | else if (!token_is(token_t::e_comma)) 28354 | { 28355 | set_error(make_error( 28356 | parser_error::e_syntax, 28357 | current_token(), 28358 | "ERR128 - Expected ',' for call to vararg function: " 28359 | + vararg_function_name, 28360 | exprtk_error_location)); 28361 | 28362 | return error_node(); 28363 | } 28364 | } 28365 | } 28366 | } 28367 | else if (!vararg_function->allow_zero_parameters()) 28368 | { 28369 | set_error(make_error( 28370 | parser_error::e_syntax, 28371 | current_token(), 28372 | "ERR129 - Zero parameter call to vararg function: " 28373 | + vararg_function_name + " not allowed", 28374 | exprtk_error_location)); 28375 | 28376 | return error_node(); 28377 | } 28378 | 28379 | if (arg_list.size() < vararg_function->min_num_args()) 28380 | { 28381 | set_error(make_error( 28382 | parser_error::e_syntax, 28383 | current_token(), 28384 | "ERR130 - Invalid number of parameters to call to vararg function: " 28385 | + vararg_function_name + ", require at least " 28386 | + details::to_str(static_cast<int>(vararg_function->min_num_args())) + " parameters", 28387 | exprtk_error_location)); 28388 | 28389 | return error_node(); 28390 | } 28391 | else if (arg_list.size() > vararg_function->max_num_args()) 28392 | { 28393 | set_error(make_error( 28394 | parser_error::e_syntax, 28395 | current_token(), 28396 | "ERR131 - Invalid number of parameters to call to vararg function: " 28397 | + vararg_function_name + ", require no more than " 28398 | + details::to_str(static_cast<int>(vararg_function->max_num_args())) + " parameters", 28399 | exprtk_error_location)); 28400 | 28401 | return error_node(); 28402 | } 28403 | 28404 | expression_node_ptr result = expression_generator_.vararg_function_call(vararg_function,arg_list); 28405 | 28406 | sdd.delete_ptr = (0 == result); 28407 | 28408 | return result; 28409 | } 28410 | 28411 | class type_checker 28412 | { 28413 | public: 28414 | 28415 | enum return_type_t 28416 | { 28417 | e_overload = ' ', 28418 | e_numeric = 'T', 28419 | e_string = 'S' 28420 | }; 28421 | 28422 | struct function_prototype_t 28423 | { 28424 | return_type_t return_type; 28425 | std::string param_seq; 28426 | }; 28427 | 28428 | typedef parser<T> parser_t; 28429 | typedef std::vector<function_prototype_t> function_definition_list_t; 28430 | 28431 | type_checker(parser_t& p, 28432 | const std::string& func_name, 28433 | const std::string& func_prototypes, 28434 | const return_type_t default_return_type) 28435 | : invalid_state_(true) 28436 | , parser_(p) 28437 | , function_name_(func_name) 28438 | , default_return_type_(default_return_type) 28439 | { 28440 | parse_function_prototypes(func_prototypes); 28441 | } 28442 | 28443 | bool verify(const std::string& param_seq, std::size_t& pseq_index) 28444 | { 28445 | if (function_definition_list_.empty()) 28446 | return true; 28447 | 28448 | std::vector<std::pair<std::size_t,char> > error_list; 28449 | 28450 | for (std::size_t i = 0; i < function_definition_list_.size(); ++i) 28451 | { 28452 | details::char_t diff_value = 0; 28453 | std::size_t diff_index = 0; 28454 | 28455 | const bool result = details::sequence_match(function_definition_list_[i].param_seq, 28456 | param_seq, 28457 | diff_index, diff_value); 28458 | 28459 | if (result) 28460 | { 28461 | pseq_index = i; 28462 | return true; 28463 | } 28464 | else 28465 | error_list.push_back(std::make_pair(diff_index, diff_value)); 28466 | } 28467 | 28468 | if (1 == error_list.size()) 28469 | { 28470 | parser_.set_error(make_error( 28471 | parser_error::e_syntax, 28472 | parser_.current_token(), 28473 | "ERR132 - Failed parameter type check for function '" + function_name_ + "', " 28474 | "Expected '" + function_definition_list_[0].param_seq + 28475 | "' call set: '" + param_seq + "'", 28476 | exprtk_error_location)); 28477 | } 28478 | else 28479 | { 28480 | // find first with largest diff_index; 28481 | std::size_t max_diff_index = 0; 28482 | 28483 | for (std::size_t i = 1; i < error_list.size(); ++i) 28484 | { 28485 | if (error_list[i].first > error_list[max_diff_index].first) 28486 | { 28487 | max_diff_index = i; 28488 | } 28489 | } 28490 | 28491 | parser_.set_error(make_error( 28492 | parser_error::e_syntax, 28493 | parser_.current_token(), 28494 | "ERR133 - Failed parameter type check for function '" + function_name_ + "', " 28495 | "Best match: '" + function_definition_list_[max_diff_index].param_seq + 28496 | "' call set: '" + param_seq + "'", 28497 | exprtk_error_location)); 28498 | } 28499 | 28500 | return false; 28501 | } 28502 | 28503 | std::size_t paramseq_count() const 28504 | { 28505 | return function_definition_list_.size(); 28506 | } 28507 | 28508 | std::string paramseq(const std::size_t& index) const 28509 | { 28510 | return function_definition_list_[index].param_seq; 28511 | } 28512 | 28513 | return_type_t return_type(const std::size_t& index) const 28514 | { 28515 | return function_definition_list_[index].return_type; 28516 | } 28517 | 28518 | bool invalid() const 28519 | { 28520 | return !invalid_state_; 28521 | } 28522 | 28523 | bool allow_zero_parameters() const 28524 | { 28525 | 28526 | for (std::size_t i = 0; i < function_definition_list_.size(); ++i) 28527 | { 28528 | if (std::string::npos != function_definition_list_[i].param_seq.find("Z")) 28529 | { 28530 | return true; 28531 | } 28532 | } 28533 | 28534 | return false; 28535 | } 28536 | 28537 | private: 28538 | 28539 | std::vector<std::string> split_param_seq(const std::string& param_seq, const details::char_t delimiter = '|') const 28540 | { 28541 | std::string::const_iterator current_begin = param_seq.begin(); 28542 | std::string::const_iterator iter = param_seq.begin(); 28543 | 28544 | std::vector<std::string> result; 28545 | 28546 | while (iter != param_seq.end()) 28547 | { 28548 | if (*iter == delimiter) 28549 | { 28550 | result.push_back(std::string(current_begin, iter)); 28551 | current_begin = ++iter; 28552 | } 28553 | else 28554 | ++iter; 28555 | } 28556 | 28557 | if (current_begin != iter) 28558 | { 28559 | result.push_back(std::string(current_begin, iter)); 28560 | } 28561 | 28562 | return result; 28563 | } 28564 | 28565 | inline bool is_valid_token(std::string param_seq, 28566 | function_prototype_t& funcproto) const 28567 | { 28568 | // Determine return type 28569 | funcproto.return_type = default_return_type_; 28570 | 28571 | if (param_seq.size() > 2) 28572 | { 28573 | if (':' == param_seq[1]) 28574 | { 28575 | // Note: Only overloaded igeneric functions can have return 28576 | // type definitions. 28577 | if (type_checker::e_overload != default_return_type_) 28578 | return false; 28579 | 28580 | switch (param_seq[0]) 28581 | { 28582 | case 'T' : funcproto.return_type = type_checker::e_numeric; 28583 | break; 28584 | 28585 | case 'S' : funcproto.return_type = type_checker::e_string; 28586 | break; 28587 | 28588 | default : return false; 28589 | } 28590 | 28591 | param_seq.erase(0,2); 28592 | } 28593 | } 28594 | 28595 | if ( 28596 | (std::string::npos != param_seq.find("?*")) || 28597 | (std::string::npos != param_seq.find("**")) 28598 | ) 28599 | { 28600 | return false; 28601 | } 28602 | else if ( 28603 | (std::string::npos == param_seq.find_first_not_of("STV*?|")) || 28604 | ("Z" == param_seq) 28605 | ) 28606 | { 28607 | funcproto.param_seq = param_seq; 28608 | return true; 28609 | } 28610 | 28611 | return false; 28612 | } 28613 | 28614 | void parse_function_prototypes(const std::string& func_prototypes) 28615 | { 28616 | if (func_prototypes.empty()) 28617 | return; 28618 | 28619 | std::vector<std::string> param_seq_list = split_param_seq(func_prototypes); 28620 | 28621 | typedef std::map<std::string,std::size_t> param_seq_map_t; 28622 | param_seq_map_t param_seq_map; 28623 | 28624 | for (std::size_t i = 0; i < param_seq_list.size(); ++i) 28625 | { 28626 | function_prototype_t func_proto; 28627 | 28628 | if (!is_valid_token(param_seq_list[i], func_proto)) 28629 | { 28630 | invalid_state_ = false; 28631 | 28632 | parser_.set_error(make_error( 28633 | parser_error::e_syntax, 28634 | parser_.current_token(), 28635 | "ERR134 - Invalid parameter sequence of '" + param_seq_list[i] + 28636 | "' for function: " + function_name_, 28637 | exprtk_error_location)); 28638 | return; 28639 | } 28640 | 28641 | param_seq_map_t::const_iterator seq_itr = param_seq_map.find(param_seq_list[i]); 28642 | 28643 | if (param_seq_map.end() != seq_itr) 28644 | { 28645 | invalid_state_ = false; 28646 | 28647 | parser_.set_error(make_error( 28648 | parser_error::e_syntax, 28649 | parser_.current_token(), 28650 | "ERR135 - Function '" + function_name_ + "' has a parameter sequence conflict between " + 28651 | "pseq_idx[" + details::to_str(seq_itr->second) + "] and" + 28652 | "pseq_idx[" + details::to_str(i) + "] " + 28653 | "param seq: " + param_seq_list[i], 28654 | exprtk_error_location)); 28655 | return; 28656 | } 28657 | 28658 | function_definition_list_.push_back(func_proto); 28659 | } 28660 | } 28661 | 28662 | type_checker(const type_checker&) exprtk_delete; 28663 | type_checker& operator=(const type_checker&) exprtk_delete; 28664 | 28665 | bool invalid_state_; 28666 | parser_t& parser_; 28667 | std::string function_name_; 28668 | const return_type_t default_return_type_; 28669 | function_definition_list_t function_definition_list_; 28670 | }; 28671 | 28672 | inline expression_node_ptr parse_generic_function_call(igeneric_function<T>* function, const std::string& function_name) 28673 | { 28674 | std::vector<expression_node_ptr> arg_list; 28675 | 28676 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 28677 | 28678 | next_token(); 28679 | 28680 | std::string param_type_list; 28681 | 28682 | type_checker tc( 28683 | (*this), 28684 | function_name, 28685 | function->parameter_sequence, 28686 | type_checker::e_string); 28687 | 28688 | if (tc.invalid()) 28689 | { 28690 | set_error(make_error( 28691 | parser_error::e_syntax, 28692 | current_token(), 28693 | "ERR136 - Type checker instantiation failure for generic function: " + function_name, 28694 | exprtk_error_location)); 28695 | 28696 | return error_node(); 28697 | } 28698 | 28699 | if (token_is(token_t::e_lbracket)) 28700 | { 28701 | if (token_is(token_t::e_rbracket)) 28702 | { 28703 | if ( 28704 | !function->allow_zero_parameters() && 28705 | !tc .allow_zero_parameters() 28706 | ) 28707 | { 28708 | set_error(make_error( 28709 | parser_error::e_syntax, 28710 | current_token(), 28711 | "ERR137 - Zero parameter call to generic function: " 28712 | + function_name + " not allowed", 28713 | exprtk_error_location)); 28714 | 28715 | return error_node(); 28716 | } 28717 | } 28718 | else 28719 | { 28720 | for ( ; ; ) 28721 | { 28722 | expression_node_ptr arg = parse_expression(); 28723 | 28724 | if (0 == arg) 28725 | return error_node(); 28726 | 28727 | if (is_ivector_node(arg)) 28728 | param_type_list += 'V'; 28729 | else if (is_generally_string_node(arg)) 28730 | param_type_list += 'S'; 28731 | else // Everything else is assumed to be a scalar returning expression 28732 | param_type_list += 'T'; 28733 | 28734 | arg_list.push_back(arg); 28735 | 28736 | if (token_is(token_t::e_rbracket)) 28737 | break; 28738 | else if (!token_is(token_t::e_comma)) 28739 | { 28740 | set_error(make_error( 28741 | parser_error::e_syntax, 28742 | current_token(), 28743 | "ERR138 - Expected ',' for call to generic function: " + function_name, 28744 | exprtk_error_location)); 28745 | 28746 | return error_node(); 28747 | } 28748 | } 28749 | } 28750 | } 28751 | else if ( 28752 | !function->parameter_sequence.empty() && 28753 | function->allow_zero_parameters () && 28754 | !tc .allow_zero_parameters () 28755 | ) 28756 | { 28757 | set_error(make_error( 28758 | parser_error::e_syntax, 28759 | current_token(), 28760 | "ERR139 - Zero parameter call to generic function: " 28761 | + function_name + " not allowed", 28762 | exprtk_error_location)); 28763 | 28764 | return error_node(); 28765 | } 28766 | 28767 | std::size_t param_seq_index = 0; 28768 | 28769 | if ( 28770 | state_.type_check_enabled && 28771 | !tc.verify(param_type_list, param_seq_index) 28772 | ) 28773 | { 28774 | set_error(make_error( 28775 | parser_error::e_syntax, 28776 | current_token(), 28777 | "ERR140 - Invalid input parameter sequence for call to generic function: " + function_name, 28778 | exprtk_error_location)); 28779 | 28780 | return error_node(); 28781 | } 28782 | 28783 | expression_node_ptr result = 28784 | (tc.paramseq_count() <= 1) ? 28785 | expression_generator_ 28786 | .generic_function_call(function, arg_list) : 28787 | expression_generator_ 28788 | .generic_function_call(function, arg_list, param_seq_index); 28789 | 28790 | sdd.delete_ptr = (0 == result); 28791 | 28792 | return result; 28793 | } 28794 | 28795 | inline bool parse_igeneric_function_params(std::string& param_type_list, 28796 | std::vector<expression_node_ptr>& arg_list, 28797 | const std::string& function_name, 28798 | igeneric_function<T>* function, 28799 | const type_checker& tc) 28800 | { 28801 | if (token_is(token_t::e_lbracket)) 28802 | { 28803 | if (token_is(token_t::e_rbracket)) 28804 | { 28805 | if ( 28806 | !function->allow_zero_parameters() && 28807 | !tc .allow_zero_parameters() 28808 | ) 28809 | { 28810 | set_error(make_error( 28811 | parser_error::e_syntax, 28812 | current_token(), 28813 | "ERR141 - Zero parameter call to generic function: " 28814 | + function_name + " not allowed", 28815 | exprtk_error_location)); 28816 | 28817 | return false; 28818 | } 28819 | } 28820 | else 28821 | { 28822 | for ( ; ; ) 28823 | { 28824 | expression_node_ptr arg = parse_expression(); 28825 | 28826 | if (0 == arg) 28827 | return false; 28828 | 28829 | if (is_ivector_node(arg)) 28830 | param_type_list += 'V'; 28831 | else if (is_generally_string_node(arg)) 28832 | param_type_list += 'S'; 28833 | else // Everything else is a scalar returning expression 28834 | param_type_list += 'T'; 28835 | 28836 | arg_list.push_back(arg); 28837 | 28838 | if (token_is(token_t::e_rbracket)) 28839 | break; 28840 | else if (!token_is(token_t::e_comma)) 28841 | { 28842 | set_error(make_error( 28843 | parser_error::e_syntax, 28844 | current_token(), 28845 | "ERR142 - Expected ',' for call to string function: " + function_name, 28846 | exprtk_error_location)); 28847 | 28848 | return false; 28849 | } 28850 | } 28851 | } 28852 | 28853 | return true; 28854 | } 28855 | else 28856 | return false; 28857 | } 28858 | 28859 | #ifndef exprtk_disable_string_capabilities 28860 | inline expression_node_ptr parse_string_function_call(igeneric_function<T>* function, const std::string& function_name) 28861 | { 28862 | // Move pass the function name 28863 | next_token(); 28864 | 28865 | std::string param_type_list; 28866 | 28867 | type_checker tc((*this), function_name, function->parameter_sequence, type_checker::e_string); 28868 | 28869 | if ( 28870 | (!function->parameter_sequence.empty()) && 28871 | (0 == tc.paramseq_count()) 28872 | ) 28873 | { 28874 | return error_node(); 28875 | } 28876 | 28877 | std::vector<expression_node_ptr> arg_list; 28878 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 28879 | 28880 | if (!parse_igeneric_function_params(param_type_list, arg_list, function_name, function, tc)) 28881 | { 28882 | return error_node(); 28883 | } 28884 | 28885 | std::size_t param_seq_index = 0; 28886 | 28887 | if (!tc.verify(param_type_list, param_seq_index)) 28888 | { 28889 | set_error(make_error( 28890 | parser_error::e_syntax, 28891 | current_token(), 28892 | "ERR143 - Invalid input parameter sequence for call to string function: " + function_name, 28893 | exprtk_error_location)); 28894 | 28895 | return error_node(); 28896 | } 28897 | 28898 | expression_node_ptr result = 28899 | (tc.paramseq_count() <= 1) ? 28900 | expression_generator_ 28901 | .string_function_call(function, arg_list) : 28902 | expression_generator_ 28903 | .string_function_call(function, arg_list, param_seq_index); 28904 | 28905 | sdd.delete_ptr = (0 == result); 28906 | 28907 | return result; 28908 | } 28909 | 28910 | inline expression_node_ptr parse_overload_function_call(igeneric_function<T>* function, const std::string& function_name) 28911 | { 28912 | // Move pass the function name 28913 | next_token(); 28914 | 28915 | std::string param_type_list; 28916 | 28917 | type_checker tc((*this), function_name, function->parameter_sequence, type_checker::e_overload); 28918 | 28919 | if ( 28920 | (!function->parameter_sequence.empty()) && 28921 | (0 == tc.paramseq_count()) 28922 | ) 28923 | { 28924 | return error_node(); 28925 | } 28926 | 28927 | std::vector<expression_node_ptr> arg_list; 28928 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 28929 | 28930 | if (!parse_igeneric_function_params(param_type_list, arg_list, function_name, function, tc)) 28931 | { 28932 | return error_node(); 28933 | } 28934 | 28935 | std::size_t param_seq_index = 0; 28936 | 28937 | if (!tc.verify(param_type_list, param_seq_index)) 28938 | { 28939 | set_error(make_error( 28940 | parser_error::e_syntax, 28941 | current_token(), 28942 | "ERR144 - Invalid input parameter sequence for call to overloaded function: " + function_name, 28943 | exprtk_error_location)); 28944 | 28945 | return error_node(); 28946 | } 28947 | 28948 | expression_node_ptr result = error_node(); 28949 | 28950 | if (type_checker::e_numeric == tc.return_type(param_seq_index)) 28951 | { 28952 | if (tc.paramseq_count() <= 1) 28953 | result = expression_generator_ 28954 | .generic_function_call(function, arg_list); 28955 | else 28956 | result = expression_generator_ 28957 | .generic_function_call(function, arg_list, param_seq_index); 28958 | } 28959 | else if (type_checker::e_string == tc.return_type(param_seq_index)) 28960 | { 28961 | if (tc.paramseq_count() <= 1) 28962 | result = expression_generator_ 28963 | .string_function_call(function, arg_list); 28964 | else 28965 | result = expression_generator_ 28966 | .string_function_call(function, arg_list, param_seq_index); 28967 | } 28968 | else 28969 | { 28970 | set_error(make_error( 28971 | parser_error::e_syntax, 28972 | current_token(), 28973 | "ERR145 - Invalid return type for call to overloaded function: " + function_name, 28974 | exprtk_error_location)); 28975 | } 28976 | 28977 | sdd.delete_ptr = (0 == result); 28978 | return result; 28979 | } 28980 | #endif 28981 | 28982 | template <typename Type, std::size_t NumberOfParameters> 28983 | struct parse_special_function_impl 28984 | { 28985 | static inline expression_node_ptr process(parser<Type>& p, const details::operator_type opt_type, const std::string& sf_name) 28986 | { 28987 | expression_node_ptr branch[NumberOfParameters]; 28988 | expression_node_ptr result = error_node(); 28989 | 28990 | std::fill_n(branch, NumberOfParameters, reinterpret_cast<expression_node_ptr>(0)); 28991 | 28992 | scoped_delete<expression_node_t,NumberOfParameters> sd(p,branch); 28993 | 28994 | p.next_token(); 28995 | 28996 | if (!p.token_is(token_t::e_lbracket)) 28997 | { 28998 | p.set_error(make_error( 28999 | parser_error::e_syntax, 29000 | p.current_token(), 29001 | "ERR146 - Expected '(' for special function '" + sf_name + "'", 29002 | exprtk_error_location)); 29003 | 29004 | return error_node(); 29005 | } 29006 | 29007 | for (std::size_t i = 0; i < NumberOfParameters; ++i) 29008 | { 29009 | branch[i] = p.parse_expression(); 29010 | 29011 | if (0 == branch[i]) 29012 | { 29013 | return p.error_node(); 29014 | } 29015 | else if (i < (NumberOfParameters - 1)) 29016 | { 29017 | if (!p.token_is(token_t::e_comma)) 29018 | { 29019 | p.set_error(make_error( 29020 | parser_error::e_syntax, 29021 | p.current_token(), 29022 | "ERR147 - Expected ',' before next parameter of special function '" + sf_name + "'", 29023 | exprtk_error_location)); 29024 | 29025 | return p.error_node(); 29026 | } 29027 | } 29028 | } 29029 | 29030 | if (!p.token_is(token_t::e_rbracket)) 29031 | { 29032 | p.set_error(make_error( 29033 | parser_error::e_syntax, 29034 | p.current_token(), 29035 | "ERR148 - Invalid number of parameters for special function '" + sf_name + "'", 29036 | exprtk_error_location)); 29037 | 29038 | return p.error_node(); 29039 | } 29040 | else 29041 | result = p.expression_generator_.special_function(opt_type,branch); 29042 | 29043 | sd.delete_ptr = (0 == result); 29044 | 29045 | return result; 29046 | } 29047 | }; 29048 | 29049 | inline expression_node_ptr parse_special_function() 29050 | { 29051 | const std::string sf_name = current_token().value; 29052 | 29053 | // Expect: $fDD(expr0,expr1,expr2) or $fDD(expr0,expr1,expr2,expr3) 29054 | if ( 29055 | !details::is_digit(sf_name[2]) || 29056 | !details::is_digit(sf_name[3]) 29057 | ) 29058 | { 29059 | set_error(make_error( 29060 | parser_error::e_token, 29061 | current_token(), 29062 | "ERR149 - Invalid special function[1]: " + sf_name, 29063 | exprtk_error_location)); 29064 | 29065 | return error_node(); 29066 | } 29067 | 29068 | const int id = (sf_name[2] - '0') * 10 + 29069 | (sf_name[3] - '0'); 29070 | 29071 | if (id >= details::e_sffinal) 29072 | { 29073 | set_error(make_error( 29074 | parser_error::e_token, 29075 | current_token(), 29076 | "ERR150 - Invalid special function[2]: " + sf_name, 29077 | exprtk_error_location)); 29078 | 29079 | return error_node(); 29080 | } 29081 | 29082 | const int sf_3_to_4 = details::e_sf48; 29083 | const details::operator_type opt_type = details::operator_type(id + 1000); 29084 | const std::size_t NumberOfParameters = (id < (sf_3_to_4 - 1000)) ? 3U : 4U; 29085 | 29086 | switch (NumberOfParameters) 29087 | { 29088 | case 3 : return parse_special_function_impl<T,3>::process((*this), opt_type, sf_name); 29089 | case 4 : return parse_special_function_impl<T,4>::process((*this), opt_type, sf_name); 29090 | default : return error_node(); 29091 | } 29092 | } 29093 | 29094 | inline expression_node_ptr parse_null_statement() 29095 | { 29096 | next_token(); 29097 | return node_allocator_.allocate<details::null_node<T> >(); 29098 | } 29099 | 29100 | #ifndef exprtk_disable_break_continue 29101 | inline expression_node_ptr parse_break_statement() 29102 | { 29103 | if (state_.parsing_break_stmt) 29104 | { 29105 | set_error(make_error( 29106 | parser_error::e_syntax, 29107 | current_token(), 29108 | "ERR151 - Invoking 'break' within a break call is not allowed", 29109 | exprtk_error_location)); 29110 | 29111 | return error_node(); 29112 | } 29113 | else if (0 == state_.parsing_loop_stmt_count) 29114 | { 29115 | set_error(make_error( 29116 | parser_error::e_syntax, 29117 | current_token(), 29118 | "ERR152 - Invalid use of 'break', allowed only in the scope of a loop", 29119 | exprtk_error_location)); 29120 | 29121 | return error_node(); 29122 | } 29123 | 29124 | scoped_bool_negator sbn(state_.parsing_break_stmt); 29125 | 29126 | if (!brkcnt_list_.empty()) 29127 | { 29128 | next_token(); 29129 | 29130 | brkcnt_list_.front() = true; 29131 | 29132 | expression_node_ptr return_expr = error_node(); 29133 | 29134 | if (token_is(token_t::e_lsqrbracket)) 29135 | { 29136 | if (0 == (return_expr = parse_expression())) 29137 | { 29138 | set_error(make_error( 29139 | parser_error::e_syntax, 29140 | current_token(), 29141 | "ERR153 - Failed to parse return expression for 'break' statement", 29142 | exprtk_error_location)); 29143 | 29144 | return error_node(); 29145 | } 29146 | else if (!token_is(token_t::e_rsqrbracket)) 29147 | { 29148 | set_error(make_error( 29149 | parser_error::e_syntax, 29150 | current_token(), 29151 | "ERR154 - Expected ']' at the completion of break's return expression", 29152 | exprtk_error_location)); 29153 | 29154 | free_node(node_allocator_, return_expr); 29155 | 29156 | return error_node(); 29157 | } 29158 | } 29159 | 29160 | state_.activate_side_effect("parse_break_statement()"); 29161 | 29162 | return node_allocator_.allocate<details::break_node<T> >(return_expr); 29163 | } 29164 | else 29165 | { 29166 | set_error(make_error( 29167 | parser_error::e_syntax, 29168 | current_token(), 29169 | "ERR155 - Invalid use of 'break', allowed only in the scope of a loop", 29170 | exprtk_error_location)); 29171 | } 29172 | 29173 | return error_node(); 29174 | } 29175 | 29176 | inline expression_node_ptr parse_continue_statement() 29177 | { 29178 | if (0 == state_.parsing_loop_stmt_count) 29179 | { 29180 | set_error(make_error( 29181 | parser_error::e_syntax, 29182 | current_token(), 29183 | "ERR156 - Invalid use of 'continue', allowed only in the scope of a loop", 29184 | exprtk_error_location)); 29185 | 29186 | return error_node(); 29187 | } 29188 | else 29189 | { 29190 | next_token(); 29191 | 29192 | brkcnt_list_.front() = true; 29193 | state_.activate_side_effect("parse_continue_statement()"); 29194 | 29195 | return node_allocator_.allocate<details::continue_node<T> >(); 29196 | } 29197 | } 29198 | #endif 29199 | 29200 | inline expression_node_ptr parse_define_vector_statement(const std::string& vec_name) 29201 | { 29202 | expression_node_ptr size_expression_node = error_node(); 29203 | 29204 | if (!token_is(token_t::e_lsqrbracket)) 29205 | { 29206 | set_error(make_error( 29207 | parser_error::e_syntax, 29208 | current_token(), 29209 | "ERR157 - Expected '[' as part of vector size definition", 29210 | exprtk_error_location)); 29211 | 29212 | return error_node(); 29213 | } 29214 | else if (0 == (size_expression_node = parse_expression())) 29215 | { 29216 | set_error(make_error( 29217 | parser_error::e_syntax, 29218 | current_token(), 29219 | "ERR158 - Failed to determine size of vector '" + vec_name + "'", 29220 | exprtk_error_location)); 29221 | 29222 | return error_node(); 29223 | } 29224 | else if (!is_constant_node(size_expression_node)) 29225 | { 29226 | const bool is_rebaseble_vector = 29227 | (size_expression_node->type() == details::expression_node<T>::e_vecsize) && 29228 | static_cast<details::vector_size_node<T>*>(size_expression_node)->vec_holder()->rebaseable(); 29229 | 29230 | free_node(node_allocator_, size_expression_node); 29231 | 29232 | const std::string error_msg = (is_rebaseble_vector) ? 29233 | std::string("Rebasable/Resizable vector cannot be used to define the size of vector") : 29234 | std::string("Expected a constant literal number as size of vector"); 29235 | set_error(make_error( 29236 | parser_error::e_syntax, 29237 | current_token(), 29238 | "ERR159 - " + error_msg + " '" + vec_name + "'", 29239 | exprtk_error_location)); 29240 | 29241 | return error_node(); 29242 | } 29243 | 29244 | const T vector_size = size_expression_node->value(); 29245 | 29246 | free_node(node_allocator_, size_expression_node); 29247 | 29248 | const std::size_t max_vector_size = settings_.max_local_vector_size(); 29249 | 29250 | if ( 29251 | (vector_size <= T(0)) || 29252 | std::not_equal_to<T>() 29253 | (T(0),vector_size - details::numeric::trunc(vector_size)) || 29254 | (static_cast<std::size_t>(vector_size) > max_vector_size) 29255 | ) 29256 | { 29257 | set_error(make_error( 29258 | parser_error::e_syntax, 29259 | current_token(), 29260 | "ERR160 - Invalid vector size. Must be an integer in the " 29261 | "range [0," + details::to_str(static_cast<std::size_t>(max_vector_size)) + "], size: " + 29262 | details::to_str(details::numeric::to_int32(vector_size)), 29263 | exprtk_error_location)); 29264 | 29265 | return error_node(); 29266 | } 29267 | 29268 | typename symbol_table_t::vector_holder_ptr vec_holder = typename symbol_table_t::vector_holder_ptr(0); 29269 | 29270 | const std::size_t vec_size = static_cast<std::size_t>(details::numeric::to_int32(vector_size)); 29271 | const std::size_t predicted_total_lclsymb_size = sizeof(T) * vec_size + sem_.total_local_symb_size_bytes(); 29272 | 29273 | if (predicted_total_lclsymb_size > settings().max_total_local_symbol_size_bytes()) 29274 | { 29275 | set_error(make_error( 29276 | parser_error::e_syntax, 29277 | current_token(), 29278 | "ERR161 - Adding vector '" + vec_name + "' of size " + details::to_str(vec_size) + " bytes " 29279 | "will exceed max total local symbol size of: " + details::to_str(settings().max_total_local_symbol_size_bytes()) + " bytes, " 29280 | "current total size: " + details::to_str(sem_.total_local_symb_size_bytes()) + " bytes", 29281 | exprtk_error_location)); 29282 | 29283 | return error_node(); 29284 | } 29285 | 29286 | scope_element& se = sem_.get_element(vec_name); 29287 | 29288 | if (se.name == vec_name) 29289 | { 29290 | if (se.active) 29291 | { 29292 | set_error(make_error( 29293 | parser_error::e_syntax, 29294 | current_token(), 29295 | "ERR162 - Illegal redefinition of local vector: '" + vec_name + "'", 29296 | exprtk_error_location)); 29297 | 29298 | return error_node(); 29299 | } 29300 | else if ( 29301 | (se.size == vec_size) && 29302 | (scope_element::e_vector == se.type) 29303 | ) 29304 | { 29305 | vec_holder = se.vec_node; 29306 | se.active = true; 29307 | se.depth = state_.scope_depth; 29308 | se.ref_count++; 29309 | } 29310 | } 29311 | 29312 | if (0 == vec_holder) 29313 | { 29314 | scope_element nse; 29315 | nse.name = vec_name; 29316 | nse.active = true; 29317 | nse.ref_count = 1; 29318 | nse.type = scope_element::e_vector; 29319 | nse.depth = state_.scope_depth; 29320 | nse.size = vec_size; 29321 | nse.data = new T[vec_size]; 29322 | nse.vec_node = new typename scope_element::vector_holder_t(reinterpret_cast<T*>(nse.data),nse.size); 29323 | 29324 | details::set_zero_value(reinterpret_cast<T*>(nse.data),vec_size); 29325 | 29326 | if (!sem_.add_element(nse)) 29327 | { 29328 | set_error(make_error( 29329 | parser_error::e_syntax, 29330 | current_token(), 29331 | "ERR163 - Failed to add new local vector '" + vec_name + "' to SEM", 29332 | exprtk_error_location)); 29333 | 29334 | sem_.free_element(nse); 29335 | 29336 | return error_node(); 29337 | } 29338 | 29339 | assert(sem_.total_local_symb_size_bytes() <= settings().max_total_local_symbol_size_bytes()); 29340 | 29341 | vec_holder = nse.vec_node; 29342 | 29343 | exprtk_debug(("parse_define_vector_statement() - INFO - Added new local vector: %s[%d]\n", 29344 | nse.name.c_str(), 29345 | static_cast<int>(nse.size))); 29346 | } 29347 | 29348 | state_.activate_side_effect("parse_define_vector_statement()"); 29349 | 29350 | lodge_symbol(vec_name, e_st_local_vector); 29351 | 29352 | std::vector<expression_node_ptr> vec_initilizer_list; 29353 | 29354 | scoped_vec_delete<expression_node_t> svd((*this),vec_initilizer_list); 29355 | 29356 | bool single_value_initialiser = false; 29357 | bool range_value_initialiser = false; 29358 | bool vec_to_vec_initialiser = false; 29359 | bool null_initialisation = false; 29360 | 29361 | if (!token_is(token_t::e_rsqrbracket)) 29362 | { 29363 | set_error(make_error( 29364 | parser_error::e_syntax, 29365 | current_token(), 29366 | "ERR164 - Expected ']' as part of vector size definition", 29367 | exprtk_error_location)); 29368 | 29369 | return error_node(); 29370 | } 29371 | else if (!token_is(token_t::e_eof, prsrhlpr_t::e_hold)) 29372 | { 29373 | if (!token_is(token_t::e_assign)) 29374 | { 29375 | set_error(make_error( 29376 | parser_error::e_syntax, 29377 | current_token(), 29378 | "ERR165 - Expected ':=' as part of vector definition", 29379 | exprtk_error_location)); 29380 | 29381 | return error_node(); 29382 | } 29383 | else if (token_is(token_t::e_lsqrbracket)) 29384 | { 29385 | expression_node_ptr initialiser_component = parse_expression(); 29386 | 29387 | if (0 == initialiser_component) 29388 | { 29389 | set_error(make_error( 29390 | parser_error::e_syntax, 29391 | current_token(), 29392 | "ERR166 - Failed to parse first component of vector initialiser for vector: " + vec_name, 29393 | exprtk_error_location)); 29394 | 29395 | return error_node(); 29396 | } 29397 | 29398 | vec_initilizer_list.push_back(initialiser_component); 29399 | 29400 | if (token_is(token_t::e_colon)) 29401 | { 29402 | initialiser_component = parse_expression(); 29403 | 29404 | if (0 == initialiser_component) 29405 | { 29406 | set_error(make_error( 29407 | parser_error::e_syntax, 29408 | current_token(), 29409 | "ERR167 - Failed to parse second component of vector initialiser for vector: " + vec_name, 29410 | exprtk_error_location)); 29411 | 29412 | return error_node(); 29413 | } 29414 | 29415 | vec_initilizer_list.push_back(initialiser_component); 29416 | } 29417 | 29418 | if (!token_is(token_t::e_rsqrbracket)) 29419 | { 29420 | set_error(make_error( 29421 | parser_error::e_syntax, 29422 | current_token(), 29423 | "ERR168 - Expected ']' to close single value vector initialiser", 29424 | exprtk_error_location)); 29425 | 29426 | return error_node(); 29427 | } 29428 | 29429 | switch (vec_initilizer_list.size()) 29430 | { 29431 | case 1 : single_value_initialiser = true; break; 29432 | case 2 : range_value_initialiser = true; break; 29433 | } 29434 | } 29435 | else if (!token_is(token_t::e_lcrlbracket)) 29436 | { 29437 | expression_node_ptr initialiser = error_node(); 29438 | 29439 | // Is this a vector to vector assignment and initialisation? 29440 | if (token_t::e_symbol == current_token().type) 29441 | { 29442 | // Is it a locally defined vector? 29443 | const scope_element& lcl_se = sem_.get_active_element(current_token().value); 29444 | 29445 | if (scope_element::e_vector == lcl_se.type) 29446 | { 29447 | if (0 != (initialiser = parse_expression())) 29448 | vec_initilizer_list.push_back(initialiser); 29449 | else 29450 | return error_node(); 29451 | } 29452 | // Are we dealing with a user defined vector? 29453 | else if (symtab_store_.is_vector(current_token().value)) 29454 | { 29455 | lodge_symbol(current_token().value, e_st_vector); 29456 | 29457 | if (0 != (initialiser = parse_expression())) 29458 | vec_initilizer_list.push_back(initialiser); 29459 | else 29460 | return error_node(); 29461 | } 29462 | // Are we dealing with a null initialisation vector definition? 29463 | else if (token_is(token_t::e_symbol,"null")) 29464 | null_initialisation = true; 29465 | } 29466 | 29467 | if (!null_initialisation) 29468 | { 29469 | if (0 == initialiser) 29470 | { 29471 | set_error(make_error( 29472 | parser_error::e_syntax, 29473 | current_token(), 29474 | "ERR169 - Expected '{' as part of vector initialiser list", 29475 | exprtk_error_location)); 29476 | 29477 | return error_node(); 29478 | } 29479 | else 29480 | vec_to_vec_initialiser = true; 29481 | } 29482 | } 29483 | else if (!token_is(token_t::e_rcrlbracket)) 29484 | { 29485 | for ( ; ; ) 29486 | { 29487 | expression_node_ptr initialiser = parse_expression(); 29488 | 29489 | if (0 == initialiser) 29490 | { 29491 | set_error(make_error( 29492 | parser_error::e_syntax, 29493 | current_token(), 29494 | "ERR170 - Expected '{' as part of vector initialiser list", 29495 | exprtk_error_location)); 29496 | 29497 | return error_node(); 29498 | } 29499 | else 29500 | vec_initilizer_list.push_back(initialiser); 29501 | 29502 | if (token_is(token_t::e_rcrlbracket)) 29503 | break; 29504 | 29505 | const bool is_next_close = peek_token_is(token_t::e_rcrlbracket); 29506 | 29507 | if (!token_is(token_t::e_comma) && is_next_close) 29508 | { 29509 | set_error(make_error( 29510 | parser_error::e_syntax, 29511 | current_token(), 29512 | "ERR171 - Expected ',' between vector initialisers", 29513 | exprtk_error_location)); 29514 | 29515 | return error_node(); 29516 | } 29517 | 29518 | if (token_is(token_t::e_rcrlbracket)) 29519 | break; 29520 | } 29521 | } 29522 | 29523 | if ( 29524 | !token_is(token_t::e_rbracket , prsrhlpr_t::e_hold) && 29525 | !token_is(token_t::e_rcrlbracket, prsrhlpr_t::e_hold) && 29526 | !token_is(token_t::e_rsqrbracket, prsrhlpr_t::e_hold) 29527 | ) 29528 | { 29529 | if (!token_is(token_t::e_eof,prsrhlpr_t::e_hold)) 29530 | { 29531 | set_error(make_error( 29532 | parser_error::e_syntax, 29533 | current_token(), 29534 | "ERR172 - Expected ';' at end of vector definition", 29535 | exprtk_error_location)); 29536 | 29537 | return error_node(); 29538 | } 29539 | } 29540 | 29541 | if ( 29542 | !single_value_initialiser && 29543 | !range_value_initialiser && 29544 | (T(vec_initilizer_list.size()) > vector_size) 29545 | ) 29546 | { 29547 | set_error(make_error( 29548 | parser_error::e_syntax, 29549 | current_token(), 29550 | "ERR173 - Initialiser list larger than the number of elements in the vector: '" + vec_name + "'", 29551 | exprtk_error_location)); 29552 | 29553 | return error_node(); 29554 | } 29555 | } 29556 | 29557 | expression_node_ptr result = error_node(); 29558 | 29559 | if ( 29560 | (vec_initilizer_list.size() == 1) && 29561 | single_value_initialiser 29562 | ) 29563 | { 29564 | if (details::is_constant_node(vec_initilizer_list[0])) 29565 | { 29566 | // vector_init_zero_value_node var v[10] := [0] 29567 | if (T(0) == vec_initilizer_list[0]->value()) 29568 | { 29569 | result = node_allocator_ 29570 | .allocate<details::vector_init_zero_value_node<T> >( 29571 | (*vec_holder)[0], 29572 | vec_size, 29573 | vec_initilizer_list); 29574 | } 29575 | else 29576 | { 29577 | // vector_init_single_constvalue_node var v[10] := [123] 29578 | result = node_allocator_ 29579 | .allocate<details::vector_init_single_constvalue_node<T> >( 29580 | (*vec_holder)[0], 29581 | vec_size, 29582 | vec_initilizer_list); 29583 | } 29584 | } 29585 | else 29586 | { 29587 | // vector_init_single_value_node var v[10] := [123 + (x / y)] 29588 | result = node_allocator_ 29589 | .allocate<details::vector_init_single_value_node<T> >( 29590 | (*vec_holder)[0], 29591 | vec_size, 29592 | vec_initilizer_list); 29593 | } 29594 | } 29595 | else if ( 29596 | (vec_initilizer_list.size() == 2) && 29597 | range_value_initialiser 29598 | ) 29599 | { 29600 | bool base_const = details::is_constant_node(vec_initilizer_list[0]); 29601 | bool inc_const = details::is_constant_node(vec_initilizer_list[1]); 29602 | 29603 | if (base_const && inc_const) 29604 | { 29605 | // vector_init_single_value_node var v[10] := [1 : 3.5] 29606 | result = node_allocator_ 29607 | .allocate<details::vector_init_iota_constconst_node<T> >( 29608 | (*vec_holder)[0], 29609 | vec_size, 29610 | vec_initilizer_list); 29611 | } 29612 | else if (base_const && !inc_const) 29613 | { 29614 | // vector_init_single_value_node var v[10] := [1 : x + y] 29615 | result = node_allocator_ 29616 | .allocate<details::vector_init_iota_constnconst_node<T> >( 29617 | (*vec_holder)[0], 29618 | vec_size, 29619 | vec_initilizer_list); 29620 | } 29621 | else if (!base_const && inc_const) 29622 | { 29623 | // vector_init_single_value_node var v[10] := [x + y : 3] 29624 | result = node_allocator_ 29625 | .allocate<details::vector_init_iota_nconstconst_node<T> >( 29626 | (*vec_holder)[0], 29627 | vec_size, 29628 | vec_initilizer_list); 29629 | } 29630 | else if (!base_const && !inc_const) 29631 | { 29632 | // vector_init_single_value_node var v[10] := [x + y : z / w] 29633 | result = node_allocator_ 29634 | .allocate<details::vector_init_iota_nconstnconst_node<T> >( 29635 | (*vec_holder)[0], 29636 | vec_size, 29637 | vec_initilizer_list); 29638 | } 29639 | } 29640 | else if (null_initialisation) 29641 | result = expression_generator_(T(0.0)); 29642 | else if (vec_to_vec_initialiser) 29643 | { 29644 | expression_node_ptr vec_node = node_allocator_.allocate<vector_node_t>(vec_holder); 29645 | 29646 | result = expression_generator_( 29647 | details::e_assign, 29648 | vec_node, 29649 | vec_initilizer_list[0]); 29650 | } 29651 | else 29652 | { 29653 | result = node_allocator_ 29654 | .allocate<details::vector_initialisation_node<T> >( 29655 | (*vec_holder)[0], 29656 | vec_size, 29657 | vec_initilizer_list, 29658 | single_value_initialiser); 29659 | } 29660 | 29661 | svd.delete_ptr = false; 29662 | 29663 | if (result && result->valid()) 29664 | { 29665 | return result; 29666 | } 29667 | 29668 | details::free_node(node_allocator_, result); 29669 | 29670 | set_error(make_error( 29671 | parser_error::e_synthesis, 29672 | current_token(), 29673 | "ERR174 - Failed to generate initialisation node for vector: " + vec_name, 29674 | exprtk_error_location)); 29675 | 29676 | return error_node(); 29677 | } 29678 | 29679 | #ifndef exprtk_disable_string_capabilities 29680 | inline expression_node_ptr parse_define_string_statement(const std::string& str_name, expression_node_ptr initialisation_expression) 29681 | { 29682 | stringvar_node_t* str_node = reinterpret_cast<stringvar_node_t*>(0); 29683 | 29684 | scope_element& se = sem_.get_element(str_name); 29685 | 29686 | if (se.name == str_name) 29687 | { 29688 | if (se.active) 29689 | { 29690 | set_error(make_error( 29691 | parser_error::e_syntax, 29692 | current_token(), 29693 | "ERR175 - Illegal redefinition of local variable: '" + str_name + "'", 29694 | exprtk_error_location)); 29695 | 29696 | free_node(node_allocator_, initialisation_expression); 29697 | 29698 | return error_node(); 29699 | } 29700 | else if (scope_element::e_string == se.type) 29701 | { 29702 | str_node = se.str_node; 29703 | se.active = true; 29704 | se.depth = state_.scope_depth; 29705 | se.ref_count++; 29706 | } 29707 | } 29708 | 29709 | if (0 == str_node) 29710 | { 29711 | scope_element nse; 29712 | nse.name = str_name; 29713 | nse.active = true; 29714 | nse.ref_count = 1; 29715 | nse.type = scope_element::e_string; 29716 | nse.depth = state_.scope_depth; 29717 | nse.data = new std::string; 29718 | nse.str_node = new stringvar_node_t(*reinterpret_cast<std::string*>(nse.data)); 29719 | 29720 | if (!sem_.add_element(nse)) 29721 | { 29722 | set_error(make_error( 29723 | parser_error::e_syntax, 29724 | current_token(), 29725 | "ERR176 - Failed to add new local string variable '" + str_name + "' to SEM", 29726 | exprtk_error_location)); 29727 | 29728 | free_node(node_allocator_, initialisation_expression); 29729 | 29730 | sem_.free_element(nse); 29731 | 29732 | return error_node(); 29733 | } 29734 | 29735 | assert(sem_.total_local_symb_size_bytes() <= settings().max_total_local_symbol_size_bytes()); 29736 | 29737 | str_node = nse.str_node; 29738 | 29739 | exprtk_debug(("parse_define_string_statement() - INFO - Added new local string variable: %s\n", nse.name.c_str())); 29740 | } 29741 | 29742 | lodge_symbol(str_name, e_st_local_string); 29743 | 29744 | state_.activate_side_effect("parse_define_string_statement()"); 29745 | 29746 | expression_node_ptr branch[2] = {0}; 29747 | 29748 | branch[0] = str_node; 29749 | branch[1] = initialisation_expression; 29750 | 29751 | return expression_generator_(details::e_assign,branch); 29752 | } 29753 | #else 29754 | inline expression_node_ptr parse_define_string_statement(const std::string&, expression_node_ptr) 29755 | { 29756 | return error_node(); 29757 | } 29758 | #endif 29759 | 29760 | inline bool local_variable_is_shadowed(const std::string& symbol) 29761 | { 29762 | const scope_element& se = sem_.get_element(symbol); 29763 | return (se.name == symbol) && se.active; 29764 | } 29765 | 29766 | inline expression_node_ptr parse_define_var_statement() 29767 | { 29768 | if (settings_.vardef_disabled()) 29769 | { 29770 | set_error(make_error( 29771 | parser_error::e_syntax, 29772 | current_token(), 29773 | "ERR177 - Illegal variable definition", 29774 | exprtk_error_location)); 29775 | 29776 | return error_node(); 29777 | } 29778 | else if (!details::imatch(current_token().value,"var")) 29779 | { 29780 | return error_node(); 29781 | } 29782 | else 29783 | next_token(); 29784 | 29785 | const std::string var_name = current_token().value; 29786 | 29787 | expression_node_ptr initialisation_expression = error_node(); 29788 | 29789 | if (!token_is(token_t::e_symbol)) 29790 | { 29791 | set_error(make_error( 29792 | parser_error::e_syntax, 29793 | current_token(), 29794 | "ERR178 - Expected a symbol for variable definition", 29795 | exprtk_error_location)); 29796 | 29797 | return error_node(); 29798 | } 29799 | else if (details::is_reserved_symbol(var_name)) 29800 | { 29801 | set_error(make_error( 29802 | parser_error::e_syntax, 29803 | current_token(), 29804 | "ERR179 - Illegal redefinition of reserved keyword: '" + var_name + "'", 29805 | exprtk_error_location)); 29806 | 29807 | return error_node(); 29808 | } 29809 | else if (symtab_store_.symbol_exists(var_name)) 29810 | { 29811 | set_error(make_error( 29812 | parser_error::e_syntax, 29813 | current_token(), 29814 | "ERR180 - Illegal redefinition of variable '" + var_name + "'", 29815 | exprtk_error_location)); 29816 | 29817 | return error_node(); 29818 | } 29819 | else if (local_variable_is_shadowed(var_name)) 29820 | { 29821 | set_error(make_error( 29822 | parser_error::e_syntax, 29823 | current_token(), 29824 | "ERR181 - Illegal redefinition of local variable: '" + var_name + "'", 29825 | exprtk_error_location)); 29826 | 29827 | return error_node(); 29828 | } 29829 | else if (token_is(token_t::e_lsqrbracket,prsrhlpr_t::e_hold)) 29830 | { 29831 | return parse_define_vector_statement(var_name); 29832 | } 29833 | else if (token_is(token_t::e_lcrlbracket,prsrhlpr_t::e_hold)) 29834 | { 29835 | return parse_uninitialised_var_statement(var_name); 29836 | } 29837 | else if (token_is(token_t::e_assign)) 29838 | { 29839 | if (0 == (initialisation_expression = parse_expression())) 29840 | { 29841 | set_error(make_error( 29842 | parser_error::e_syntax, 29843 | current_token(), 29844 | "ERR182 - Failed to parse initialisation expression for variable '" + var_name + "'", 29845 | exprtk_error_location)); 29846 | 29847 | return error_node(); 29848 | } 29849 | } 29850 | 29851 | if ( 29852 | !token_is(token_t::e_rbracket , prsrhlpr_t::e_hold) && 29853 | !token_is(token_t::e_rcrlbracket, prsrhlpr_t::e_hold) && 29854 | !token_is(token_t::e_rsqrbracket, prsrhlpr_t::e_hold) 29855 | ) 29856 | { 29857 | if (!token_is(token_t::e_eof,prsrhlpr_t::e_hold)) 29858 | { 29859 | set_error(make_error( 29860 | parser_error::e_syntax, 29861 | current_token(), 29862 | "ERR183 - Expected ';' after variable '" + var_name + "' definition", 29863 | exprtk_error_location)); 29864 | 29865 | free_node(node_allocator_, initialisation_expression); 29866 | 29867 | return error_node(); 29868 | } 29869 | } 29870 | 29871 | if ( 29872 | (0 != initialisation_expression) && 29873 | details::is_generally_string_node(initialisation_expression) 29874 | ) 29875 | { 29876 | return parse_define_string_statement(var_name,initialisation_expression); 29877 | } 29878 | 29879 | expression_node_ptr var_node = reinterpret_cast<expression_node_ptr>(0); 29880 | 29881 | scope_element& se = sem_.get_element(var_name); 29882 | 29883 | if (se.name == var_name) 29884 | { 29885 | if (se.active) 29886 | { 29887 | set_error(make_error( 29888 | parser_error::e_syntax, 29889 | current_token(), 29890 | "ERR184 - Illegal redefinition of local variable: '" + var_name + "'", 29891 | exprtk_error_location)); 29892 | 29893 | free_node(node_allocator_, initialisation_expression); 29894 | 29895 | return error_node(); 29896 | } 29897 | else if (scope_element::e_variable == se.type) 29898 | { 29899 | var_node = se.var_node; 29900 | se.active = true; 29901 | se.depth = state_.scope_depth; 29902 | se.ref_count++; 29903 | } 29904 | } 29905 | 29906 | if (0 == var_node) 29907 | { 29908 | const std::size_t predicted_total_lclsymb_size = sizeof(T) + sem_.total_local_symb_size_bytes(); 29909 | 29910 | if (predicted_total_lclsymb_size > settings().max_total_local_symbol_size_bytes()) 29911 | { 29912 | set_error(make_error( 29913 | parser_error::e_syntax, 29914 | current_token(), 29915 | "ERR185 - Adding variable '" + var_name + "' " 29916 | "will exceed max total local symbol size of: " + details::to_str(settings().max_total_local_symbol_size_bytes()) + " bytes, " 29917 | "current total size: " + details::to_str(sem_.total_local_symb_size_bytes()) + " bytes", 29918 | exprtk_error_location)); 29919 | 29920 | free_node(node_allocator_, initialisation_expression); 29921 | 29922 | return error_node(); 29923 | } 29924 | 29925 | scope_element nse; 29926 | nse.name = var_name; 29927 | nse.active = true; 29928 | nse.ref_count = 1; 29929 | nse.type = scope_element::e_variable; 29930 | nse.depth = state_.scope_depth; 29931 | nse.data = new T(T(0)); 29932 | nse.var_node = node_allocator_.allocate<variable_node_t>(*reinterpret_cast<T*>(nse.data)); 29933 | 29934 | if (!sem_.add_element(nse)) 29935 | { 29936 | set_error(make_error( 29937 | parser_error::e_syntax, 29938 | current_token(), 29939 | "ERR186 - Failed to add new local variable '" + var_name + "' to SEM", 29940 | exprtk_error_location)); 29941 | 29942 | free_node(node_allocator_, initialisation_expression); 29943 | 29944 | sem_.free_element(nse); 29945 | 29946 | return error_node(); 29947 | } 29948 | 29949 | assert(sem_.total_local_symb_size_bytes() <= settings().max_total_local_symbol_size_bytes()); 29950 | 29951 | var_node = nse.var_node; 29952 | 29953 | exprtk_debug(("parse_define_var_statement() - INFO - Added new local variable: %s\n", nse.name.c_str())); 29954 | } 29955 | 29956 | state_.activate_side_effect("parse_define_var_statement()"); 29957 | 29958 | lodge_symbol(var_name, e_st_local_variable); 29959 | 29960 | expression_node_ptr branch[2] = {0}; 29961 | 29962 | branch[0] = var_node; 29963 | branch[1] = initialisation_expression ? initialisation_expression : expression_generator_(T(0)); 29964 | 29965 | return expression_generator_(details::e_assign,branch); 29966 | } 29967 | 29968 | inline expression_node_ptr parse_define_constvar_statement() 29969 | { 29970 | if (settings_.vardef_disabled()) 29971 | { 29972 | set_error(make_error( 29973 | parser_error::e_syntax, 29974 | current_token(), 29975 | "ERR187 - Illegal const variable definition", 29976 | exprtk_error_location)); 29977 | 29978 | return error_node(); 29979 | } 29980 | else if (!token_is("const")) 29981 | { 29982 | set_error(make_error( 29983 | parser_error::e_syntax, 29984 | current_token(), 29985 | "ERR188 - Expected 'const' keyword for const-variable definition", 29986 | exprtk_error_location)); 29987 | 29988 | return error_node(); 29989 | } 29990 | else if (!token_is("var")) 29991 | { 29992 | set_error(make_error( 29993 | parser_error::e_syntax, 29994 | current_token(), 29995 | "ERR189 - Expected 'var' keyword for const-variable definition", 29996 | exprtk_error_location)); 29997 | 29998 | return error_node(); 29999 | } 30000 | 30001 | const std::string var_name = current_token().value; 30002 | 30003 | expression_node_ptr initialisation_expression = error_node(); 30004 | 30005 | if (!token_is(token_t::e_symbol)) 30006 | { 30007 | set_error(make_error( 30008 | parser_error::e_syntax, 30009 | current_token(), 30010 | "ERR190 - Expected a symbol for const-variable definition", 30011 | exprtk_error_location)); 30012 | 30013 | return error_node(); 30014 | } 30015 | else if (details::is_reserved_symbol(var_name)) 30016 | { 30017 | set_error(make_error( 30018 | parser_error::e_syntax, 30019 | current_token(), 30020 | "ERR191 - Illegal redefinition of reserved keyword: '" + var_name + "'", 30021 | exprtk_error_location)); 30022 | 30023 | return error_node(); 30024 | } 30025 | else if (symtab_store_.symbol_exists(var_name)) 30026 | { 30027 | set_error(make_error( 30028 | parser_error::e_syntax, 30029 | current_token(), 30030 | "ERR192 - Illegal redefinition of variable '" + var_name + "'", 30031 | exprtk_error_location)); 30032 | 30033 | return error_node(); 30034 | } 30035 | else if (local_variable_is_shadowed(var_name)) 30036 | { 30037 | set_error(make_error( 30038 | parser_error::e_syntax, 30039 | current_token(), 30040 | "ERR193 - Illegal redefinition of local variable: '" + var_name + "'", 30041 | exprtk_error_location)); 30042 | 30043 | return error_node(); 30044 | } 30045 | else if (!token_is(token_t::e_assign)) 30046 | { 30047 | set_error(make_error( 30048 | parser_error::e_syntax, 30049 | current_token(), 30050 | "ERR194 - Expected assignment operator in after const-variable: '" + var_name + "' definition", 30051 | exprtk_error_location)); 30052 | 30053 | return error_node(); 30054 | } 30055 | else if (0 == (initialisation_expression = parse_expression())) 30056 | { 30057 | set_error(make_error( 30058 | parser_error::e_syntax, 30059 | current_token(), 30060 | "ERR195 - Failed to parse initialisation expression for const-variable: '" + var_name + "'", 30061 | exprtk_error_location)); 30062 | 30063 | return error_node(); 30064 | } 30065 | 30066 | if (!details::is_literal_node(initialisation_expression)) 30067 | { 30068 | set_error(make_error( 30069 | parser_error::e_syntax, 30070 | current_token(), 30071 | "ERR196 - initialisation expression for const-variable: '" + var_name + "' must be a constant/literal", 30072 | exprtk_error_location)); 30073 | 30074 | free_node(node_allocator_, initialisation_expression); 30075 | 30076 | return error_node(); 30077 | } 30078 | 30079 | assert(initialisation_expression); 30080 | 30081 | const T init_value = initialisation_expression->value(); 30082 | 30083 | free_node(node_allocator_, initialisation_expression); 30084 | 30085 | expression_node_ptr var_node = reinterpret_cast<expression_node_ptr>(0); 30086 | 30087 | scope_element& se = sem_.get_element(var_name); 30088 | 30089 | if (se.name == var_name) 30090 | { 30091 | if (se.active) 30092 | { 30093 | set_error(make_error( 30094 | parser_error::e_syntax, 30095 | current_token(), 30096 | "ERR197 - Illegal redefinition of local variable: '" + var_name + "'", 30097 | exprtk_error_location)); 30098 | 30099 | return error_node(); 30100 | } 30101 | else if (scope_element::e_literal == se.type) 30102 | { 30103 | var_node = se.var_node; 30104 | se.active = true; 30105 | se.depth = state_.scope_depth; 30106 | se.ref_count++; 30107 | } 30108 | } 30109 | 30110 | if (0 == var_node) 30111 | { 30112 | const std::size_t predicted_total_lclsymb_size = sizeof(T) + sem_.total_local_symb_size_bytes(); 30113 | 30114 | if (predicted_total_lclsymb_size > settings().max_total_local_symbol_size_bytes()) 30115 | { 30116 | set_error(make_error( 30117 | parser_error::e_syntax, 30118 | current_token(), 30119 | "ERR198 - Adding variable '" + var_name + "' " 30120 | "will exceed max total local symbol size of: " + details::to_str(settings().max_total_local_symbol_size_bytes()) + " bytes, " 30121 | "current total size: " + details::to_str(sem_.total_local_symb_size_bytes()) + " bytes", 30122 | exprtk_error_location)); 30123 | 30124 | return error_node(); 30125 | } 30126 | 30127 | scope_element nse; 30128 | nse.name = var_name; 30129 | nse.active = true; 30130 | nse.ref_count = 1; 30131 | nse.type = scope_element::e_literal; 30132 | nse.depth = state_.scope_depth; 30133 | nse.data = 0; 30134 | nse.var_node = node_allocator_.allocate<literal_node_t>(init_value); 30135 | 30136 | if (!sem_.add_element(nse)) 30137 | { 30138 | set_error(make_error( 30139 | parser_error::e_syntax, 30140 | current_token(), 30141 | "ERR199 - Failed to add new local const-variable '" + var_name + "' to SEM", 30142 | exprtk_error_location)); 30143 | 30144 | sem_.free_element(nse); 30145 | 30146 | return error_node(); 30147 | } 30148 | 30149 | assert(sem_.total_local_symb_size_bytes() <= settings().max_total_local_symbol_size_bytes()); 30150 | 30151 | var_node = nse.var_node; 30152 | 30153 | exprtk_debug(("parse_define_constvar_statement() - INFO - Added new local const-variable: %s\n", nse.name.c_str())); 30154 | } 30155 | 30156 | state_.activate_side_effect("parse_define_constvar_statement()"); 30157 | 30158 | lodge_symbol(var_name, e_st_local_variable); 30159 | 30160 | return expression_generator_(var_node->value()); 30161 | } 30162 | 30163 | inline expression_node_ptr parse_uninitialised_var_statement(const std::string& var_name) 30164 | { 30165 | if ( 30166 | !token_is(token_t::e_lcrlbracket) || 30167 | !token_is(token_t::e_rcrlbracket) 30168 | ) 30169 | { 30170 | set_error(make_error( 30171 | parser_error::e_syntax, 30172 | current_token(), 30173 | "ERR200 - Expected a '{}' for uninitialised var definition", 30174 | exprtk_error_location)); 30175 | 30176 | return error_node(); 30177 | } 30178 | else if (!token_is(token_t::e_eof,prsrhlpr_t::e_hold)) 30179 | { 30180 | set_error(make_error( 30181 | parser_error::e_syntax, 30182 | current_token(), 30183 | "ERR201 - Expected ';' after uninitialised variable definition", 30184 | exprtk_error_location)); 30185 | 30186 | return error_node(); 30187 | } 30188 | 30189 | expression_node_ptr var_node = reinterpret_cast<expression_node_ptr>(0); 30190 | 30191 | scope_element& se = sem_.get_element(var_name); 30192 | 30193 | if (se.name == var_name) 30194 | { 30195 | if (se.active) 30196 | { 30197 | set_error(make_error( 30198 | parser_error::e_syntax, 30199 | current_token(), 30200 | "ERR202 - Illegal redefinition of local variable: '" + var_name + "'", 30201 | exprtk_error_location)); 30202 | 30203 | return error_node(); 30204 | } 30205 | else if (scope_element::e_variable == se.type) 30206 | { 30207 | var_node = se.var_node; 30208 | se.active = true; 30209 | se.ref_count++; 30210 | } 30211 | } 30212 | 30213 | if (0 == var_node) 30214 | { 30215 | const std::size_t predicted_total_lclsymb_size = sizeof(T) + sem_.total_local_symb_size_bytes(); 30216 | 30217 | if (predicted_total_lclsymb_size > settings().max_total_local_symbol_size_bytes()) 30218 | { 30219 | set_error(make_error( 30220 | parser_error::e_syntax, 30221 | current_token(), 30222 | "ERR203 - Adding variable '" + var_name + "' " 30223 | "will exceed max total local symbol size of: " + details::to_str(settings().max_total_local_symbol_size_bytes()) + " bytes, " 30224 | "current total size: " + details::to_str(sem_.total_local_symb_size_bytes()) + " bytes", 30225 | exprtk_error_location)); 30226 | 30227 | return error_node(); 30228 | } 30229 | 30230 | scope_element nse; 30231 | nse.name = var_name; 30232 | nse.active = true; 30233 | nse.ref_count = 1; 30234 | nse.type = scope_element::e_variable; 30235 | nse.depth = state_.scope_depth; 30236 | nse.ip_index = sem_.next_ip_index(); 30237 | nse.data = new T(T(0)); 30238 | nse.var_node = node_allocator_.allocate<variable_node_t>(*reinterpret_cast<T*>(nse.data)); 30239 | 30240 | if (!sem_.add_element(nse)) 30241 | { 30242 | set_error(make_error( 30243 | parser_error::e_syntax, 30244 | current_token(), 30245 | "ERR204 - Failed to add new local variable '" + var_name + "' to SEM", 30246 | exprtk_error_location)); 30247 | 30248 | sem_.free_element(nse); 30249 | 30250 | return error_node(); 30251 | } 30252 | 30253 | assert(sem_.total_local_symb_size_bytes() <= settings().max_total_local_symbol_size_bytes()); 30254 | 30255 | exprtk_debug(("parse_uninitialised_var_statement() - INFO - Added new local variable: %s\n", 30256 | nse.name.c_str())); 30257 | } 30258 | 30259 | lodge_symbol(var_name, e_st_local_variable); 30260 | 30261 | state_.activate_side_effect("parse_uninitialised_var_statement()"); 30262 | 30263 | return expression_generator_(T(0)); 30264 | } 30265 | 30266 | inline expression_node_ptr parse_swap_statement() 30267 | { 30268 | if (!details::imatch(current_token().value,"swap")) 30269 | { 30270 | return error_node(); 30271 | } 30272 | else 30273 | next_token(); 30274 | 30275 | if (!token_is(token_t::e_lbracket)) 30276 | { 30277 | set_error(make_error( 30278 | parser_error::e_syntax, 30279 | current_token(), 30280 | "ERR205 - Expected '(' at start of swap statement", 30281 | exprtk_error_location)); 30282 | 30283 | return error_node(); 30284 | } 30285 | 30286 | expression_node_ptr variable0 = error_node(); 30287 | expression_node_ptr variable1 = error_node(); 30288 | 30289 | bool variable0_generated = false; 30290 | bool variable1_generated = false; 30291 | 30292 | const std::string var0_name = current_token().value; 30293 | 30294 | if (!token_is(token_t::e_symbol,prsrhlpr_t::e_hold)) 30295 | { 30296 | set_error(make_error( 30297 | parser_error::e_syntax, 30298 | current_token(), 30299 | "ERR206 - Expected a symbol for variable or vector element definition", 30300 | exprtk_error_location)); 30301 | 30302 | return error_node(); 30303 | } 30304 | else if (peek_token_is(token_t::e_lsqrbracket)) 30305 | { 30306 | if (0 == (variable0 = parse_vector())) 30307 | { 30308 | set_error(make_error( 30309 | parser_error::e_syntax, 30310 | current_token(), 30311 | "ERR207 - First parameter to swap is an invalid vector element: '" + var0_name + "'", 30312 | exprtk_error_location)); 30313 | 30314 | return error_node(); 30315 | } 30316 | 30317 | variable0_generated = true; 30318 | } 30319 | else 30320 | { 30321 | if (symtab_store_.is_variable(var0_name)) 30322 | { 30323 | variable0 = symtab_store_.get_variable(var0_name); 30324 | } 30325 | 30326 | const scope_element& se = sem_.get_element(var0_name); 30327 | 30328 | if ( 30329 | (se.active) && 30330 | (se.name == var0_name) && 30331 | (scope_element::e_variable == se.type) 30332 | ) 30333 | { 30334 | variable0 = se.var_node; 30335 | } 30336 | 30337 | lodge_symbol(var0_name, e_st_variable); 30338 | 30339 | if (0 == variable0) 30340 | { 30341 | set_error(make_error( 30342 | parser_error::e_syntax, 30343 | current_token(), 30344 | "ERR208 - First parameter to swap is an invalid variable: '" + var0_name + "'", 30345 | exprtk_error_location)); 30346 | 30347 | return error_node(); 30348 | } 30349 | else 30350 | next_token(); 30351 | } 30352 | 30353 | if (!token_is(token_t::e_comma)) 30354 | { 30355 | set_error(make_error( 30356 | parser_error::e_syntax, 30357 | current_token(), 30358 | "ERR209 - Expected ',' between parameters to swap", 30359 | exprtk_error_location)); 30360 | 30361 | if (variable0_generated) 30362 | { 30363 | free_node(node_allocator_, variable0); 30364 | } 30365 | 30366 | return error_node(); 30367 | } 30368 | 30369 | const std::string var1_name = current_token().value; 30370 | 30371 | if (!token_is(token_t::e_symbol,prsrhlpr_t::e_hold)) 30372 | { 30373 | set_error(make_error( 30374 | parser_error::e_syntax, 30375 | current_token(), 30376 | "ERR210 - Expected a symbol for variable or vector element definition", 30377 | exprtk_error_location)); 30378 | 30379 | if (variable0_generated) 30380 | { 30381 | free_node(node_allocator_, variable0); 30382 | } 30383 | 30384 | return error_node(); 30385 | } 30386 | else if (peek_token_is(token_t::e_lsqrbracket)) 30387 | { 30388 | if (0 == (variable1 = parse_vector())) 30389 | { 30390 | set_error(make_error( 30391 | parser_error::e_syntax, 30392 | current_token(), 30393 | "ERR211 - Second parameter to swap is an invalid vector element: '" + var1_name + "'", 30394 | exprtk_error_location)); 30395 | 30396 | if (variable0_generated) 30397 | { 30398 | free_node(node_allocator_, variable0); 30399 | } 30400 | 30401 | return error_node(); 30402 | } 30403 | 30404 | variable1_generated = true; 30405 | } 30406 | else 30407 | { 30408 | if (symtab_store_.is_variable(var1_name)) 30409 | { 30410 | variable1 = symtab_store_.get_variable(var1_name); 30411 | } 30412 | 30413 | const scope_element& se = sem_.get_element(var1_name); 30414 | 30415 | if ( 30416 | (se.active) && 30417 | (se.name == var1_name) && 30418 | (scope_element::e_variable == se.type) 30419 | ) 30420 | { 30421 | variable1 = se.var_node; 30422 | } 30423 | 30424 | lodge_symbol(var1_name, e_st_variable); 30425 | 30426 | if (0 == variable1) 30427 | { 30428 | set_error(make_error( 30429 | parser_error::e_syntax, 30430 | current_token(), 30431 | "ERR212 - Second parameter to swap is an invalid variable: '" + var1_name + "'", 30432 | exprtk_error_location)); 30433 | 30434 | if (variable0_generated) 30435 | { 30436 | free_node(node_allocator_, variable0); 30437 | } 30438 | 30439 | return error_node(); 30440 | } 30441 | else 30442 | next_token(); 30443 | } 30444 | 30445 | if (!token_is(token_t::e_rbracket)) 30446 | { 30447 | set_error(make_error( 30448 | parser_error::e_syntax, 30449 | current_token(), 30450 | "ERR213 - Expected ')' at end of swap statement", 30451 | exprtk_error_location)); 30452 | 30453 | if (variable0_generated) 30454 | { 30455 | free_node(node_allocator_, variable0); 30456 | } 30457 | 30458 | if (variable1_generated) 30459 | { 30460 | free_node(node_allocator_, variable1); 30461 | } 30462 | 30463 | return error_node(); 30464 | } 30465 | 30466 | typedef details::variable_node<T>* variable_node_ptr; 30467 | 30468 | variable_node_ptr v0 = variable_node_ptr(0); 30469 | variable_node_ptr v1 = variable_node_ptr(0); 30470 | 30471 | expression_node_ptr result = error_node(); 30472 | 30473 | if ( 30474 | (0 != (v0 = dynamic_cast<variable_node_ptr>(variable0))) && 30475 | (0 != (v1 = dynamic_cast<variable_node_ptr>(variable1))) 30476 | ) 30477 | { 30478 | result = node_allocator_.allocate<details::swap_node<T> >(v0, v1); 30479 | 30480 | if (variable0_generated) 30481 | { 30482 | free_node(node_allocator_, variable0); 30483 | } 30484 | 30485 | if (variable1_generated) 30486 | { 30487 | free_node(node_allocator_, variable1); 30488 | } 30489 | } 30490 | else 30491 | result = node_allocator_.allocate<details::swap_generic_node<T> > 30492 | (variable0, variable1); 30493 | 30494 | state_.activate_side_effect("parse_swap_statement()"); 30495 | 30496 | return result; 30497 | } 30498 | 30499 | #ifndef exprtk_disable_return_statement 30500 | inline expression_node_ptr parse_return_statement() 30501 | { 30502 | if (state_.parsing_return_stmt) 30503 | { 30504 | set_error(make_error( 30505 | parser_error::e_syntax, 30506 | current_token(), 30507 | "ERR214 - Return call within a return call is not allowed", 30508 | exprtk_error_location)); 30509 | 30510 | return error_node(); 30511 | } 30512 | 30513 | scoped_bool_negator sbn(state_.parsing_return_stmt); 30514 | 30515 | std::vector<expression_node_ptr> arg_list; 30516 | 30517 | scoped_vec_delete<expression_node_t> sdd((*this),arg_list); 30518 | 30519 | if (!details::imatch(current_token().value,"return")) 30520 | { 30521 | return error_node(); 30522 | } 30523 | else 30524 | next_token(); 30525 | 30526 | if (!token_is(token_t::e_lsqrbracket)) 30527 | { 30528 | set_error(make_error( 30529 | parser_error::e_syntax, 30530 | current_token(), 30531 | "ERR215 - Expected '[' at start of return statement", 30532 | exprtk_error_location)); 30533 | 30534 | return error_node(); 30535 | } 30536 | else if (!token_is(token_t::e_rsqrbracket)) 30537 | { 30538 | for ( ; ; ) 30539 | { 30540 | expression_node_ptr arg = parse_expression(); 30541 | 30542 | if (0 == arg) 30543 | return error_node(); 30544 | 30545 | arg_list.push_back(arg); 30546 | 30547 | if (token_is(token_t::e_rsqrbracket)) 30548 | break; 30549 | else if (!token_is(token_t::e_comma)) 30550 | { 30551 | set_error(make_error( 30552 | parser_error::e_syntax, 30553 | current_token(), 30554 | "ERR216 - Expected ',' between values during call to return", 30555 | exprtk_error_location)); 30556 | 30557 | return error_node(); 30558 | } 30559 | } 30560 | } 30561 | else if (settings_.zero_return_disabled()) 30562 | { 30563 | set_error(make_error( 30564 | parser_error::e_syntax, 30565 | current_token(), 30566 | "ERR217 - Zero parameter return statement not allowed", 30567 | exprtk_error_location)); 30568 | 30569 | return error_node(); 30570 | } 30571 | 30572 | const lexer::token prev_token = current_token(); 30573 | 30574 | if (token_is(token_t::e_rsqrbracket)) 30575 | { 30576 | if (!arg_list.empty()) 30577 | { 30578 | set_error(make_error( 30579 | parser_error::e_syntax, 30580 | prev_token, 30581 | "ERR218 - Invalid ']' found during return call", 30582 | exprtk_error_location)); 30583 | 30584 | return error_node(); 30585 | } 30586 | } 30587 | 30588 | std::string ret_param_type_list; 30589 | 30590 | for (std::size_t i = 0; i < arg_list.size(); ++i) 30591 | { 30592 | if (0 == arg_list[i]) 30593 | return error_node(); 30594 | else if (is_ivector_node(arg_list[i])) 30595 | ret_param_type_list += 'V'; 30596 | else if (is_generally_string_node(arg_list[i])) 30597 | ret_param_type_list += 'S'; 30598 | else 30599 | ret_param_type_list += 'T'; 30600 | } 30601 | 30602 | dec_.retparam_list_.push_back(ret_param_type_list); 30603 | 30604 | expression_node_ptr result = expression_generator_.return_call(arg_list); 30605 | 30606 | sdd.delete_ptr = (0 == result); 30607 | 30608 | state_.return_stmt_present = true; 30609 | 30610 | state_.activate_side_effect("parse_return_statement()"); 30611 | 30612 | return result; 30613 | } 30614 | #else 30615 | inline expression_node_ptr parse_return_statement() 30616 | { 30617 | return error_node(); 30618 | } 30619 | #endif 30620 | 30621 | inline expression_node_ptr parse_assert_statement() 30622 | { 30623 | assert(details::imatch(current_token().value, "assert")); 30624 | 30625 | if (state_.parsing_assert_stmt) 30626 | { 30627 | set_error(make_error( 30628 | parser_error::e_syntax, 30629 | current_token(), 30630 | "ERR219 - Assert statement within an assert statement is not allowed", 30631 | exprtk_error_location)); 30632 | 30633 | return error_node(); 30634 | } 30635 | 30636 | scoped_bool_negator sbn(state_.parsing_assert_stmt); 30637 | 30638 | next_token(); 30639 | 30640 | std::vector<expression_node_ptr> assert_arg_list(3, error_node()); 30641 | scoped_vec_delete<expression_node_t> sdd((*this), assert_arg_list); 30642 | 30643 | expression_node_ptr& assert_condition = assert_arg_list[0]; 30644 | expression_node_ptr& assert_message = assert_arg_list[1]; 30645 | expression_node_ptr& assert_id = assert_arg_list[2]; 30646 | 30647 | if (!token_is(token_t::e_lbracket)) 30648 | { 30649 | set_error(make_error( 30650 | parser_error::e_syntax, 30651 | current_token(), 30652 | "ERR220 - Expected '(' at start of assert statement", 30653 | exprtk_error_location)); 30654 | 30655 | return error_node(); 30656 | } 30657 | 30658 | const token_t start_token = current_token(); 30659 | 30660 | // Parse the assert condition 30661 | if (0 == (assert_condition = parse_expression())) 30662 | { 30663 | set_error(make_error( 30664 | parser_error::e_syntax, 30665 | current_token(), 30666 | "ERR221 - Failed to parse condition for assert statement", 30667 | exprtk_error_location)); 30668 | 30669 | return error_node(); 30670 | } 30671 | 30672 | const token_t end_token = current_token(); 30673 | 30674 | if (!token_is(token_t::e_rbracket)) 30675 | { 30676 | if (!token_is(token_t::e_comma)) 30677 | { 30678 | set_error(make_error( 30679 | parser_error::e_syntax, 30680 | current_token(), 30681 | "ERR222 - Expected ',' between condition and message for assert statement", 30682 | exprtk_error_location)); 30683 | 30684 | return error_node(); 30685 | } 30686 | // Parse the assert message 30687 | else if ( 30688 | (0 == (assert_message = parse_expression())) || 30689 | !details::is_generally_string_node(assert_message) 30690 | ) 30691 | { 30692 | set_error(make_error( 30693 | parser_error::e_syntax, 30694 | current_token(), 30695 | "ERR223 - " + 30696 | (assert_message ? 30697 | std::string("Expected string for assert message") : 30698 | std::string("Failed to parse message for assert statement")), 30699 | exprtk_error_location)); 30700 | 30701 | return error_node(); 30702 | } 30703 | else if (!token_is(token_t::e_rbracket)) 30704 | { 30705 | if (!token_is(token_t::e_comma)) 30706 | { 30707 | set_error(make_error( 30708 | parser_error::e_syntax, 30709 | current_token(), 30710 | "ERR224 - Expected ',' between message and ID for assert statement", 30711 | exprtk_error_location)); 30712 | 30713 | return error_node(); 30714 | } 30715 | // Parse assert ID 30716 | else if ( 30717 | (0 == (assert_id = parse_expression())) || 30718 | !details::is_const_string_node(assert_id) 30719 | ) 30720 | { 30721 | set_error(make_error( 30722 | parser_error::e_syntax, 30723 | current_token(), 30724 | "ERR225 - " + 30725 | (assert_id ? 30726 | std::string("Expected literal string for assert ID") : 30727 | std::string("Failed to parse string for assert ID")), 30728 | exprtk_error_location)); 30729 | 30730 | return error_node(); 30731 | } 30732 | else if (!token_is(token_t::e_rbracket)) 30733 | { 30734 | set_error(make_error( 30735 | parser_error::e_syntax, 30736 | current_token(), 30737 | "ERR226 - Expected ')' at start of assert statement", 30738 | exprtk_error_location)); 30739 | 30740 | return error_node(); 30741 | } 30742 | } 30743 | } 30744 | 30745 | exprtk::assert_check::assert_context context; 30746 | context.condition = lexer().substr(start_token.position, end_token.position); 30747 | context.offet = start_token.position; 30748 | 30749 | if (0 == assert_check_) 30750 | { 30751 | exprtk_debug(("parse_assert_statement() - assert functionality is disabled. assert condition: %s\n", 30752 | context.condition.c_str())); 30753 | 30754 | return new details::null_node<T>(); 30755 | } 30756 | 30757 | #ifndef exprtk_disable_string_capabilities 30758 | if (assert_message && details::is_const_string_node(assert_message)) 30759 | { 30760 | context.message = dynamic_cast<details::string_base_node<T>*>(assert_message)->str(); 30761 | } 30762 | 30763 | if (assert_id && details::is_const_string_node(assert_id)) 30764 | { 30765 | context.id = dynamic_cast<details::string_base_node<T>*>(assert_id)->str(); 30766 | 30767 | if (assert_ids_.end() != assert_ids_.find(context.id)) 30768 | { 30769 | set_error(make_error( 30770 | parser_error::e_syntax, 30771 | current_token(), 30772 | "ERR227 - Duplicate assert ID: " + context.id, 30773 | exprtk_error_location)); 30774 | 30775 | return error_node(); 30776 | } 30777 | 30778 | assert_ids_.insert(context.id); 30779 | free_node(node_allocator_, assert_id); 30780 | } 30781 | #endif 30782 | 30783 | expression_node_ptr result_node = 30784 | expression_generator_.assert_call( 30785 | assert_condition, 30786 | assert_message, 30787 | context); 30788 | 30789 | exprtk_debug(("parse_assert_statement() - assert condition: [%s]\n", context.condition.c_str() )); 30790 | exprtk_debug(("parse_assert_statement() - assert message: [%s]\n", context.message .c_str() )); 30791 | exprtk_debug(("parse_assert_statement() - assert id: [%s]\n", context.id .c_str() )); 30792 | exprtk_debug(("parse_assert_statement() - assert offset: [%d]\n", static_cast<int>(context.offet))); 30793 | 30794 | if (0 == result_node) 30795 | { 30796 | set_error(make_error( 30797 | parser_error::e_syntax, 30798 | current_token(), 30799 | "ERR228 - Failed to synthesize assert", 30800 | exprtk_error_location)); 30801 | 30802 | return error_node(); 30803 | } 30804 | 30805 | sdd.delete_ptr = false; 30806 | return result_node; 30807 | } 30808 | 30809 | inline bool post_variable_process(const std::string& symbol) 30810 | { 30811 | if ( 30812 | peek_token_is(token_t::e_lbracket ) || 30813 | peek_token_is(token_t::e_lcrlbracket) || 30814 | peek_token_is(token_t::e_lsqrbracket) 30815 | ) 30816 | { 30817 | if (!settings_.commutative_check_enabled()) 30818 | { 30819 | set_error(make_error( 30820 | parser_error::e_syntax, 30821 | current_token(), 30822 | "ERR229 - Invalid sequence of variable '" + symbol + "' and bracket", 30823 | exprtk_error_location)); 30824 | 30825 | return false; 30826 | } 30827 | 30828 | lexer().insert_front(token_t::e_mul); 30829 | } 30830 | 30831 | return true; 30832 | } 30833 | 30834 | inline bool post_bracket_process(const typename token_t::token_type& token, expression_node_ptr& branch) 30835 | { 30836 | bool implied_mul = false; 30837 | 30838 | if (details::is_generally_string_node(branch)) 30839 | return true; 30840 | 30841 | if (details::is_ivector_node(branch)) 30842 | return true; 30843 | 30844 | const lexer::parser_helper::token_advance_mode hold = prsrhlpr_t::e_hold; 30845 | 30846 | switch (token) 30847 | { 30848 | case token_t::e_lcrlbracket : implied_mul = token_is(token_t::e_lbracket , hold) || 30849 | token_is(token_t::e_lcrlbracket, hold) || 30850 | token_is(token_t::e_lsqrbracket, hold) ; 30851 | break; 30852 | 30853 | case token_t::e_lbracket : implied_mul = token_is(token_t::e_lbracket , hold) || 30854 | token_is(token_t::e_lcrlbracket, hold) || 30855 | token_is(token_t::e_lsqrbracket, hold) ; 30856 | break; 30857 | 30858 | case token_t::e_lsqrbracket : implied_mul = token_is(token_t::e_lbracket , hold) || 30859 | token_is(token_t::e_lcrlbracket, hold) || 30860 | token_is(token_t::e_lsqrbracket, hold) ; 30861 | break; 30862 | 30863 | default : return true; 30864 | } 30865 | 30866 | if (implied_mul) 30867 | { 30868 | if (!settings_.commutative_check_enabled()) 30869 | { 30870 | set_error(make_error( 30871 | parser_error::e_syntax, 30872 | current_token(), 30873 | "ERR230 - Invalid sequence of brackets", 30874 | exprtk_error_location)); 30875 | 30876 | return false; 30877 | } 30878 | else if (token_t::e_eof != current_token().type) 30879 | { 30880 | lexer().insert_front(current_token().type); 30881 | lexer().insert_front(token_t::e_mul); 30882 | next_token(); 30883 | } 30884 | } 30885 | 30886 | return true; 30887 | } 30888 | 30889 | typedef typename interval_container_t<const void*>::interval_t interval_t; 30890 | typedef interval_container_t<const void*> immutable_memory_map_t; 30891 | typedef std::map<interval_t,token_t> immutable_symtok_map_t; 30892 | 30893 | inline interval_t make_memory_range(const T& t) 30894 | { 30895 | const T* begin = reinterpret_cast<const T*>(&t); 30896 | const T* end = begin + 1; 30897 | return interval_t(begin, end); 30898 | } 30899 | 30900 | inline interval_t make_memory_range(const T* begin, const std::size_t size) 30901 | { 30902 | return interval_t(begin, begin + size); 30903 | } 30904 | 30905 | inline interval_t make_memory_range(details::char_cptr begin, const std::size_t size) 30906 | { 30907 | return interval_t(begin, begin + size); 30908 | } 30909 | 30910 | void lodge_immutable_symbol(const lexer::token& token, const interval_t interval) 30911 | { 30912 | immutable_memory_map_.add_interval(interval); 30913 | immutable_symtok_map_[interval] = token; 30914 | } 30915 | 30916 | inline expression_node_ptr parse_symtab_symbol() 30917 | { 30918 | const std::string symbol = current_token().value; 30919 | 30920 | // Are we dealing with a variable or a special constant? 30921 | typedef typename symtab_store::variable_context var_ctxt_t; 30922 | var_ctxt_t var_ctx = symtab_store_.get_variable_context(symbol); 30923 | 30924 | if (var_ctx.variable) 30925 | { 30926 | assert(var_ctx.symbol_table); 30927 | 30928 | expression_node_ptr result_variable = var_ctx.variable; 30929 | 30930 | if (symtab_store_.is_constant_node(symbol)) 30931 | { 30932 | result_variable = expression_generator_(var_ctx.variable->value()); 30933 | } 30934 | else if (symbol_table_t::e_immutable == var_ctx.symbol_table->mutability()) 30935 | { 30936 | lodge_immutable_symbol(current_token(), make_memory_range(var_ctx.variable->ref())); 30937 | result_variable = var_ctx.variable; 30938 | } 30939 | 30940 | if (!post_variable_process(symbol)) 30941 | return error_node(); 30942 | 30943 | lodge_symbol(symbol, e_st_variable); 30944 | 30945 | next_token(); 30946 | 30947 | return result_variable; 30948 | } 30949 | 30950 | // Are we dealing with a locally defined variable, vector or string? 30951 | if (!sem_.empty()) 30952 | { 30953 | scope_element& se = sem_.get_active_element(symbol); 30954 | 30955 | if (se.active && details::imatch(se.name, symbol)) 30956 | { 30957 | if ( 30958 | (scope_element::e_variable == se.type) || 30959 | (scope_element::e_literal == se.type) 30960 | ) 30961 | { 30962 | se.active = true; 30963 | lodge_symbol(symbol, e_st_local_variable); 30964 | 30965 | if (!post_variable_process(symbol)) 30966 | return error_node(); 30967 | 30968 | next_token(); 30969 | 30970 | return (scope_element::e_variable == se.type) ? 30971 | se.var_node : 30972 | expression_generator_(se.var_node->value()); 30973 | } 30974 | else if (scope_element::e_vector == se.type) 30975 | { 30976 | return parse_vector(); 30977 | } 30978 | #ifndef exprtk_disable_string_capabilities 30979 | else if (scope_element::e_string == se.type) 30980 | { 30981 | return parse_string(); 30982 | } 30983 | #endif 30984 | } 30985 | } 30986 | 30987 | #ifndef exprtk_disable_string_capabilities 30988 | // Are we dealing with a string variable? 30989 | if (symtab_store_.is_stringvar(symbol)) 30990 | { 30991 | return parse_string(); 30992 | } 30993 | #endif 30994 | 30995 | { 30996 | // Are we dealing with a function? 30997 | ifunction<T>* function = symtab_store_.get_function(symbol); 30998 | 30999 | if (function) 31000 | { 31001 | lodge_symbol(symbol, e_st_function); 31002 | 31003 | expression_node_ptr func_node = 31004 | parse_function_invocation(function,symbol); 31005 | 31006 | if (func_node) 31007 | return func_node; 31008 | else 31009 | { 31010 | set_error(make_error( 31011 | parser_error::e_syntax, 31012 | current_token(), 31013 | "ERR231 - Failed to generate node for function: '" + symbol + "'", 31014 | exprtk_error_location)); 31015 | 31016 | return error_node(); 31017 | } 31018 | } 31019 | } 31020 | 31021 | { 31022 | // Are we dealing with a vararg function? 31023 | ivararg_function<T>* vararg_function = symtab_store_.get_vararg_function(symbol); 31024 | 31025 | if (vararg_function) 31026 | { 31027 | lodge_symbol(symbol, e_st_function); 31028 | 31029 | expression_node_ptr vararg_func_node = 31030 | parse_vararg_function_call(vararg_function, symbol); 31031 | 31032 | if (vararg_func_node) 31033 | return vararg_func_node; 31034 | else 31035 | { 31036 | set_error(make_error( 31037 | parser_error::e_syntax, 31038 | current_token(), 31039 | "ERR232 - Failed to generate node for vararg function: '" + symbol + "'", 31040 | exprtk_error_location)); 31041 | 31042 | return error_node(); 31043 | } 31044 | } 31045 | } 31046 | 31047 | { 31048 | // Are we dealing with a vararg generic function? 31049 | igeneric_function<T>* generic_function = symtab_store_.get_generic_function(symbol); 31050 | 31051 | if (generic_function) 31052 | { 31053 | lodge_symbol(symbol, e_st_function); 31054 | 31055 | expression_node_ptr genericfunc_node = 31056 | parse_generic_function_call(generic_function, symbol); 31057 | 31058 | if (genericfunc_node) 31059 | return genericfunc_node; 31060 | else 31061 | { 31062 | set_error(make_error( 31063 | parser_error::e_syntax, 31064 | current_token(), 31065 | "ERR233 - Failed to generate node for generic function: '" + symbol + "'", 31066 | exprtk_error_location)); 31067 | 31068 | return error_node(); 31069 | } 31070 | } 31071 | } 31072 | 31073 | #ifndef exprtk_disable_string_capabilities 31074 | { 31075 | // Are we dealing with a vararg string returning function? 31076 | igeneric_function<T>* string_function = symtab_store_.get_string_function(symbol); 31077 | 31078 | if (string_function) 31079 | { 31080 | lodge_symbol(symbol, e_st_function); 31081 | 31082 | expression_node_ptr stringfunc_node = 31083 | parse_string_function_call(string_function, symbol); 31084 | 31085 | if (stringfunc_node) 31086 | return stringfunc_node; 31087 | else 31088 | { 31089 | set_error(make_error( 31090 | parser_error::e_syntax, 31091 | current_token(), 31092 | "ERR234 - Failed to generate node for string function: '" + symbol + "'", 31093 | exprtk_error_location)); 31094 | 31095 | return error_node(); 31096 | } 31097 | } 31098 | } 31099 | 31100 | { 31101 | // Are we dealing with a vararg overloaded scalar/string returning function? 31102 | igeneric_function<T>* overload_function = symtab_store_.get_overload_function(symbol); 31103 | 31104 | if (overload_function) 31105 | { 31106 | lodge_symbol(symbol, e_st_function); 31107 | 31108 | expression_node_ptr overloadfunc_node = 31109 | parse_overload_function_call(overload_function, symbol); 31110 | 31111 | if (overloadfunc_node) 31112 | return overloadfunc_node; 31113 | else 31114 | { 31115 | set_error(make_error( 31116 | parser_error::e_syntax, 31117 | current_token(), 31118 | "ERR235 - Failed to generate node for overload function: '" + symbol + "'", 31119 | exprtk_error_location)); 31120 | 31121 | return error_node(); 31122 | } 31123 | } 31124 | } 31125 | #endif 31126 | 31127 | // Are we dealing with a vector? 31128 | if (symtab_store_.is_vector(symbol)) 31129 | { 31130 | lodge_symbol(symbol, e_st_vector); 31131 | return parse_vector(); 31132 | } 31133 | 31134 | if (details::is_reserved_symbol(symbol)) 31135 | { 31136 | if ( 31137 | settings_.function_enabled(symbol) || 31138 | !details::is_base_function(symbol) 31139 | ) 31140 | { 31141 | set_error(make_error( 31142 | parser_error::e_syntax, 31143 | current_token(), 31144 | "ERR236 - Invalid use of reserved symbol '" + symbol + "'", 31145 | exprtk_error_location)); 31146 | 31147 | return error_node(); 31148 | } 31149 | } 31150 | 31151 | // Should we handle unknown symbols? 31152 | if (resolve_unknown_symbol_ && unknown_symbol_resolver_) 31153 | { 31154 | if (!(settings_.rsrvd_sym_usr_disabled() && details::is_reserved_symbol(symbol))) 31155 | { 31156 | symbol_table_t& symtab = symtab_store_.get_symbol_table(); 31157 | 31158 | std::string error_message; 31159 | 31160 | if (unknown_symbol_resolver::e_usrmode_default == unknown_symbol_resolver_->mode) 31161 | { 31162 | T default_value = T(0); 31163 | 31164 | typename unknown_symbol_resolver::usr_symbol_type usr_symbol_type = unknown_symbol_resolver::e_usr_unknown_type; 31165 | 31166 | if (unknown_symbol_resolver_->process(symbol, usr_symbol_type, default_value, error_message)) 31167 | { 31168 | bool create_result = false; 31169 | 31170 | switch (usr_symbol_type) 31171 | { 31172 | case unknown_symbol_resolver::e_usr_variable_type : 31173 | create_result = symtab.create_variable(symbol, default_value); 31174 | break; 31175 | 31176 | case unknown_symbol_resolver::e_usr_constant_type : 31177 | create_result = symtab.add_constant(symbol, default_value); 31178 | break; 31179 | 31180 | default : create_result = false; 31181 | } 31182 | 31183 | if (create_result) 31184 | { 31185 | expression_node_ptr var = symtab_store_.get_variable(symbol); 31186 | 31187 | if (var) 31188 | { 31189 | if (symtab_store_.is_constant_node(symbol)) 31190 | { 31191 | var = expression_generator_(var->value()); 31192 | } 31193 | 31194 | lodge_symbol(symbol, e_st_variable); 31195 | 31196 | if (!post_variable_process(symbol)) 31197 | return error_node(); 31198 | 31199 | next_token(); 31200 | 31201 | return var; 31202 | } 31203 | } 31204 | } 31205 | 31206 | set_error(make_error( 31207 | parser_error::e_symtab, 31208 | current_token(), 31209 | "ERR237 - Failed to create variable: '" + symbol + "'" + 31210 | (error_message.empty() ? "" : " - " + error_message), 31211 | exprtk_error_location)); 31212 | 31213 | } 31214 | else if (unknown_symbol_resolver::e_usrmode_extended == unknown_symbol_resolver_->mode) 31215 | { 31216 | if (unknown_symbol_resolver_->process(symbol, symtab, error_message)) 31217 | { 31218 | expression_node_ptr result = parse_symtab_symbol(); 31219 | 31220 | if (result) 31221 | { 31222 | return result; 31223 | } 31224 | } 31225 | 31226 | set_error(make_error( 31227 | parser_error::e_symtab, 31228 | current_token(), 31229 | "ERR238 - Failed to resolve symbol: '" + symbol + "'" + 31230 | (error_message.empty() ? "" : " - " + error_message), 31231 | exprtk_error_location)); 31232 | } 31233 | 31234 | return error_node(); 31235 | } 31236 | } 31237 | 31238 | set_error(make_error( 31239 | parser_error::e_syntax, 31240 | current_token(), 31241 | "ERR239 - Undefined symbol: '" + symbol + "'", 31242 | exprtk_error_location)); 31243 | 31244 | return error_node(); 31245 | } 31246 | 31247 | inline expression_node_ptr check_block_statement_closure(expression_node_ptr expression) 31248 | { 31249 | if ( 31250 | expression && 31251 | ( 31252 | (current_token().type == token_t::e_symbol) || 31253 | (current_token().type == token_t::e_number) 31254 | ) 31255 | ) 31256 | { 31257 | free_node(node_allocator_, expression); 31258 | 31259 | set_error(make_error( 31260 | parser_error::e_syntax, 31261 | current_token(), 31262 | "ERR240 - Invalid syntax '" + current_token().value + "' possible missing operator or context", 31263 | exprtk_error_location)); 31264 | 31265 | return error_node(); 31266 | } 31267 | 31268 | return expression; 31269 | } 31270 | 31271 | inline expression_node_ptr parse_symbol() 31272 | { 31273 | static const std::string symbol_if = "if" ; 31274 | static const std::string symbol_while = "while" ; 31275 | static const std::string symbol_repeat = "repeat" ; 31276 | static const std::string symbol_for = "for" ; 31277 | static const std::string symbol_switch = "switch" ; 31278 | static const std::string symbol_null = "null" ; 31279 | static const std::string symbol_break = "break" ; 31280 | static const std::string symbol_continue = "continue" 31281 | static const std::string symbol_var = "var" ; 31282 | static const std::string symbol_const = "const" ; 31283 | static const std::string symbol_swap = "swap" ; 31284 | static const std::string symbol_return = "return" ; 31285 | static const std::string symbol_not = "not" ; 31286 | static const std::string symbol_assert = "assert" ; 31287 | 31288 | const std::string symbol = current_token().value; 31289 | 31290 | if (valid_vararg_operation(symbol)) 31291 | { 31292 | return parse_vararg_function(); 31293 | } 31294 | else if (details::imatch(symbol, symbol_not)) 31295 | { 31296 | return parse_not_statement(); 31297 | } 31298 | else if (valid_base_operation(symbol)) 31299 | { 31300 | return parse_base_operation(); 31301 | } 31302 | else if ( 31303 | details::imatch(symbol, symbol_if) && 31304 | settings_.control_struct_enabled(symbol) 31305 | ) 31306 | { 31307 | return parse_conditional_statement(); 31308 | } 31309 | else if ( 31310 | details::imatch(symbol, symbol_while) && 31311 | settings_.control_struct_enabled(symbol) 31312 | ) 31313 | { 31314 | return check_block_statement_closure(parse_while_loop()); 31315 | } 31316 | else if ( 31317 | details::imatch(symbol, symbol_repeat) && 31318 | settings_.control_struct_enabled(symbol) 31319 | ) 31320 | { 31321 | return check_block_statement_closure(parse_repeat_until_loop()); 31322 | } 31323 | else if ( 31324 | details::imatch(symbol, symbol_for) && 31325 | settings_.control_struct_enabled(symbol) 31326 | ) 31327 | { 31328 | return check_block_statement_closure(parse_for_loop()); 31329 | } 31330 | else if ( 31331 | details::imatch(symbol, symbol_switch) && 31332 | settings_.control_struct_enabled(symbol) 31333 | ) 31334 | { 31335 | return check_block_statement_closure(parse_switch_statement()); 31336 | } 31337 | else if (details::is_valid_sf_symbol(symbol)) 31338 | { 31339 | return parse_special_function(); 31340 | } 31341 | else if (details::imatch(symbol, symbol_null)) 31342 | { 31343 | return parse_null_statement(); 31344 | } 31345 | #ifndef exprtk_disable_break_continue 31346 | else if (details::imatch(symbol, symbol_break)) 31347 | { 31348 | return parse_break_statement(); 31349 | } 31350 | else if (details::imatch(symbol, symbol_continue)) 31351 | { 31352 | return parse_continue_statement(); 31353 | } 31354 | #endif 31355 | else if (details::imatch(symbol, symbol_var)) 31356 | { 31357 | return parse_define_var_statement(); 31358 | } 31359 | else if (details::imatch(symbol, symbol_const)) 31360 | { 31361 | return parse_define_constvar_statement(); 31362 | } 31363 | else if (details::imatch(symbol, symbol_swap)) 31364 | { 31365 | return parse_swap_statement(); 31366 | } 31367 | #ifndef exprtk_disable_return_statement 31368 | else if ( 31369 | details::imatch(symbol, symbol_return) && 31370 | settings_.control_struct_enabled(symbol) 31371 | ) 31372 | { 31373 | return check_block_statement_closure(parse_return_statement()); 31374 | } 31375 | #endif 31376 | else if (details::imatch(symbol, symbol_assert)) 31377 | { 31378 | return parse_assert_statement(); 31379 | } 31380 | else if (symtab_store_.valid() || !sem_.empty()) 31381 | { 31382 | return parse_symtab_symbol(); 31383 | } 31384 | else 31385 | { 31386 | set_error(make_error( 31387 | parser_error::e_symtab, 31388 | current_token(), 31389 | "ERR241 - Unknown variable or function encountered. Symbol table(s) " 31390 | "is either invalid or does not contain symbol: '" + symbol + "'", 31391 | exprtk_error_location)); 31392 | 31393 | return error_node(); 31394 | } 31395 | } 31396 | 31397 | inline expression_node_ptr parse_branch(precedence_level precedence = e_level00) 31398 | { 31399 | stack_limit_handler slh(*this); 31400 | 31401 | if (!slh) 31402 | { 31403 | return error_node(); 31404 | } 31405 | 31406 | expression_node_ptr branch = error_node(); 31407 | 31408 | if (token_t::e_number == current_token().type) 31409 | { 31410 | T numeric_value = T(0); 31411 | 31412 | if (details::string_to_real(current_token().value, numeric_value)) 31413 | { 31414 | expression_node_ptr literal_exp = expression_generator_(numeric_value); 31415 | 31416 | if (0 == literal_exp) 31417 | { 31418 | set_error(make_error( 31419 | parser_error::e_numeric, 31420 | current_token(), 31421 | "ERR242 - Failed generate node for scalar: '" + current_token().value + "'", 31422 | exprtk_error_location)); 31423 | 31424 | return error_node(); 31425 | } 31426 | 31427 | next_token(); 31428 | branch = literal_exp; 31429 | } 31430 | else 31431 | { 31432 | set_error(make_error( 31433 | parser_error::e_numeric, 31434 | current_token(), 31435 | "ERR243 - Failed to convert '" + current_token().value + "' to a number", 31436 | exprtk_error_location)); 31437 | 31438 | return error_node(); 31439 | } 31440 | } 31441 | else if (token_t::e_symbol == current_token().type) 31442 | { 31443 | branch = parse_symbol(); 31444 | } 31445 | #ifndef exprtk_disable_string_capabilities 31446 | else if (token_t::e_string == current_token().type) 31447 | { 31448 | branch = parse_const_string(); 31449 | } 31450 | #endif 31451 | else if (token_t::e_lbracket == current_token().type) 31452 | { 31453 | next_token(); 31454 | 31455 | if (0 == (branch = parse_expression())) 31456 | { 31457 | return error_node(); 31458 | } 31459 | 31460 | token_is(token_t::e_eof); 31461 | 31462 | if (!token_is(token_t::e_rbracket)) 31463 | { 31464 | set_error(make_error( 31465 | parser_error::e_syntax, 31466 | current_token(), 31467 | "ERR244 - Expected ')' instead of: '" + current_token().value + "'", 31468 | exprtk_error_location)); 31469 | 31470 | details::free_node(node_allocator_, branch); 31471 | 31472 | return error_node(); 31473 | } 31474 | else if (!post_bracket_process(token_t::e_lbracket,branch)) 31475 | { 31476 | details::free_node(node_allocator_, branch); 31477 | 31478 | return error_node(); 31479 | } 31480 | 31481 | parse_pending_vector_index_operator(branch); 31482 | } 31483 | else if (token_t::e_lsqrbracket == current_token().type) 31484 | { 31485 | next_token(); 31486 | 31487 | if (0 == (branch = parse_expression())) 31488 | return error_node(); 31489 | else if (!token_is(token_t::e_rsqrbracket)) 31490 | { 31491 | set_error(make_error( 31492 | parser_error::e_syntax, 31493 | current_token(), 31494 | "ERR245 - Expected ']' instead of: '" + current_token().value + "'", 31495 | exprtk_error_location)); 31496 | 31497 | details::free_node(node_allocator_, branch); 31498 | 31499 | return error_node(); 31500 | } 31501 | else if (!post_bracket_process(token_t::e_lsqrbracket,branch)) 31502 | { 31503 | details::free_node(node_allocator_, branch); 31504 | 31505 | return error_node(); 31506 | } 31507 | } 31508 | else if (token_t::e_lcrlbracket == current_token().type) 31509 | { 31510 | next_token(); 31511 | 31512 | if (0 == (branch = parse_expression())) 31513 | return error_node(); 31514 | else if (!token_is(token_t::e_rcrlbracket)) 31515 | { 31516 | set_error(make_error( 31517 | parser_error::e_syntax, 31518 | current_token(), 31519 | "ERR246 - Expected '}' instead of: '" + current_token().value + "'", 31520 | exprtk_error_location)); 31521 | 31522 | details::free_node(node_allocator_, branch); 31523 | 31524 | return error_node(); 31525 | } 31526 | else if (!post_bracket_process(token_t::e_lcrlbracket,branch)) 31527 | { 31528 | details::free_node(node_allocator_, branch); 31529 | 31530 | return error_node(); 31531 | } 31532 | } 31533 | else if (token_t::e_sub == current_token().type) 31534 | { 31535 | next_token(); 31536 | branch = parse_expression(e_level11); 31537 | 31538 | if ( 31539 | branch && 31540 | !( 31541 | details::is_neg_unary_node (branch) && 31542 | simplify_unary_negation_branch(branch) 31543 | ) 31544 | ) 31545 | { 31546 | expression_node_ptr result = expression_generator_(details::e_neg,branch); 31547 | 31548 | if (0 == result) 31549 | { 31550 | details::free_node(node_allocator_, branch); 31551 | 31552 | return error_node(); 31553 | } 31554 | else 31555 | branch = result; 31556 | } 31557 | } 31558 | else if (token_t::e_add == current_token().type) 31559 | { 31560 | next_token(); 31561 | branch = parse_expression(e_level13); 31562 | } 31563 | else if (token_t::e_eof == current_token().type) 31564 | { 31565 | set_error(make_error( 31566 | parser_error::e_syntax, 31567 | current_token(), 31568 | "ERR247 - Premature end of expression[1]", 31569 | exprtk_error_location)); 31570 | 31571 | return error_node(); 31572 | } 31573 | else 31574 | { 31575 | set_error(make_error( 31576 | parser_error::e_syntax, 31577 | current_token(), 31578 | "ERR248 - Premature end of expression[2]", 31579 | exprtk_error_location)); 31580 | 31581 | return error_node(); 31582 | } 31583 | 31584 | if ( 31585 | branch && 31586 | (e_level00 == precedence) && 31587 | token_is(token_t::e_ternary,prsrhlpr_t::e_hold) 31588 | ) 31589 | { 31590 | branch = parse_ternary_conditional_statement(branch); 31591 | } 31592 | 31593 | parse_pending_string_rangesize(branch); 31594 | 31595 | return branch; 31596 | } 31597 | 31598 | template <typename Type> 31599 | class expression_generator 31600 | { 31601 | public: 31602 | 31603 | typedef details::expression_node<Type>* expression_node_ptr; 31604 | typedef expression_node_ptr (*synthesize_functor_t)(expression_generator<T>&, const details::operator_type& operation, expression_node_ptr (&branch)[2]); 31605 | typedef std::map<std::string,synthesize_functor_t> synthesize_map_t; 31606 | typedef typename exprtk::parser<Type> parser_t; 31607 | typedef const Type& vtype; 31608 | typedef const Type ctype; 31609 | 31610 | inline void init_synthesize_map() 31611 | { 31612 | #ifndef exprtk_disable_enhanced_features 31613 | synthesize_map_["(v)o(v)"] = synthesize_vov_expression::process; 31614 | synthesize_map_["(c)o(v)"] = synthesize_cov_expression::process; 31615 | synthesize_map_["(v)o(c)"] = synthesize_voc_expression::process; 31616 | 31617 | #define register_synthezier(S) \ 31618 | synthesize_map_[S ::node_type::id()] = S ::process; \ 31619 | 31620 | register_synthezier(synthesize_vovov_expression0) 31621 | register_synthezier(synthesize_vovov_expression1) 31622 | register_synthezier(synthesize_vovoc_expression0) 31623 | register_synthezier(synthesize_vovoc_expression1) 31624 | register_synthezier(synthesize_vocov_expression0) 31625 | register_synthezier(synthesize_vocov_expression1) 31626 | register_synthezier(synthesize_covov_expression0) 31627 | register_synthezier(synthesize_covov_expression1) 31628 | register_synthezier(synthesize_covoc_expression0) 31629 | register_synthezier(synthesize_covoc_expression1) 31630 | register_synthezier(synthesize_cocov_expression1) 31631 | register_synthezier(synthesize_vococ_expression0) 31632 | 31633 | register_synthezier(synthesize_vovovov_expression0) 31634 | register_synthezier(synthesize_vovovoc_expression0) 31635 | register_synthezier(synthesize_vovocov_expression0) 31636 | register_synthezier(synthesize_vocovov_expression0) 31637 | register_synthezier(synthesize_covovov_expression0) 31638 | register_synthezier(synthesize_covocov_expression0) 31639 | register_synthezier(synthesize_vocovoc_expression0) 31640 | register_synthezier(synthesize_covovoc_expression0) 31641 | register_synthezier(synthesize_vococov_expression0) 31642 | 31643 | register_synthezier(synthesize_vovovov_expression1) 31644 | register_synthezier(synthesize_vovovoc_expression1) 31645 | register_synthezier(synthesize_vovocov_expression1) 31646 | register_synthezier(synthesize_vocovov_expression1) 31647 | register_synthezier(synthesize_covovov_expression1) 31648 | register_synthezier(synthesize_covocov_expression1) 31649 | register_synthezier(synthesize_vocovoc_expression1) 31650 | register_synthezier(synthesize_covovoc_expression1) 31651 | register_synthezier(synthesize_vococov_expression1) 31652 | 31653 | register_synthezier(synthesize_vovovov_expression2) 31654 | register_synthezier(synthesize_vovovoc_expression2) 31655 | register_synthezier(synthesize_vovocov_expression2) 31656 | register_synthezier(synthesize_vocovov_expression2) 31657 | register_synthezier(synthesize_covovov_expression2) 31658 | register_synthezier(synthesize_covocov_expression2) 31659 | register_synthezier(synthesize_vocovoc_expression2) 31660 | register_synthezier(synthesize_covovoc_expression2) 31661 | 31662 | register_synthezier(synthesize_vovovov_expression3) 31663 | register_synthezier(synthesize_vovovoc_expression3) 31664 | register_synthezier(synthesize_vovocov_expression3) 31665 | register_synthezier(synthesize_vocovov_expression3) 31666 | register_synthezier(synthesize_covovov_expression3) 31667 | register_synthezier(synthesize_covocov_expression3) 31668 | register_synthezier(synthesize_vocovoc_expression3) 31669 | register_synthezier(synthesize_covovoc_expression3) 31670 | register_synthezier(synthesize_vococov_expression3) 31671 | 31672 | register_synthezier(synthesize_vovovov_expression4) 31673 | register_synthezier(synthesize_vovovoc_expression4) 31674 | register_synthezier(synthesize_vovocov_expression4) 31675 | register_synthezier(synthesize_vocovov_expression4) 31676 | register_synthezier(synthesize_covovov_expression4) 31677 | register_synthezier(synthesize_covocov_expression4) 31678 | register_synthezier(synthesize_vocovoc_expression4) 31679 | register_synthezier(synthesize_covovoc_expression4) 31680 | 31681 | #undef register_synthezier 31682 | #endif 31683 | } 31684 | 31685 | inline void set_parser(parser_t& p) 31686 | { 31687 | parser_ = &p; 31688 | } 31689 | 31690 | inline void set_uom(unary_op_map_t& unary_op_map) 31691 | { 31692 | unary_op_map_ = &unary_op_map; 31693 | } 31694 | 31695 | inline void set_bom(binary_op_map_t& binary_op_map) 31696 | { 31697 | binary_op_map_ = &binary_op_map; 31698 | } 31699 | 31700 | inline void set_ibom(inv_binary_op_map_t& inv_binary_op_map) 31701 | { 31702 | inv_binary_op_map_ = &inv_binary_op_map; 31703 | } 31704 | 31705 | inline void set_sf3m(sf3_map_t& sf3_map) 31706 | { 31707 | sf3_map_ = &sf3_map; 31708 | } 31709 | 31710 | inline void set_sf4m(sf4_map_t& sf4_map) 31711 | { 31712 | sf4_map_ = &sf4_map; 31713 | } 31714 | 31715 | inline void set_allocator(details::node_allocator& na) 31716 | { 31717 | node_allocator_ = &na; 31718 | } 31719 | 31720 | inline void set_strength_reduction_state(const bool enabled) 31721 | { 31722 | strength_reduction_enabled_ = enabled; 31723 | } 31724 | 31725 | inline bool strength_reduction_enabled() const 31726 | { 31727 | return strength_reduction_enabled_; 31728 | } 31729 | 31730 | inline bool valid_operator(const details::operator_type& operation, binary_functor_t& bop) 31731 | { 31732 | typename binary_op_map_t::iterator bop_itr = binary_op_map_->find(operation); 31733 | 31734 | if (binary_op_map_->end() == bop_itr) 31735 | return false; 31736 | 31737 | bop = bop_itr->second; 31738 | 31739 | return true; 31740 | } 31741 | 31742 | inline bool valid_operator(const details::operator_type& operation, unary_functor_t& uop) 31743 | { 31744 | typename unary_op_map_t::iterator uop_itr = unary_op_map_->find(operation); 31745 | 31746 | if ((*unary_op_map_).end() == uop_itr) 31747 | return false; 31748 | 31749 | uop = uop_itr->second; 31750 | 31751 | return true; 31752 | } 31753 | 31754 | inline details::operator_type get_operator(const binary_functor_t& bop) const 31755 | { 31756 | return (*inv_binary_op_map_).find(bop)->second; 31757 | } 31758 | 31759 | inline expression_node_ptr operator() (const Type& v) const 31760 | { 31761 | return node_allocator_->allocate<literal_node_t>(v); 31762 | } 31763 | 31764 | #ifndef exprtk_disable_string_capabilities 31765 | inline expression_node_ptr operator() (const std::string& s) const 31766 | { 31767 | return node_allocator_->allocate<string_literal_node_t>(s); 31768 | } 31769 | 31770 | inline expression_node_ptr operator() (std::string& s, range_t& rp) const 31771 | { 31772 | return node_allocator_->allocate_rr<string_range_node_t>(s,rp); 31773 | } 31774 | 31775 | inline expression_node_ptr operator() (const std::string& s, range_t& rp) const 31776 | { 31777 | return node_allocator_->allocate_tt<const_string_range_node_t>(s,rp); 31778 | } 31779 | 31780 | inline expression_node_ptr operator() (expression_node_ptr branch, range_t& rp) const 31781 | { 31782 | if (is_generally_string_node(branch)) 31783 | return node_allocator_->allocate_tt<generic_string_range_node_t>(branch,rp); 31784 | else 31785 | return error_node(); 31786 | } 31787 | #endif 31788 | 31789 | inline bool unary_optimisable(const details::operator_type& operation) const 31790 | { 31791 | return (details::e_abs == operation) || (details::e_acos == operation) || 31792 | (details::e_acosh == operation) || (details::e_asin == operation) || 31793 | (details::e_asinh == operation) || (details::e_atan == operation) || 31794 | (details::e_atanh == operation) || (details::e_ceil == operation) || 31795 | (details::e_cos == operation) || (details::e_cosh == operation) || 31796 | (details::e_exp == operation) || (details::e_expm1 == operation) || 31797 | (details::e_floor == operation) || (details::e_log == operation) || 31798 | (details::e_log10 == operation) || (details::e_log2 == operation) || 31799 | (details::e_log1p == operation) || (details::e_neg == operation) || 31800 | (details::e_pos == operation) || (details::e_round == operation) || 31801 | (details::e_sin == operation) || (details::e_sinc == operation) || 31802 | (details::e_sinh == operation) || (details::e_sqrt == operation) || 31803 | (details::e_tan == operation) || (details::e_tanh == operation) || 31804 | (details::e_cot == operation) || (details::e_sec == operation) || 31805 | (details::e_csc == operation) || (details::e_r2d == operation) || 31806 | (details::e_d2r == operation) || (details::e_d2g == operation) || 31807 | (details::e_g2d == operation) || (details::e_notl == operation) || 31808 | (details::e_sgn == operation) || (details::e_erf == operation) || 31809 | (details::e_erfc == operation) || (details::e_ncdf == operation) || 31810 | (details::e_frac == operation) || (details::e_trunc == operation) ; 31811 | } 31812 | 31813 | inline bool sf3_optimisable(const std::string& sf3id, trinary_functor_t& tfunc) const 31814 | { 31815 | typename sf3_map_t::const_iterator itr = sf3_map_->find(sf3id); 31816 | 31817 | if (sf3_map_->end() == itr) 31818 | return false; 31819 | else 31820 | tfunc = itr->second.first; 31821 | 31822 | return true; 31823 | } 31824 | 31825 | inline bool sf4_optimisable(const std::string& sf4id, quaternary_functor_t& qfunc) const 31826 | { 31827 | typename sf4_map_t::const_iterator itr = sf4_map_->find(sf4id); 31828 | 31829 | if (sf4_map_->end() == itr) 31830 | return false; 31831 | else 31832 | qfunc = itr->second.first; 31833 | 31834 | return true; 31835 | } 31836 | 31837 | inline bool sf3_optimisable(const std::string& sf3id, details::operator_type& operation) const 31838 | { 31839 | typename sf3_map_t::const_iterator itr = sf3_map_->find(sf3id); 31840 | 31841 | if (sf3_map_->end() == itr) 31842 | return false; 31843 | else 31844 | operation = itr->second.second; 31845 | 31846 | return true; 31847 | } 31848 | 31849 | inline bool sf4_optimisable(const std::string& sf4id, details::operator_type& operation) const 31850 | { 31851 | typename sf4_map_t::const_iterator itr = sf4_map_->find(sf4id); 31852 | 31853 | if (sf4_map_->end() == itr) 31854 | return false; 31855 | else 31856 | operation = itr->second.second; 31857 | 31858 | return true; 31859 | } 31860 | 31861 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr (&branch)[1]) 31862 | { 31863 | if (0 == branch[0]) 31864 | { 31865 | return error_node(); 31866 | } 31867 | else if (details::is_null_node(branch[0])) 31868 | { 31869 | return branch[0]; 31870 | } 31871 | else if (details::is_break_node(branch[0])) 31872 | { 31873 | return error_node(); 31874 | } 31875 | else if (details::is_continue_node(branch[0])) 31876 | { 31877 | return error_node(); 31878 | } 31879 | else if (details::is_constant_node(branch[0])) 31880 | { 31881 | return synthesize_expression<unary_node_t,1>(operation,branch); 31882 | } 31883 | else if (unary_optimisable(operation) && details::is_variable_node(branch[0])) 31884 | { 31885 | return synthesize_uv_expression(operation,branch); 31886 | } 31887 | else if (unary_optimisable(operation) && details::is_ivector_node(branch[0])) 31888 | { 31889 | return synthesize_uvec_expression(operation,branch); 31890 | } 31891 | else 31892 | return synthesize_unary_expression(operation,branch); 31893 | } 31894 | 31895 | inline bool is_assignment_operation(const details::operator_type& operation) const 31896 | { 31897 | return ( 31898 | (details::e_addass == operation) || 31899 | (details::e_subass == operation) || 31900 | (details::e_mulass == operation) || 31901 | (details::e_divass == operation) || 31902 | (details::e_modass == operation) 31903 | ) && 31904 | parser_->settings_.assignment_enabled(operation); 31905 | } 31906 | 31907 | #ifndef exprtk_disable_string_capabilities 31908 | inline bool valid_string_operation(const details::operator_type& operation) const 31909 | { 31910 | return (details::e_add == operation) || 31911 | (details::e_lt == operation) || 31912 | (details::e_lte == operation) || 31913 | (details::e_gt == operation) || 31914 | (details::e_gte == operation) || 31915 | (details::e_eq == operation) || 31916 | (details::e_ne == operation) || 31917 | (details::e_in == operation) || 31918 | (details::e_like == operation) || 31919 | (details::e_ilike == operation) || 31920 | (details::e_assign == operation) || 31921 | (details::e_addass == operation) || 31922 | (details::e_swap == operation) ; 31923 | } 31924 | #else 31925 | inline bool valid_string_operation(const details::operator_type&) const 31926 | { 31927 | return false; 31928 | } 31929 | #endif 31930 | 31931 | inline std::string to_str(const details::operator_type& operation) const 31932 | { 31933 | switch (operation) 31934 | { 31935 | case details::e_add : return "+" ; 31936 | case details::e_sub : return "-" ; 31937 | case details::e_mul : return "*" ; 31938 | case details::e_div : return "/" ; 31939 | case details::e_mod : return "%" ; 31940 | case details::e_pow : return "^" ; 31941 | case details::e_lt : return "<" ; 31942 | case details::e_lte : return "<=" ; 31943 | case details::e_gt : return ">" ; 31944 | case details::e_gte : return ">=" ; 31945 | case details::e_eq : return "==" ; 31946 | case details::e_ne : return "!=" ; 31947 | case details::e_and : return "and" ; 31948 | case details::e_nand : return "nand" ; 31949 | case details::e_or : return "or" ; 31950 | case details::e_nor : return "nor" ; 31951 | case details::e_xor : return "xor" ; 31952 | case details::e_xnor : return "xnor" ; 31953 | default : return "UNKNOWN" 31954 | } 31955 | } 31956 | 31957 | inline bool operation_optimisable(const details::operator_type& operation) const 31958 | { 31959 | return (details::e_add == operation) || 31960 | (details::e_sub == operation) || 31961 | (details::e_mul == operation) || 31962 | (details::e_div == operation) || 31963 | (details::e_mod == operation) || 31964 | (details::e_pow == operation) || 31965 | (details::e_lt == operation) || 31966 | (details::e_lte == operation) || 31967 | (details::e_gt == operation) || 31968 | (details::e_gte == operation) || 31969 | (details::e_eq == operation) || 31970 | (details::e_ne == operation) || 31971 | (details::e_and == operation) || 31972 | (details::e_nand == operation) || 31973 | (details::e_or == operation) || 31974 | (details::e_nor == operation) || 31975 | (details::e_xor == operation) || 31976 | (details::e_xnor == operation) ; 31977 | } 31978 | 31979 | inline std::string branch_to_id(expression_node_ptr branch) const 31980 | { 31981 | static const std::string null_str ("(null)" ); 31982 | static const std::string const_str ("(c)" ); 31983 | static const std::string var_str ("(v)" ); 31984 | static const std::string vov_str ("(vov)" ); 31985 | static const std::string cov_str ("(cov)" ); 31986 | static const std::string voc_str ("(voc)" ); 31987 | static const std::string str_str ("(s)" ); 31988 | static const std::string strrng_str ("(rngs)" ); 31989 | static const std::string cs_str ("(cs)" ); 31990 | static const std::string cstrrng_str("(crngs)"); 31991 | 31992 | if (details::is_null_node(branch)) 31993 | return null_str; 31994 | else if (details::is_constant_node(branch)) 31995 | return const_str; 31996 | else if (details::is_variable_node(branch)) 31997 | return var_str; 31998 | else if (details::is_vov_node(branch)) 31999 | return vov_str; 32000 | else if (details::is_cov_node(branch)) 32001 | return cov_str; 32002 | else if (details::is_voc_node(branch)) 32003 | return voc_str; 32004 | else if (details::is_string_node(branch)) 32005 | return str_str; 32006 | else if (details::is_const_string_node(branch)) 32007 | return cs_str; 32008 | else if (details::is_string_range_node(branch)) 32009 | return strrng_str; 32010 | else if (details::is_const_string_range_node(branch)) 32011 | return cstrrng_str; 32012 | else if (details::is_t0ot1ot2_node(branch)) 32013 | return "(" + dynamic_cast<details::T0oT1oT2_base_node<T>*>(branch)->type_id() + ")" 32014 | else if (details::is_t0ot1ot2ot3_node(branch)) 32015 | return "(" + dynamic_cast<details::T0oT1oT2oT3_base_node<T>*>(branch)->type_id() + ")" 32016 | else 32017 | return "ERROR" 32018 | } 32019 | 32020 | inline std::string branch_to_id(expression_node_ptr (&branch)[2]) const 32021 | { 32022 | return branch_to_id(branch[0]) + std::string("o") + branch_to_id(branch[1]); 32023 | } 32024 | 32025 | inline bool cov_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32026 | { 32027 | if (!operation_optimisable(operation)) 32028 | return false; 32029 | else 32030 | return details::is_constant_node(branch[0]) && 32031 | details::is_variable_node(branch[1]) ; 32032 | } 32033 | 32034 | inline bool voc_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32035 | { 32036 | if (!operation_optimisable(operation)) 32037 | return false; 32038 | else 32039 | return details::is_variable_node(branch[0]) && 32040 | details::is_constant_node(branch[1]) ; 32041 | } 32042 | 32043 | inline bool vov_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32044 | { 32045 | if (!operation_optimisable(operation)) 32046 | return false; 32047 | else 32048 | return details::is_variable_node(branch[0]) && 32049 | details::is_variable_node(branch[1]) ; 32050 | } 32051 | 32052 | inline bool cob_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32053 | { 32054 | if (!operation_optimisable(operation)) 32055 | return false; 32056 | else 32057 | return details::is_constant_node(branch[0]) && 32058 | !details::is_constant_node(branch[1]) ; 32059 | } 32060 | 32061 | inline bool boc_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32062 | { 32063 | if (!operation_optimisable(operation)) 32064 | return false; 32065 | else 32066 | return !details::is_constant_node(branch[0]) && 32067 | details::is_constant_node(branch[1]) ; 32068 | } 32069 | 32070 | inline bool cocob_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32071 | { 32072 | if ( 32073 | (details::e_add == operation) || 32074 | (details::e_sub == operation) || 32075 | (details::e_mul == operation) || 32076 | (details::e_div == operation) 32077 | ) 32078 | { 32079 | return (details::is_constant_node(branch[0]) && details::is_cob_node(branch[1])) || 32080 | (details::is_constant_node(branch[1]) && details::is_cob_node(branch[0])) ; 32081 | } 32082 | else 32083 | return false; 32084 | } 32085 | 32086 | inline bool coboc_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32087 | { 32088 | if ( 32089 | (details::e_add == operation) || 32090 | (details::e_sub == operation) || 32091 | (details::e_mul == operation) || 32092 | (details::e_div == operation) 32093 | ) 32094 | { 32095 | return (details::is_constant_node(branch[0]) && details::is_boc_node(branch[1])) || 32096 | (details::is_constant_node(branch[1]) && details::is_boc_node(branch[0])) ; 32097 | } 32098 | else 32099 | return false; 32100 | } 32101 | 32102 | inline bool uvouv_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32103 | { 32104 | if (!operation_optimisable(operation)) 32105 | return false; 32106 | else 32107 | return details::is_uv_node(branch[0]) && 32108 | details::is_uv_node(branch[1]) ; 32109 | } 32110 | 32111 | inline bool vob_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32112 | { 32113 | if (!operation_optimisable(operation)) 32114 | return false; 32115 | else 32116 | return details::is_variable_node(branch[0]) && 32117 | !details::is_variable_node(branch[1]) ; 32118 | } 32119 | 32120 | inline bool bov_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32121 | { 32122 | if (!operation_optimisable(operation)) 32123 | return false; 32124 | else 32125 | return !details::is_variable_node(branch[0]) && 32126 | details::is_variable_node(branch[1]) ; 32127 | } 32128 | 32129 | inline bool binext_optimisable(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32130 | { 32131 | if (!operation_optimisable(operation)) 32132 | return false; 32133 | else 32134 | return !details::is_constant_node(branch[0]) || 32135 | !details::is_constant_node(branch[1]) ; 32136 | } 32137 | 32138 | inline bool is_invalid_assignment_op(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32139 | { 32140 | if (is_assignment_operation(operation)) 32141 | { 32142 | const bool b1_is_genstring = details::is_generally_string_node(branch[1]); 32143 | 32144 | if (details::is_string_node(branch[0])) 32145 | return !b1_is_genstring; 32146 | else if (details::is_literal_node(branch[0])) 32147 | return true; 32148 | else 32149 | return ( 32150 | !details::is_variable_node (branch[0]) && 32151 | !details::is_vector_elem_node (branch[0]) && 32152 | !details::is_vector_celem_node (branch[0]) && 32153 | !details::is_vector_elem_rtc_node (branch[0]) && 32154 | !details::is_vector_celem_rtc_node (branch[0]) && 32155 | !details::is_rebasevector_elem_node (branch[0]) && 32156 | !details::is_rebasevector_celem_node (branch[0]) && 32157 | !details::is_rebasevector_elem_rtc_node (branch[0]) && 32158 | !details::is_rebasevector_celem_rtc_node(branch[0]) && 32159 | !details::is_vector_node (branch[0]) 32160 | ) 32161 | || b1_is_genstring; 32162 | } 32163 | else 32164 | return false; 32165 | } 32166 | 32167 | inline bool is_constpow_operation(const details::operator_type& operation, expression_node_ptr(&branch)[2]) const 32168 | { 32169 | if ( 32170 | !details::is_constant_node(branch[1]) || 32171 | details::is_constant_node(branch[0]) || 32172 | details::is_variable_node(branch[0]) || 32173 | details::is_vector_node (branch[0]) || 32174 | details::is_generally_string_node(branch[0]) 32175 | ) 32176 | return false; 32177 | 32178 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 32179 | 32180 | return cardinal_pow_optimisable(operation, c); 32181 | } 32182 | 32183 | inline bool is_invalid_break_continue_op(expression_node_ptr (&branch)[2]) const 32184 | { 32185 | return ( 32186 | details::is_break_node (branch[0]) || 32187 | details::is_break_node (branch[1]) || 32188 | details::is_continue_node(branch[0]) || 32189 | details::is_continue_node(branch[1]) 32190 | ); 32191 | } 32192 | 32193 | inline bool is_invalid_string_op(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32194 | { 32195 | const bool b0_string = is_generally_string_node(branch[0]); 32196 | const bool b1_string = is_generally_string_node(branch[1]); 32197 | 32198 | bool result = false; 32199 | 32200 | if (b0_string != b1_string) 32201 | result = true; 32202 | else if (!valid_string_operation(operation) && b0_string && b1_string) 32203 | result = true; 32204 | 32205 | if (result) 32206 | { 32207 | parser_->set_synthesis_error("Invalid string operation"); 32208 | } 32209 | 32210 | return result; 32211 | } 32212 | 32213 | inline bool is_invalid_string_op(const details::operator_type& operation, expression_node_ptr (&branch)[3]) const 32214 | { 32215 | const bool b0_string = is_generally_string_node(branch[0]); 32216 | const bool b1_string = is_generally_string_node(branch[1]); 32217 | const bool b2_string = is_generally_string_node(branch[2]); 32218 | 32219 | bool result = false; 32220 | 32221 | if ((b0_string != b1_string) || (b1_string != b2_string)) 32222 | result = true; 32223 | else if ((details::e_inrange != operation) && b0_string && b1_string && b2_string) 32224 | result = true; 32225 | 32226 | if (result) 32227 | { 32228 | parser_->set_synthesis_error("Invalid string operation"); 32229 | } 32230 | 32231 | return result; 32232 | } 32233 | 32234 | inline bool is_string_operation(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32235 | { 32236 | const bool b0_string = is_generally_string_node(branch[0]); 32237 | const bool b1_string = is_generally_string_node(branch[1]); 32238 | 32239 | return (b0_string && b1_string && valid_string_operation(operation)); 32240 | } 32241 | 32242 | inline bool is_string_operation(const details::operator_type& operation, expression_node_ptr (&branch)[3]) const 32243 | { 32244 | const bool b0_string = is_generally_string_node(branch[0]); 32245 | const bool b1_string = is_generally_string_node(branch[1]); 32246 | const bool b2_string = is_generally_string_node(branch[2]); 32247 | 32248 | return (b0_string && b1_string && b2_string && (details::e_inrange == operation)); 32249 | } 32250 | 32251 | #ifndef exprtk_disable_sc_andor 32252 | inline bool is_shortcircuit_expression(const details::operator_type& operation) const 32253 | { 32254 | return ( 32255 | (details::e_scand == operation) || 32256 | (details::e_scor == operation) 32257 | ); 32258 | } 32259 | #else 32260 | inline bool is_shortcircuit_expression(const details::operator_type&) const 32261 | { 32262 | return false; 32263 | } 32264 | #endif 32265 | 32266 | inline bool is_null_present(expression_node_ptr (&branch)[2]) const 32267 | { 32268 | return ( 32269 | details::is_null_node(branch[0]) || 32270 | details::is_null_node(branch[1]) 32271 | ); 32272 | } 32273 | 32274 | inline bool is_vector_eqineq_logic_operation(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32275 | { 32276 | if (!is_ivector_node(branch[0]) && !is_ivector_node(branch[1])) 32277 | return false; 32278 | else 32279 | return ( 32280 | (details::e_lt == operation) || 32281 | (details::e_lte == operation) || 32282 | (details::e_gt == operation) || 32283 | (details::e_gte == operation) || 32284 | (details::e_eq == operation) || 32285 | (details::e_ne == operation) || 32286 | (details::e_equal == operation) || 32287 | (details::e_and == operation) || 32288 | (details::e_nand == operation) || 32289 | (details::e_or == operation) || 32290 | (details::e_nor == operation) || 32291 | (details::e_xor == operation) || 32292 | (details::e_xnor == operation) 32293 | ); 32294 | } 32295 | 32296 | inline bool is_vector_arithmetic_operation(const details::operator_type& operation, expression_node_ptr (&branch)[2]) const 32297 | { 32298 | if (!is_ivector_node(branch[0]) && !is_ivector_node(branch[1])) 32299 | return false; 32300 | else 32301 | return ( 32302 | (details::e_add == operation) || 32303 | (details::e_sub == operation) || 32304 | (details::e_mul == operation) || 32305 | (details::e_div == operation) || 32306 | (details::e_pow == operation) 32307 | ); 32308 | } 32309 | 32310 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr (&branch)[2]) 32311 | { 32312 | if ((0 == branch[0]) || (0 == branch[1])) 32313 | { 32314 | parser_->set_error(parser_error::make_error( 32315 | parser_error::e_syntax, 32316 | parser_->current_state().token, 32317 | "ERR249 - Invalid branches received for operator '" + details::to_str(operation) + "'", 32318 | exprtk_error_location)); 32319 | 32320 | return error_node(); 32321 | } 32322 | else if (is_invalid_string_op(operation,branch)) 32323 | { 32324 | parser_->set_error(parser_error::make_error( 32325 | parser_error::e_syntax, 32326 | parser_->current_state().token, 32327 | "ERR250 - Invalid branch pair for string operator '" + details::to_str(operation) + "'", 32328 | exprtk_error_location)); 32329 | 32330 | return error_node(); 32331 | } 32332 | else if (is_invalid_assignment_op(operation,branch)) 32333 | { 32334 | parser_->set_error(parser_error::make_error( 32335 | parser_error::e_syntax, 32336 | parser_->current_state().token, 32337 | "ERR251 - Invalid branch pair for assignment operator '" + details::to_str(operation) + "'", 32338 | exprtk_error_location)); 32339 | 32340 | return error_node(); 32341 | } 32342 | else if (is_invalid_break_continue_op(branch)) 32343 | { 32344 | parser_->set_error(parser_error::make_error( 32345 | parser_error::e_syntax, 32346 | parser_->current_state().token, 32347 | "ERR252 - Invalid branch pair for break/continue operator '" + details::to_str(operation) + "'", 32348 | exprtk_error_location)); 32349 | 32350 | return error_node(); 32351 | } 32352 | else if (details::e_assign == operation) 32353 | { 32354 | return synthesize_assignment_expression(operation, branch); 32355 | } 32356 | else if (details::e_swap == operation) 32357 | { 32358 | return synthesize_swap_expression(branch); 32359 | } 32360 | else if (is_assignment_operation(operation)) 32361 | { 32362 | return synthesize_assignment_operation_expression(operation, branch); 32363 | } 32364 | else if (is_vector_eqineq_logic_operation(operation, branch)) 32365 | { 32366 | return synthesize_veceqineqlogic_operation_expression(operation, branch); 32367 | } 32368 | else if (is_vector_arithmetic_operation(operation, branch)) 32369 | { 32370 | return synthesize_vecarithmetic_operation_expression(operation, branch); 32371 | } 32372 | else if (is_shortcircuit_expression(operation)) 32373 | { 32374 | return synthesize_shortcircuit_expression(operation, branch); 32375 | } 32376 | else if (is_string_operation(operation, branch)) 32377 | { 32378 | return synthesize_string_expression(operation, branch); 32379 | } 32380 | else if (is_null_present(branch)) 32381 | { 32382 | return synthesize_null_expression(operation, branch); 32383 | } 32384 | #ifndef exprtk_disable_cardinal_pow_optimisation 32385 | else if (is_constpow_operation(operation, branch)) 32386 | { 32387 | return cardinal_pow_optimisation(branch); 32388 | } 32389 | #endif 32390 | 32391 | expression_node_ptr result = error_node(); 32392 | 32393 | #ifndef exprtk_disable_enhanced_features 32394 | if (synthesize_expression(operation, branch, result)) 32395 | { 32396 | return result; 32397 | } 32398 | else 32399 | #endif 32400 | 32401 | { 32402 | /* 32403 | Possible reductions: 32404 | 1. c o cob -> cob 32405 | 2. cob o c -> cob 32406 | 3. c o boc -> boc 32407 | 4. boc o c -> boc 32408 | */ 32409 | result = error_node(); 32410 | 32411 | if (cocob_optimisable(operation, branch)) 32412 | { 32413 | result = synthesize_cocob_expression::process((*this), operation, branch); 32414 | } 32415 | else if (coboc_optimisable(operation, branch) && (0 == result)) 32416 | { 32417 | result = synthesize_coboc_expression::process((*this), operation, branch); 32418 | } 32419 | 32420 | if (result) 32421 | return result; 32422 | } 32423 | 32424 | if (uvouv_optimisable(operation, branch)) 32425 | { 32426 | return synthesize_uvouv_expression(operation, branch); 32427 | } 32428 | else if (vob_optimisable(operation, branch)) 32429 | { 32430 | return synthesize_vob_expression::process((*this), operation, branch); 32431 | } 32432 | else if (bov_optimisable(operation, branch)) 32433 | { 32434 | return synthesize_bov_expression::process((*this), operation, branch); 32435 | } 32436 | else if (cob_optimisable(operation, branch)) 32437 | { 32438 | return synthesize_cob_expression::process((*this), operation, branch); 32439 | } 32440 | else if (boc_optimisable(operation, branch)) 32441 | { 32442 | return synthesize_boc_expression::process((*this), operation, branch); 32443 | } 32444 | #ifndef exprtk_disable_enhanced_features 32445 | else if (cov_optimisable(operation, branch)) 32446 | { 32447 | return synthesize_cov_expression::process((*this), operation, branch); 32448 | } 32449 | #endif 32450 | else if (binext_optimisable(operation, branch)) 32451 | { 32452 | return synthesize_binary_ext_expression::process((*this), operation, branch); 32453 | } 32454 | else 32455 | return synthesize_expression<binary_node_t,2>(operation, branch); 32456 | } 32457 | 32458 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr (&branch)[3]) 32459 | { 32460 | if ( 32461 | (0 == branch[0]) || 32462 | (0 == branch[1]) || 32463 | (0 == branch[2]) 32464 | ) 32465 | { 32466 | details::free_all_nodes(*node_allocator_,branch); 32467 | 32468 | parser_->set_error(parser_error::make_error( 32469 | parser_error::e_syntax, 32470 | parser_->current_state().token, 32471 | "ERR253 - Invalid branches operator '" + details::to_str(operation) + "'", 32472 | exprtk_error_location)); 32473 | 32474 | return error_node(); 32475 | } 32476 | else if (is_invalid_string_op(operation, branch)) 32477 | { 32478 | parser_->set_error(parser_error::make_error( 32479 | parser_error::e_syntax, 32480 | parser_->current_state().token, 32481 | "ERR254 - Invalid branches for string operator '" + details::to_str(operation) + "'", 32482 | exprtk_error_location)); 32483 | 32484 | return error_node(); 32485 | } 32486 | else if (is_string_operation(operation, branch)) 32487 | { 32488 | return synthesize_string_expression(operation, branch); 32489 | } 32490 | else 32491 | return synthesize_expression<trinary_node_t,3>(operation, branch); 32492 | } 32493 | 32494 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr (&branch)[4]) 32495 | { 32496 | return synthesize_expression<quaternary_node_t,4>(operation,branch); 32497 | } 32498 | 32499 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr b0) 32500 | { 32501 | expression_node_ptr branch[1] = { b0 }; 32502 | return (*this)(operation,branch); 32503 | } 32504 | 32505 | inline expression_node_ptr operator() (const details::operator_type& operation, expression_node_ptr& b0, expression_node_ptr& b1) 32506 | { 32507 | expression_node_ptr result = error_node(); 32508 | 32509 | if ((0 != b0) && (0 != b1)) 32510 | { 32511 | expression_node_ptr branch[2] = { b0, b1 }; 32512 | result = expression_generator<Type>::operator()(operation, branch); 32513 | b0 = branch[0]; 32514 | b1 = branch[1]; 32515 | } 32516 | 32517 | return result; 32518 | } 32519 | 32520 | inline expression_node_ptr conditional(expression_node_ptr condition, 32521 | expression_node_ptr consequent, 32522 | expression_node_ptr alternative) const 32523 | { 32524 | if ((0 == condition) || (0 == consequent)) 32525 | { 32526 | details::free_node(*node_allocator_, condition ); 32527 | details::free_node(*node_allocator_, consequent ); 32528 | details::free_node(*node_allocator_, alternative); 32529 | 32530 | const std::string invalid_branches = 32531 | ((0 == condition ) ? std::string("condition ") : "") + 32532 | ((0 == consequent) ? std::string("consequent") : "") ; 32533 | 32534 | parser_->set_error(parser_error::make_error( 32535 | parser_error::e_parser, 32536 | parser_->current_state().token, 32537 | "ERR255 - Invalid " + invalid_branches + " for conditional statement", 32538 | exprtk_error_location)); 32539 | 32540 | return error_node(); 32541 | } 32542 | // Can the condition be immediately evaluated? if so optimise. 32543 | else if (details::is_constant_node(condition)) 32544 | { 32545 | // True branch 32546 | if (details::is_true(condition)) 32547 | { 32548 | details::free_node(*node_allocator_, condition ); 32549 | details::free_node(*node_allocator_, alternative); 32550 | 32551 | return consequent; 32552 | } 32553 | // False branch 32554 | else 32555 | { 32556 | details::free_node(*node_allocator_, condition ); 32557 | details::free_node(*node_allocator_, consequent); 32558 | 32559 | if (alternative) 32560 | return alternative; 32561 | else 32562 | return node_allocator_->allocate<details::null_node<T> >(); 32563 | } 32564 | } 32565 | 32566 | expression_node_ptr result = error_node(); 32567 | std::string node_name = "Unknown!" 32568 | 32569 | if ((0 != consequent) && (0 != alternative)) 32570 | { 32571 | result = node_allocator_->allocate<conditional_node_t>(condition, consequent, alternative); 32572 | node_name = "conditional_node_t" 32573 | } 32574 | else 32575 | { 32576 | result = node_allocator_->allocate<cons_conditional_node_t>(condition, consequent); 32577 | node_name = "cons_conditional_node_t" 32578 | } 32579 | 32580 | if (result && result->valid()) 32581 | { 32582 | return result; 32583 | } 32584 | 32585 | parser_->set_error(parser_error::make_error( 32586 | parser_error::e_parser, 32587 | token_t(), 32588 | "ERR256 - Failed to synthesize node: " + node_name, 32589 | exprtk_error_location)); 32590 | 32591 | details::free_node(*node_allocator_, result); 32592 | return error_node(); 32593 | } 32594 | 32595 | #ifndef exprtk_disable_string_capabilities 32596 | inline expression_node_ptr conditional_string(expression_node_ptr condition, 32597 | expression_node_ptr consequent, 32598 | expression_node_ptr alternative) const 32599 | { 32600 | if ((0 == condition) || (0 == consequent)) 32601 | { 32602 | details::free_node(*node_allocator_, condition ); 32603 | details::free_node(*node_allocator_, consequent ); 32604 | details::free_node(*node_allocator_, alternative); 32605 | 32606 | const std::string invalid_branches = 32607 | ((0 == condition ) ? std::string("condition ") : "") + 32608 | ((0 == consequent) ? std::string("consequent") : "") ; 32609 | 32610 | parser_->set_error(parser_error::make_error( 32611 | parser_error::e_parser, 32612 | parser_->current_state().token, 32613 | "ERR257 - Invalid " + invalid_branches + " for string conditional statement", 32614 | exprtk_error_location)); 32615 | 32616 | return error_node(); 32617 | } 32618 | // Can the condition be immediately evaluated? if so optimise. 32619 | else if (details::is_constant_node(condition)) 32620 | { 32621 | // True branch 32622 | if (details::is_true(condition)) 32623 | { 32624 | details::free_node(*node_allocator_, condition ); 32625 | details::free_node(*node_allocator_, alternative); 32626 | 32627 | return consequent; 32628 | } 32629 | // False branch 32630 | else 32631 | { 32632 | details::free_node(*node_allocator_, condition ); 32633 | details::free_node(*node_allocator_, consequent); 32634 | 32635 | if (alternative) 32636 | return alternative; 32637 | else 32638 | return node_allocator_-> 32639 | allocate_c<details::string_literal_node<Type> >(""); 32640 | } 32641 | } 32642 | else if ((0 != consequent) && (0 != alternative)) 32643 | { 32644 | expression_node_ptr result = 32645 | node_allocator_->allocate<conditional_string_node_t>(condition, consequent, alternative); 32646 | 32647 | if (result && result->valid()) 32648 | { 32649 | return result; 32650 | } 32651 | 32652 | parser_->set_error(parser_error::make_error( 32653 | parser_error::e_parser, 32654 | token_t(), 32655 | "ERR258 - Failed to synthesize node: conditional_string_node_t", 32656 | exprtk_error_location)); 32657 | 32658 | details::free_node(*node_allocator_, result); 32659 | } 32660 | 32661 | return error_node(); 32662 | } 32663 | #else 32664 | inline expression_node_ptr conditional_string(expression_node_ptr, 32665 | expression_node_ptr, 32666 | expression_node_ptr) const 32667 | { 32668 | return error_node(); 32669 | } 32670 | #endif 32671 | 32672 | inline expression_node_ptr conditional_vector(expression_node_ptr condition, 32673 | expression_node_ptr consequent, 32674 | expression_node_ptr alternative) const 32675 | { 32676 | if ((0 == condition) || (0 == consequent)) 32677 | { 32678 | details::free_node(*node_allocator_, condition ); 32679 | details::free_node(*node_allocator_, consequent ); 32680 | details::free_node(*node_allocator_, alternative); 32681 | 32682 | const std::string invalid_branches = 32683 | ((0 == condition ) ? std::string("condition ") : "") + 32684 | ((0 == consequent) ? std::string("consequent") : "") ; 32685 | 32686 | parser_->set_error(parser_error::make_error( 32687 | parser_error::e_parser, 32688 | parser_->current_state().token, 32689 | "ERR259 - Invalid " + invalid_branches + " for vector conditional statement", 32690 | exprtk_error_location)); 32691 | 32692 | return error_node(); 32693 | } 32694 | // Can the condition be immediately evaluated? if so optimise. 32695 | else if (details::is_constant_node(condition)) 32696 | { 32697 | // True branch 32698 | if (details::is_true(condition)) 32699 | { 32700 | details::free_node(*node_allocator_, condition ); 32701 | details::free_node(*node_allocator_, alternative); 32702 | 32703 | return consequent; 32704 | } 32705 | // False branch 32706 | else 32707 | { 32708 | details::free_node(*node_allocator_, condition ); 32709 | details::free_node(*node_allocator_, consequent); 32710 | 32711 | if (alternative) 32712 | return alternative; 32713 | else 32714 | return node_allocator_->allocate<details::null_node<T> >(); 32715 | 32716 | } 32717 | } 32718 | else if ((0 != consequent) && (0 != alternative)) 32719 | { 32720 | return node_allocator_-> 32721 | allocate<conditional_vector_node_t>(condition, consequent, alternative); 32722 | } 32723 | else 32724 | return error_node(); 32725 | } 32726 | 32727 | inline loop_runtime_check_ptr get_loop_runtime_check(const loop_runtime_check::loop_types loop_type) const 32728 | { 32729 | if ( 32730 | parser_->loop_runtime_check_ && 32731 | (loop_type == (parser_->loop_runtime_check_->loop_set & loop_type)) 32732 | ) 32733 | { 32734 | return parser_->loop_runtime_check_; 32735 | } 32736 | 32737 | return loop_runtime_check_ptr(0); 32738 | } 32739 | 32740 | inline vector_access_runtime_check_ptr get_vector_access_runtime_check() const 32741 | { 32742 | return parser_->vector_access_runtime_check_; 32743 | } 32744 | 32745 | inline expression_node_ptr while_loop(expression_node_ptr& condition, 32746 | expression_node_ptr& branch, 32747 | const bool break_continue_present = false) const 32748 | { 32749 | if ( 32750 | !break_continue_present && 32751 | !parser_->state_.return_stmt_present && 32752 | details::is_constant_node(condition) 32753 | ) 32754 | { 32755 | expression_node_ptr result = error_node(); 32756 | if (details::is_true(condition)) 32757 | { 32758 | // Infinite loops are not allowed. 32759 | 32760 | parser_->set_error(parser_error::make_error( 32761 | parser_error::e_parser, 32762 | parser_->current_state().token, 32763 | "ERR260 - Infinite loop condition without 'break' or 'return' not allowed in while-loops", 32764 | exprtk_error_location)); 32765 | 32766 | result = error_node(); 32767 | } 32768 | else 32769 | result = node_allocator_->allocate<details::null_node<Type> >(); 32770 | 32771 | details::free_node(*node_allocator_, condition); 32772 | details::free_node(*node_allocator_, branch ); 32773 | 32774 | return result; 32775 | } 32776 | else if (details::is_null_node(condition)) 32777 | { 32778 | details::free_node(*node_allocator_,condition); 32779 | 32780 | return branch; 32781 | } 32782 | 32783 | loop_runtime_check_ptr rtc = get_loop_runtime_check(loop_runtime_check::e_while_loop); 32784 | 32785 | if (!break_continue_present) 32786 | { 32787 | if (rtc) 32788 | return node_allocator_->allocate<while_loop_rtc_node_t> 32789 | (condition, branch, rtc); 32790 | else 32791 | return node_allocator_->allocate<while_loop_node_t> 32792 | (condition, branch); 32793 | } 32794 | #ifndef exprtk_disable_break_continue 32795 | else 32796 | { 32797 | if (rtc) 32798 | return node_allocator_->allocate<while_loop_bc_rtc_node_t> 32799 | (condition, branch, rtc); 32800 | else 32801 | return node_allocator_->allocate<while_loop_bc_node_t> 32802 | (condition, branch); 32803 | } 32804 | #else 32805 | return error_node(); 32806 | #endif 32807 | } 32808 | 32809 | inline expression_node_ptr repeat_until_loop(expression_node_ptr& condition, 32810 | expression_node_ptr& branch, 32811 | const bool break_continue_present = false) const 32812 | { 32813 | if (!break_continue_present && details::is_constant_node(condition)) 32814 | { 32815 | if ( 32816 | details::is_true(condition) && 32817 | details::is_constant_node(branch) 32818 | ) 32819 | { 32820 | free_node(*node_allocator_,condition); 32821 | 32822 | return branch; 32823 | } 32824 | 32825 | details::free_node(*node_allocator_, condition); 32826 | details::free_node(*node_allocator_, branch ); 32827 | 32828 | return error_node(); 32829 | } 32830 | else if (details::is_null_node(condition)) 32831 | { 32832 | details::free_node(*node_allocator_,condition); 32833 | 32834 | return branch; 32835 | } 32836 | 32837 | loop_runtime_check_ptr rtc = get_loop_runtime_check(loop_runtime_check::e_repeat_until_loop); 32838 | 32839 | if (!break_continue_present) 32840 | { 32841 | if (rtc) 32842 | return node_allocator_->allocate<repeat_until_loop_rtc_node_t> 32843 | (condition, branch, rtc); 32844 | else 32845 | return node_allocator_->allocate<repeat_until_loop_node_t> 32846 | (condition, branch); 32847 | } 32848 | #ifndef exprtk_disable_break_continue 32849 | else 32850 | { 32851 | if (rtc) 32852 | return node_allocator_->allocate<repeat_until_loop_bc_rtc_node_t> 32853 | (condition, branch, rtc); 32854 | else 32855 | return node_allocator_->allocate<repeat_until_loop_bc_node_t> 32856 | (condition, branch); 32857 | } 32858 | #else 32859 | return error_node(); 32860 | #endif 32861 | } 32862 | 32863 | inline expression_node_ptr for_loop(expression_node_ptr& initialiser, 32864 | expression_node_ptr& condition, 32865 | expression_node_ptr& incrementor, 32866 | expression_node_ptr& loop_body, 32867 | bool break_continue_present = false) const 32868 | { 32869 | if ( 32870 | !break_continue_present && 32871 | !parser_->state_.return_stmt_present && 32872 | details::is_constant_node(condition) 32873 | ) 32874 | { 32875 | expression_node_ptr result = error_node(); 32876 | 32877 | if (details::is_true(condition)) 32878 | { 32879 | // Infinite loops are not allowed. 32880 | 32881 | parser_->set_error(parser_error::make_error( 32882 | parser_error::e_parser, 32883 | parser_->current_state().token, 32884 | "ERR261 - Infinite loop condition without 'break' or 'return' not allowed in for-loop", 32885 | exprtk_error_location)); 32886 | 32887 | result = error_node(); 32888 | } 32889 | else 32890 | result = node_allocator_->allocate<details::null_node<Type> >(); 32891 | 32892 | details::free_node(*node_allocator_, initialiser); 32893 | details::free_node(*node_allocator_, condition ); 32894 | details::free_node(*node_allocator_, incrementor); 32895 | details::free_node(*node_allocator_, loop_body ); 32896 | 32897 | return result; 32898 | } 32899 | else if (details::is_null_node(condition) || (0 == condition)) 32900 | { 32901 | details::free_node(*node_allocator_, initialiser); 32902 | details::free_node(*node_allocator_, condition ); 32903 | details::free_node(*node_allocator_, incrementor); 32904 | 32905 | return loop_body; 32906 | } 32907 | 32908 | loop_runtime_check_ptr rtc = get_loop_runtime_check(loop_runtime_check::e_for_loop); 32909 | 32910 | if (!break_continue_present) 32911 | { 32912 | if (rtc) 32913 | return node_allocator_->allocate<for_loop_rtc_node_t> 32914 | ( 32915 | initialiser, 32916 | condition, 32917 | incrementor, 32918 | loop_body, 32919 | rtc 32920 | ); 32921 | else 32922 | return node_allocator_->allocate<for_loop_node_t> 32923 | ( 32924 | initialiser, 32925 | condition, 32926 | incrementor, 32927 | loop_body 32928 | ); 32929 | } 32930 | #ifndef exprtk_disable_break_continue 32931 | else 32932 | { 32933 | if (rtc) 32934 | return node_allocator_->allocate<for_loop_bc_rtc_node_t> 32935 | ( 32936 | initialiser, 32937 | condition, 32938 | incrementor, 32939 | loop_body, 32940 | rtc 32941 | ); 32942 | else 32943 | return node_allocator_->allocate<for_loop_bc_node_t> 32944 | ( 32945 | initialiser, 32946 | condition, 32947 | incrementor, 32948 | loop_body 32949 | ); 32950 | } 32951 | #else 32952 | return error_node(); 32953 | #endif 32954 | } 32955 | 32956 | template <typename Allocator, 32957 | template <typename, typename> class Sequence> 32958 | inline expression_node_ptr const_optimise_switch(Sequence<expression_node_ptr,Allocator>& arg_list) 32959 | { 32960 | expression_node_ptr result = error_node(); 32961 | 32962 | for (std::size_t i = 0; i < (arg_list.size() / 2); ++i) 32963 | { 32964 | expression_node_ptr condition = arg_list[(2 * i) ]; 32965 | expression_node_ptr consequent = arg_list[(2 * i) + 1]; 32966 | 32967 | if ((0 == result) && details::is_true(condition)) 32968 | { 32969 | result = consequent; 32970 | break; 32971 | } 32972 | } 32973 | 32974 | if (0 == result) 32975 | { 32976 | result = arg_list.back(); 32977 | } 32978 | 32979 | for (std::size_t i = 0; i < arg_list.size(); ++i) 32980 | { 32981 | expression_node_ptr current_expr = arg_list[i]; 32982 | 32983 | if (current_expr && (current_expr != result)) 32984 | { 32985 | free_node(*node_allocator_,current_expr); 32986 | } 32987 | } 32988 | 32989 | return result; 32990 | } 32991 | 32992 | template <typename Allocator, 32993 | template <typename, typename> class Sequence> 32994 | inline expression_node_ptr const_optimise_mswitch(Sequence<expression_node_ptr,Allocator>& arg_list) 32995 | { 32996 | expression_node_ptr result = error_node(); 32997 | 32998 | for (std::size_t i = 0; i < (arg_list.size() / 2); ++i) 32999 | { 33000 | expression_node_ptr condition = arg_list[(2 * i) ]; 33001 | expression_node_ptr consequent = arg_list[(2 * i) + 1]; 33002 | 33003 | if (details::is_true(condition)) 33004 | { 33005 | result = consequent; 33006 | } 33007 | } 33008 | 33009 | if (0 == result) 33010 | { 33011 | const T zero = T(0); 33012 | result = node_allocator_->allocate<literal_node_t>(zero); 33013 | } 33014 | 33015 | for (std::size_t i = 0; i < arg_list.size(); ++i) 33016 | { 33017 | expression_node_ptr& current_expr = arg_list[i]; 33018 | 33019 | if (current_expr && (current_expr != result)) 33020 | { 33021 | details::free_node(*node_allocator_,current_expr); 33022 | } 33023 | } 33024 | 33025 | return result; 33026 | } 33027 | 33028 | struct switch_nodes 33029 | { 33030 | typedef std::vector<std::pair<expression_node_ptr,bool> > arg_list_t; 33031 | 33032 | #define case_stmt(N) \ 33033 | if (is_true(arg[(2 * N)].first)) { return arg[(2 * N) + 1].first->value(); } \ 33034 | 33035 | struct switch_impl_1 33036 | { 33037 | static inline T process(const arg_list_t& arg) 33038 | { 33039 | case_stmt(0) 33040 | 33041 | assert(arg.size() == ((2 * 1) + 1)); 33042 | 33043 | return arg.back().first->value(); 33044 | } 33045 | }; 33046 | 33047 | struct switch_impl_2 33048 | { 33049 | static inline T process(const arg_list_t& arg) 33050 | { 33051 | case_stmt(0) case_stmt(1) 33052 | 33053 | assert(arg.size() == ((2 * 2) + 1)); 33054 | 33055 | return arg.back().first->value(); 33056 | } 33057 | }; 33058 | 33059 | struct switch_impl_3 33060 | { 33061 | static inline T process(const arg_list_t& arg) 33062 | { 33063 | case_stmt(0) case_stmt(1) 33064 | case_stmt(2) 33065 | 33066 | assert(arg.size() == ((2 * 3) + 1)); 33067 | 33068 | return arg.back().first->value(); 33069 | } 33070 | }; 33071 | 33072 | struct switch_impl_4 33073 | { 33074 | static inline T process(const arg_list_t& arg) 33075 | { 33076 | case_stmt(0) case_stmt(1) 33077 | case_stmt(2) case_stmt(3) 33078 | 33079 | assert(arg.size() == ((2 * 4) + 1)); 33080 | 33081 | return arg.back().first->value(); 33082 | } 33083 | }; 33084 | 33085 | struct switch_impl_5 33086 | { 33087 | static inline T process(const arg_list_t& arg) 33088 | { 33089 | case_stmt(0) case_stmt(1) 33090 | case_stmt(2) case_stmt(3) 33091 | case_stmt(4) 33092 | 33093 | assert(arg.size() == ((2 * 5) + 1)); 33094 | 33095 | return arg.back().first->value(); 33096 | } 33097 | }; 33098 | 33099 | struct switch_impl_6 33100 | { 33101 | static inline T process(const arg_list_t& arg) 33102 | { 33103 | case_stmt(0) case_stmt(1) 33104 | case_stmt(2) case_stmt(3) 33105 | case_stmt(4) case_stmt(5) 33106 | 33107 | assert(arg.size() == ((2 * 6) + 1)); 33108 | 33109 | return arg.back().first->value(); 33110 | } 33111 | }; 33112 | 33113 | struct switch_impl_7 33114 | { 33115 | static inline T process(const arg_list_t& arg) 33116 | { 33117 | case_stmt(0) case_stmt(1) 33118 | case_stmt(2) case_stmt(3) 33119 | case_stmt(4) case_stmt(5) 33120 | case_stmt(6) 33121 | 33122 | assert(arg.size() == ((2 * 7) + 1)); 33123 | 33124 | return arg.back().first->value(); 33125 | } 33126 | }; 33127 | 33128 | #undef case_stmt 33129 | }; 33130 | 33131 | template <typename Allocator, 33132 | template <typename, typename> class Sequence> 33133 | inline expression_node_ptr switch_statement(Sequence<expression_node_ptr,Allocator>& arg_list, const bool default_statement_present) 33134 | { 33135 | if (arg_list.empty()) 33136 | return error_node(); 33137 | else if ( 33138 | !all_nodes_valid(arg_list) || 33139 | (!default_statement_present && (arg_list.size() < 2)) 33140 | ) 33141 | { 33142 | details::free_all_nodes(*node_allocator_,arg_list); 33143 | 33144 | return error_node(); 33145 | } 33146 | else if (is_constant_foldable(arg_list)) 33147 | return const_optimise_switch(arg_list); 33148 | 33149 | switch ((arg_list.size() - 1) / 2) 33150 | { 33151 | #define case_stmt(N) \ 33152 | case N : \ 33153 | return node_allocator_-> \ 33154 | allocate<details::switch_n_node \ 33155 | <Type,typename switch_nodes::switch_impl_##N > >(arg_list); \ 33156 | 33157 | case_stmt(1) 33158 | case_stmt(2) 33159 | case_stmt(3) 33160 | case_stmt(4) 33161 | case_stmt(5) 33162 | case_stmt(6) 33163 | case_stmt(7) 33164 | #undef case_stmt 33165 | 33166 | default : return node_allocator_->allocate<details::switch_node<Type> >(arg_list); 33167 | } 33168 | } 33169 | 33170 | template <typename Allocator, 33171 | template <typename, typename> class Sequence> 33172 | inline expression_node_ptr multi_switch_statement(Sequence<expression_node_ptr,Allocator>& arg_list) 33173 | { 33174 | if (!all_nodes_valid(arg_list)) 33175 | { 33176 | details::free_all_nodes(*node_allocator_,arg_list); 33177 | 33178 | return error_node(); 33179 | } 33180 | else if (is_constant_foldable(arg_list)) 33181 | return const_optimise_mswitch(arg_list); 33182 | else 33183 | return node_allocator_->allocate<details::multi_switch_node<Type> >(arg_list); 33184 | } 33185 | 33186 | inline expression_node_ptr assert_call(expression_node_ptr& assert_condition, 33187 | expression_node_ptr& assert_message, 33188 | const assert_check::assert_context& context) 33189 | { 33190 | typedef details::assert_node<Type> alloc_type; 33191 | 33192 | expression_node_ptr result = node_allocator_->allocate_rrrr<alloc_type> 33193 | (assert_condition, assert_message, parser_->assert_check_, context); 33194 | 33195 | if (result && result->valid()) 33196 | { 33197 | parser_->state_.activate_side_effect("assert_call()"); 33198 | return result; 33199 | } 33200 | 33201 | details::free_node(*node_allocator_, result ); 33202 | details::free_node(*node_allocator_, assert_condition); 33203 | details::free_node(*node_allocator_, assert_message ); 33204 | 33205 | return error_node(); 33206 | } 33207 | 33208 | #define unary_opr_switch_statements \ 33209 | case_stmt(details::e_abs , details::abs_op ) \ 33210 | case_stmt(details::e_acos , details::acos_op ) \ 33211 | case_stmt(details::e_acosh , details::acosh_op) \ 33212 | case_stmt(details::e_asin , details::asin_op ) \ 33213 | case_stmt(details::e_asinh , details::asinh_op) \ 33214 | case_stmt(details::e_atan , details::atan_op ) \ 33215 | case_stmt(details::e_atanh , details::atanh_op) \ 33216 | case_stmt(details::e_ceil , details::ceil_op ) \ 33217 | case_stmt(details::e_cos , details::cos_op ) \ 33218 | case_stmt(details::e_cosh , details::cosh_op ) \ 33219 | case_stmt(details::e_exp , details::exp_op ) \ 33220 | case_stmt(details::e_expm1 , details::expm1_op) \ 33221 | case_stmt(details::e_floor , details::floor_op) \ 33222 | case_stmt(details::e_log , details::log_op ) \ 33223 | case_stmt(details::e_log10 , details::log10_op) \ 33224 | case_stmt(details::e_log2 , details::log2_op ) \ 33225 | case_stmt(details::e_log1p , details::log1p_op) \ 33226 | case_stmt(details::e_neg , details::neg_op ) \ 33227 | case_stmt(details::e_pos , details::pos_op ) \ 33228 | case_stmt(details::e_round , details::round_op) \ 33229 | case_stmt(details::e_sin , details::sin_op ) \ 33230 | case_stmt(details::e_sinc , details::sinc_op ) \ 33231 | case_stmt(details::e_sinh , details::sinh_op ) \ 33232 | case_stmt(details::e_sqrt , details::sqrt_op ) \ 33233 | case_stmt(details::e_tan , details::tan_op ) \ 33234 | case_stmt(details::e_tanh , details::tanh_op ) \ 33235 | case_stmt(details::e_cot , details::cot_op ) \ 33236 | case_stmt(details::e_sec , details::sec_op ) \ 33237 | case_stmt(details::e_csc , details::csc_op ) \ 33238 | case_stmt(details::e_r2d , details::r2d_op ) \ 33239 | case_stmt(details::e_d2r , details::d2r_op ) \ 33240 | case_stmt(details::e_d2g , details::d2g_op ) \ 33241 | case_stmt(details::e_g2d , details::g2d_op ) \ 33242 | case_stmt(details::e_notl , details::notl_op ) \ 33243 | case_stmt(details::e_sgn , details::sgn_op ) \ 33244 | case_stmt(details::e_erf , details::erf_op ) \ 33245 | case_stmt(details::e_erfc , details::erfc_op ) \ 33246 | case_stmt(details::e_ncdf , details::ncdf_op ) \ 33247 | case_stmt(details::e_frac , details::frac_op ) \ 33248 | case_stmt(details::e_trunc , details::trunc_op) \ 33249 | 33250 | inline expression_node_ptr synthesize_uv_expression(const details::operator_type& operation, 33251 | expression_node_ptr (&branch)[1]) 33252 | { 33253 | T& v = static_cast<details::variable_node<T>*>(branch[0])->ref(); 33254 | 33255 | switch (operation) 33256 | { 33257 | #define case_stmt(op0, op1) \ 33258 | case op0 : return node_allocator_-> \ 33259 | allocate<typename details::unary_variable_node<Type,op1<Type> > >(v); \ 33260 | 33261 | unary_opr_switch_statements 33262 | #undef case_stmt 33263 | default : return error_node(); 33264 | } 33265 | } 33266 | 33267 | inline expression_node_ptr synthesize_uvec_expression(const details::operator_type& operation, 33268 | expression_node_ptr (&branch)[1]) 33269 | { 33270 | switch (operation) 33271 | { 33272 | #define case_stmt(op0, op1) \ 33273 | case op0 : return node_allocator_-> \ 33274 | allocate<typename details::unary_vector_node<Type,op1<Type> > > \ 33275 | (operation, branch[0]); \ 33276 | 33277 | unary_opr_switch_statements 33278 | #undef case_stmt 33279 | default : return error_node(); 33280 | } 33281 | } 33282 | 33283 | inline expression_node_ptr synthesize_unary_expression(const details::operator_type& operation, 33284 | expression_node_ptr (&branch)[1]) 33285 | { 33286 | switch (operation) 33287 | { 33288 | #define case_stmt(op0, op1) \ 33289 | case op0 : return node_allocator_-> \ 33290 | allocate<typename details::unary_branch_node<Type,op1<Type> > >(branch[0]); \ 33291 | 33292 | unary_opr_switch_statements 33293 | #undef case_stmt 33294 | default : return error_node(); 33295 | } 33296 | } 33297 | 33298 | inline expression_node_ptr const_optimise_sf3(const details::operator_type& operation, 33299 | expression_node_ptr (&branch)[3]) 33300 | { 33301 | expression_node_ptr temp_node = error_node(); 33302 | 33303 | switch (operation) 33304 | { 33305 | #define case_stmt(op) \ 33306 | case details::e_sf##op : temp_node = node_allocator_-> \ 33307 | allocate<details::sf3_node<Type,details::sf##op##_op<Type> > > \ 33308 | (operation, branch); \ 33309 | break; \ 33310 | 33311 | case_stmt(00) case_stmt(01) case_stmt(02) case_stmt(03) 33312 | case_stmt(04) case_stmt(05) case_stmt(06) case_stmt(07) 33313 | case_stmt(08) case_stmt(09) case_stmt(10) case_stmt(11) 33314 | case_stmt(12) case_stmt(13) case_stmt(14) case_stmt(15) 33315 | case_stmt(16) case_stmt(17) case_stmt(18) case_stmt(19) 33316 | case_stmt(20) case_stmt(21) case_stmt(22) case_stmt(23) 33317 | case_stmt(24) case_stmt(25) case_stmt(26) case_stmt(27) 33318 | case_stmt(28) case_stmt(29) case_stmt(30) case_stmt(31) 33319 | case_stmt(32) case_stmt(33) case_stmt(34) case_stmt(35) 33320 | case_stmt(36) case_stmt(37) case_stmt(38) case_stmt(39) 33321 | case_stmt(40) case_stmt(41) case_stmt(42) case_stmt(43) 33322 | case_stmt(44) case_stmt(45) case_stmt(46) case_stmt(47) 33323 | #undef case_stmt 33324 | default : return error_node(); 33325 | } 33326 | 33327 | assert(temp_node); 33328 | 33329 | const T v = temp_node->value(); 33330 | 33331 | details::free_node(*node_allocator_,temp_node); 33332 | 33333 | return node_allocator_->allocate<literal_node_t>(v); 33334 | } 33335 | 33336 | inline expression_node_ptr varnode_optimise_sf3(const details::operator_type& operation, expression_node_ptr (&branch)[3]) 33337 | { 33338 | typedef details::variable_node<Type>* variable_ptr; 33339 | 33340 | const Type& v0 = static_cast<variable_ptr>(branch[0])->ref(); 33341 | const Type& v1 = static_cast<variable_ptr>(branch[1])->ref(); 33342 | const Type& v2 = static_cast<variable_ptr>(branch[2])->ref(); 33343 | 33344 | switch (operation) 33345 | { 33346 | #define case_stmt(op) \ 33347 | case details::e_sf##op : return node_allocator_-> \ 33348 | allocate_rrr<details::sf3_var_node<Type,details::sf##op##_op<Type> > > \ 33349 | (v0, v1, v2); \ 33350 | 33351 | case_stmt(00) case_stmt(01) case_stmt(02) case_stmt(03) 33352 | case_stmt(04) case_stmt(05) case_stmt(06) case_stmt(07) 33353 | case_stmt(08) case_stmt(09) case_stmt(10) case_stmt(11) 33354 | case_stmt(12) case_stmt(13) case_stmt(14) case_stmt(15) 33355 | case_stmt(16) case_stmt(17) case_stmt(18) case_stmt(19) 33356 | case_stmt(20) case_stmt(21) case_stmt(22) case_stmt(23) 33357 | case_stmt(24) case_stmt(25) case_stmt(26) case_stmt(27) 33358 | case_stmt(28) case_stmt(29) case_stmt(30) case_stmt(31) 33359 | case_stmt(32) case_stmt(33) case_stmt(34) case_stmt(35) 33360 | case_stmt(36) case_stmt(37) case_stmt(38) case_stmt(39) 33361 | case_stmt(40) case_stmt(41) case_stmt(42) case_stmt(43) 33362 | case_stmt(44) case_stmt(45) case_stmt(46) case_stmt(47) 33363 | #undef case_stmt 33364 | default : return error_node(); 33365 | } 33366 | } 33367 | 33368 | inline expression_node_ptr special_function(const details::operator_type& operation, expression_node_ptr (&branch)[3]) 33369 | { 33370 | if (!all_nodes_valid(branch)) 33371 | return error_node(); 33372 | else if (is_constant_foldable(branch)) 33373 | return const_optimise_sf3(operation,branch); 33374 | else if (all_nodes_variables(branch)) 33375 | return varnode_optimise_sf3(operation,branch); 33376 | else 33377 | { 33378 | switch (operation) 33379 | { 33380 | #define case_stmt(op) \ 33381 | case details::e_sf##op : return node_allocator_-> \ 33382 | allocate<details::sf3_node<Type,details::sf##op##_op<Type> > > \ 33383 | (operation, branch); \ 33384 | 33385 | case_stmt(00) case_stmt(01) case_stmt(02) case_stmt(03) 33386 | case_stmt(04) case_stmt(05) case_stmt(06) case_stmt(07) 33387 | case_stmt(08) case_stmt(09) case_stmt(10) case_stmt(11) 33388 | case_stmt(12) case_stmt(13) case_stmt(14) case_stmt(15) 33389 | case_stmt(16) case_stmt(17) case_stmt(18) case_stmt(19) 33390 | case_stmt(20) case_stmt(21) case_stmt(22) case_stmt(23) 33391 | case_stmt(24) case_stmt(25) case_stmt(26) case_stmt(27) 33392 | case_stmt(28) case_stmt(29) case_stmt(30) case_stmt(31) 33393 | case_stmt(32) case_stmt(33) case_stmt(34) case_stmt(35) 33394 | case_stmt(36) case_stmt(37) case_stmt(38) case_stmt(39) 33395 | case_stmt(40) case_stmt(41) case_stmt(42) case_stmt(43) 33396 | case_stmt(44) case_stmt(45) case_stmt(46) case_stmt(47) 33397 | #undef case_stmt 33398 | default : return error_node(); 33399 | } 33400 | } 33401 | } 33402 | 33403 | inline expression_node_ptr const_optimise_sf4(const details::operator_type& operation, expression_node_ptr (&branch)[4]) 33404 | { 33405 | expression_node_ptr temp_node = error_node(); 33406 | 33407 | switch (operation) 33408 | { 33409 | #define case_stmt(op) \ 33410 | case details::e_sf##op : temp_node = node_allocator_-> \ 33411 | allocate<details::sf4_node<Type,details::sf##op##_op<Type> > > \ 33412 | (operation, branch); \ 33413 | break; \ 33414 | 33415 | case_stmt(48) case_stmt(49) case_stmt(50) case_stmt(51) 33416 | case_stmt(52) case_stmt(53) case_stmt(54) case_stmt(55) 33417 | case_stmt(56) case_stmt(57) case_stmt(58) case_stmt(59) 33418 | case_stmt(60) case_stmt(61) case_stmt(62) case_stmt(63) 33419 | case_stmt(64) case_stmt(65) case_stmt(66) case_stmt(67) 33420 | case_stmt(68) case_stmt(69) case_stmt(70) case_stmt(71) 33421 | case_stmt(72) case_stmt(73) case_stmt(74) case_stmt(75) 33422 | case_stmt(76) case_stmt(77) case_stmt(78) case_stmt(79) 33423 | case_stmt(80) case_stmt(81) case_stmt(82) case_stmt(83) 33424 | case_stmt(84) case_stmt(85) case_stmt(86) case_stmt(87) 33425 | case_stmt(88) case_stmt(89) case_stmt(90) case_stmt(91) 33426 | case_stmt(92) case_stmt(93) case_stmt(94) case_stmt(95) 33427 | case_stmt(96) case_stmt(97) case_stmt(98) case_stmt(99) 33428 | #undef case_stmt 33429 | default : return error_node(); 33430 | } 33431 | 33432 | assert(temp_node); 33433 | 33434 | const T v = temp_node->value(); 33435 | 33436 | details::free_node(*node_allocator_,temp_node); 33437 | 33438 | return node_allocator_->allocate<literal_node_t>(v); 33439 | } 33440 | 33441 | inline expression_node_ptr varnode_optimise_sf4(const details::operator_type& operation, expression_node_ptr (&branch)[4]) 33442 | { 33443 | typedef details::variable_node<Type>* variable_ptr; 33444 | 33445 | const Type& v0 = static_cast<variable_ptr>(branch[0])->ref(); 33446 | const Type& v1 = static_cast<variable_ptr>(branch[1])->ref(); 33447 | const Type& v2 = static_cast<variable_ptr>(branch[2])->ref(); 33448 | const Type& v3 = static_cast<variable_ptr>(branch[3])->ref(); 33449 | 33450 | switch (operation) 33451 | { 33452 | #define case_stmt(op) \ 33453 | case details::e_sf##op : return node_allocator_-> \ 33454 | allocate_rrrr<details::sf4_var_node<Type,details::sf##op##_op<Type> > > \ 33455 | (v0, v1, v2, v3); \ 33456 | 33457 | case_stmt(48) case_stmt(49) case_stmt(50) case_stmt(51) 33458 | case_stmt(52) case_stmt(53) case_stmt(54) case_stmt(55) 33459 | case_stmt(56) case_stmt(57) case_stmt(58) case_stmt(59) 33460 | case_stmt(60) case_stmt(61) case_stmt(62) case_stmt(63) 33461 | case_stmt(64) case_stmt(65) case_stmt(66) case_stmt(67) 33462 | case_stmt(68) case_stmt(69) case_stmt(70) case_stmt(71) 33463 | case_stmt(72) case_stmt(73) case_stmt(74) case_stmt(75) 33464 | case_stmt(76) case_stmt(77) case_stmt(78) case_stmt(79) 33465 | case_stmt(80) case_stmt(81) case_stmt(82) case_stmt(83) 33466 | case_stmt(84) case_stmt(85) case_stmt(86) case_stmt(87) 33467 | case_stmt(88) case_stmt(89) case_stmt(90) case_stmt(91) 33468 | case_stmt(92) case_stmt(93) case_stmt(94) case_stmt(95) 33469 | case_stmt(96) case_stmt(97) case_stmt(98) case_stmt(99) 33470 | #undef case_stmt 33471 | default : return error_node(); 33472 | } 33473 | } 33474 | 33475 | inline expression_node_ptr special_function(const details::operator_type& operation, expression_node_ptr (&branch)[4]) 33476 | { 33477 | if (!all_nodes_valid(branch)) 33478 | return error_node(); 33479 | else if (is_constant_foldable(branch)) 33480 | return const_optimise_sf4(operation,branch); 33481 | else if (all_nodes_variables(branch)) 33482 | return varnode_optimise_sf4(operation,branch); 33483 | switch (operation) 33484 | { 33485 | #define case_stmt(op) \ 33486 | case details::e_sf##op : return node_allocator_-> \ 33487 | allocate<details::sf4_node<Type,details::sf##op##_op<Type> > > \ 33488 | (operation, branch); \ 33489 | 33490 | case_stmt(48) case_stmt(49) case_stmt(50) case_stmt(51) 33491 | case_stmt(52) case_stmt(53) case_stmt(54) case_stmt(55) 33492 | case_stmt(56) case_stmt(57) case_stmt(58) case_stmt(59) 33493 | case_stmt(60) case_stmt(61) case_stmt(62) case_stmt(63) 33494 | case_stmt(64) case_stmt(65) case_stmt(66) case_stmt(67) 33495 | case_stmt(68) case_stmt(69) case_stmt(70) case_stmt(71) 33496 | case_stmt(72) case_stmt(73) case_stmt(74) case_stmt(75) 33497 | case_stmt(76) case_stmt(77) case_stmt(78) case_stmt(79) 33498 | case_stmt(80) case_stmt(81) case_stmt(82) case_stmt(83) 33499 | case_stmt(84) case_stmt(85) case_stmt(86) case_stmt(87) 33500 | case_stmt(88) case_stmt(89) case_stmt(90) case_stmt(91) 33501 | case_stmt(92) case_stmt(93) case_stmt(94) case_stmt(95) 33502 | case_stmt(96) case_stmt(97) case_stmt(98) case_stmt(99) 33503 | #undef case_stmt 33504 | default : return error_node(); 33505 | } 33506 | } 33507 | 33508 | template <typename Allocator, 33509 | template <typename, typename> class Sequence> 33510 | inline expression_node_ptr const_optimise_varargfunc(const details::operator_type& operation, Sequence<expression_node_ptr,Allocator>& arg_list) 33511 | { 33512 | expression_node_ptr temp_node = error_node(); 33513 | 33514 | switch (operation) 33515 | { 33516 | #define case_stmt(op0, op1) \ 33517 | case op0 : temp_node = node_allocator_-> \ 33518 | allocate<details::vararg_node<Type,op1<Type> > > \ 33519 | (arg_list); \ 33520 | break; \ 33521 | 33522 | case_stmt(details::e_sum , details::vararg_add_op ) 33523 | case_stmt(details::e_prod , details::vararg_mul_op ) 33524 | case_stmt(details::e_avg , details::vararg_avg_op ) 33525 | case_stmt(details::e_min , details::vararg_min_op ) 33526 | case_stmt(details::e_max , details::vararg_max_op ) 33527 | case_stmt(details::e_mand , details::vararg_mand_op ) 33528 | case_stmt(details::e_mor , details::vararg_mor_op ) 33529 | case_stmt(details::e_multi , details::vararg_multi_op) 33530 | #undef case_stmt 33531 | default : return error_node(); 33532 | } 33533 | 33534 | const T v = temp_node->value(); 33535 | 33536 | details::free_node(*node_allocator_,temp_node); 33537 | 33538 | return node_allocator_->allocate<literal_node_t>(v); 33539 | } 33540 | 33541 | inline bool special_one_parameter_vararg(const details::operator_type& operation) const 33542 | { 33543 | return ( 33544 | (details::e_sum == operation) || 33545 | (details::e_prod == operation) || 33546 | (details::e_avg == operation) || 33547 | (details::e_min == operation) || 33548 | (details::e_max == operation) 33549 | ); 33550 | } 33551 | 33552 | template <typename Allocator, 33553 | template <typename, typename> class Sequence> 33554 | inline expression_node_ptr varnode_optimise_varargfunc(const details::operator_type& operation, 33555 | Sequence<expression_node_ptr,Allocator>& arg_list) 33556 | { 33557 | switch (operation) 33558 | { 33559 | #define case_stmt(op0, op1) \ 33560 | case op0 : return node_allocator_-> \ 33561 | allocate<details::vararg_varnode<Type,op1<Type> > >(arg_list); \ 33562 | 33563 | case_stmt(details::e_sum , details::vararg_add_op ) 33564 | case_stmt(details::e_prod , details::vararg_mul_op ) 33565 | case_stmt(details::e_avg , details::vararg_avg_op ) 33566 | case_stmt(details::e_min , details::vararg_min_op ) 33567 | case_stmt(details::e_max , details::vararg_max_op ) 33568 | case_stmt(details::e_mand , details::vararg_mand_op ) 33569 | case_stmt(details::e_mor , details::vararg_mor_op ) 33570 | case_stmt(details::e_multi , details::vararg_multi_op) 33571 | #undef case_stmt 33572 | default : return error_node(); 33573 | } 33574 | } 33575 | 33576 | template <typename Allocator, 33577 | template <typename, typename> class Sequence> 33578 | inline expression_node_ptr vectorize_func(const details::operator_type& operation, 33579 | Sequence<expression_node_ptr,Allocator>& arg_list) 33580 | { 33581 | if (1 == arg_list.size()) 33582 | { 33583 | switch (operation) 33584 | { 33585 | #define case_stmt(op0, op1) \ 33586 | case op0 : return node_allocator_-> \ 33587 | allocate<details::vectorize_node<Type,op1<Type> > >(arg_list[0]); \ 33588 | 33589 | case_stmt(details::e_sum , details::vec_add_op) 33590 | case_stmt(details::e_prod , details::vec_mul_op) 33591 | case_stmt(details::e_avg , details::vec_avg_op) 33592 | case_stmt(details::e_min , details::vec_min_op) 33593 | case_stmt(details::e_max , details::vec_max_op) 33594 | #undef case_stmt 33595 | default : return error_node(); 33596 | } 33597 | } 33598 | else 33599 | return error_node(); 33600 | } 33601 | 33602 | template <typename Allocator, 33603 | template <typename, typename> class Sequence> 33604 | inline expression_node_ptr vararg_function(const details::operator_type& operation, 33605 | Sequence<expression_node_ptr,Allocator>& arg_list) 33606 | { 33607 | if (!all_nodes_valid(arg_list)) 33608 | { 33609 | details::free_all_nodes(*node_allocator_,arg_list); 33610 | 33611 | return error_node(); 33612 | } 33613 | else if (is_constant_foldable(arg_list)) 33614 | return const_optimise_varargfunc(operation,arg_list); 33615 | else if ((1 == arg_list.size()) && details::is_ivector_node(arg_list[0])) 33616 | return vectorize_func(operation,arg_list); 33617 | else if ((1 == arg_list.size()) && special_one_parameter_vararg(operation)) 33618 | return arg_list[0]; 33619 | else if (all_nodes_variables(arg_list)) 33620 | return varnode_optimise_varargfunc(operation,arg_list); 33621 | 33622 | #ifndef exprtk_disable_string_capabilities 33623 | if (details::e_smulti == operation) 33624 | { 33625 | expression_node_ptr result = node_allocator_-> 33626 | allocate<details::str_vararg_node<Type,details::vararg_multi_op<Type> > >(arg_list); 33627 | if (result && result->valid()) 33628 | { 33629 | return result; 33630 | } 33631 | 33632 | parser_->set_error(parser_error::make_error( 33633 | parser_error::e_synthesis, 33634 | token_t(), 33635 | "ERR262 - Failed to synthesize node: str_vararg_node<vararg_multi_op>", 33636 | exprtk_error_location)); 33637 | 33638 | details::free_node(*node_allocator_, result); 33639 | } 33640 | else 33641 | #endif 33642 | { 33643 | expression_node_ptr result = error_node(); 33644 | 33645 | switch (operation) 33646 | { 33647 | #define case_stmt(op0, op1) \ 33648 | case op0 : result = node_allocator_-> \ 33649 | allocate<details::vararg_node<Type,op1<Type> > >(arg_list); \ 33650 | break; \ 33651 | 33652 | case_stmt(details::e_sum , details::vararg_add_op ) 33653 | case_stmt(details::e_prod , details::vararg_mul_op ) 33654 | case_stmt(details::e_avg , details::vararg_avg_op ) 33655 | case_stmt(details::e_min , details::vararg_min_op ) 33656 | case_stmt(details::e_max , details::vararg_max_op ) 33657 | case_stmt(details::e_mand , details::vararg_mand_op ) 33658 | case_stmt(details::e_mor , details::vararg_mor_op ) 33659 | case_stmt(details::e_multi , details::vararg_multi_op) 33660 | #undef case_stmt 33661 | default : return error_node(); 33662 | } 33663 | 33664 | if (result && result->valid()) 33665 | { 33666 | return result; 33667 | } 33668 | 33669 | parser_->set_error(parser_error::make_error( 33670 | parser_error::e_synthesis, 33671 | token_t(), 33672 | "ERR263 - Failed to synthesize node: vararg_node", 33673 | exprtk_error_location)); 33674 | 33675 | details::free_node(*node_allocator_, result); 33676 | } 33677 | 33678 | return error_node(); 33679 | } 33680 | 33681 | template <std::size_t N> 33682 | inline expression_node_ptr function(ifunction_t* f, expression_node_ptr (&b)[N]) 33683 | { 33684 | typedef typename details::function_N_node<T,ifunction_t,N> function_N_node_t; 33685 | expression_node_ptr result = synthesize_expression<function_N_node_t,N>(f,b); 33686 | 33687 | if (0 == result) 33688 | return error_node(); 33689 | else 33690 | { 33691 | // Can the function call be completely optimised? 33692 | if (details::is_constant_node(result)) 33693 | return result; 33694 | else if (!all_nodes_valid(b)) 33695 | { 33696 | details::free_node(*node_allocator_,result); 33697 | std::fill_n(b, N, reinterpret_cast<expression_node_ptr>(0)); 33698 | 33699 | return error_node(); 33700 | } 33701 | else if (N != f->param_count) 33702 | { 33703 | details::free_node(*node_allocator_,result); 33704 | std::fill_n(b, N, reinterpret_cast<expression_node_ptr>(0)); 33705 | 33706 | return error_node(); 33707 | } 33708 | 33709 | function_N_node_t* func_node_ptr = reinterpret_cast<function_N_node_t*>(result); 33710 | 33711 | if (!func_node_ptr->init_branches(b)) 33712 | { 33713 | details::free_node(*node_allocator_,result); 33714 | std::fill_n(b, N, reinterpret_cast<expression_node_ptr>(0)); 33715 | 33716 | return error_node(); 33717 | } 33718 | 33719 | if (result && result->valid()) 33720 | { 33721 | return result; 33722 | } 33723 | 33724 | parser_->set_error(parser_error::make_error( 33725 | parser_error::e_synthesis, 33726 | token_t(), 33727 | "ERR264 - Failed to synthesize node: function_N_node_t", 33728 | exprtk_error_location)); 33729 | 33730 | details::free_node(*node_allocator_, result); 33731 | return error_node(); 33732 | } 33733 | } 33734 | 33735 | inline expression_node_ptr function(ifunction_t* f) 33736 | { 33737 | typedef typename details::function_N_node<Type,ifunction_t,0> function_N_node_t; 33738 | return node_allocator_->allocate<function_N_node_t>(f); 33739 | } 33740 | 33741 | inline expression_node_ptr vararg_function_call(ivararg_function_t* vaf, 33742 | std::vector<expression_node_ptr>& arg_list) 33743 | { 33744 | if (!all_nodes_valid(arg_list)) 33745 | { 33746 | details::free_all_nodes(*node_allocator_,arg_list); 33747 | 33748 | return error_node(); 33749 | } 33750 | 33751 | typedef details::vararg_function_node<Type,ivararg_function_t> alloc_type; 33752 | 33753 | expression_node_ptr result = node_allocator_->allocate<alloc_type>(vaf,arg_list); 33754 | 33755 | if ( 33756 | !arg_list.empty() && 33757 | !vaf->has_side_effects() && 33758 | is_constant_foldable(arg_list) 33759 | ) 33760 | { 33761 | const Type v = result->value(); 33762 | details::free_node(*node_allocator_,result); 33763 | result = node_allocator_->allocate<literal_node_t>(v); 33764 | } 33765 | 33766 | parser_->state_.activate_side_effect("vararg_function_call()"); 33767 | 33768 | if (result && result->valid()) 33769 | { 33770 | return result; 33771 | } 33772 | 33773 | parser_->set_error(parser_error::make_error( 33774 | parser_error::e_synthesis, 33775 | token_t(), 33776 | "ERR265 - Failed to synthesize node: vararg_function_node<ivararg_function_t>", 33777 | exprtk_error_location)); 33778 | 33779 | details::free_node(*node_allocator_, result); 33780 | return error_node(); 33781 | } 33782 | 33783 | inline expression_node_ptr generic_function_call(igeneric_function_t* gf, 33784 | std::vector<expression_node_ptr>& arg_list, 33785 | const std::size_t& param_seq_index = std::numeric_limits<std::size_t>::max()) 33786 | { 33787 | if (!all_nodes_valid(arg_list)) 33788 | { 33789 | details::free_all_nodes(*node_allocator_,arg_list); 33790 | return error_node(); 33791 | } 33792 | 33793 | typedef details::generic_function_node <Type,igeneric_function_t> alloc_type1; 33794 | typedef details::multimode_genfunction_node<Type,igeneric_function_t> alloc_type2; 33795 | 33796 | const std::size_t no_psi = std::numeric_limits<std::size_t>::max(); 33797 | 33798 | expression_node_ptr result = error_node(); 33799 | std::string node_name = "Unknown" 33800 | 33801 | if (no_psi == param_seq_index) 33802 | { 33803 | result = node_allocator_->allocate<alloc_type1>(arg_list,gf); 33804 | node_name = "generic_function_node<igeneric_function_t>" 33805 | } 33806 | else 33807 | { 33808 | result = node_allocator_->allocate<alloc_type2>(gf, param_seq_index, arg_list); 33809 | node_name = "multimode_genfunction_node<igeneric_function_t>" 33810 | } 33811 | 33812 | alloc_type1* genfunc_node_ptr = static_cast<alloc_type1*>(result); 33813 | 33814 | assert(genfunc_node_ptr); 33815 | 33816 | if ( 33817 | !arg_list.empty() && 33818 | !gf->has_side_effects() && 33819 | parser_->state_.type_check_enabled && 33820 | is_constant_foldable(arg_list) 33821 | ) 33822 | { 33823 | genfunc_node_ptr->init_branches(); 33824 | 33825 | const Type v = result->value(); 33826 | 33827 | details::free_node(*node_allocator_,result); 33828 | 33829 | return node_allocator_->allocate<literal_node_t>(v); 33830 | } 33831 | else if (genfunc_node_ptr->init_branches()) 33832 | { 33833 | if (result && result->valid()) 33834 | { 33835 | parser_->state_.activate_side_effect("generic_function_call()"); 33836 | return result; 33837 | } 33838 | 33839 | parser_->set_error(parser_error::make_error( 33840 | parser_error::e_synthesis, 33841 | token_t(), 33842 | "ERR266 - Failed to synthesize node: " + node_name, 33843 | exprtk_error_location)); 33844 | 33845 | details::free_node(*node_allocator_, result); 33846 | return error_node(); 33847 | } 33848 | else 33849 | { 33850 | details::free_node(*node_allocator_, result); 33851 | details::free_all_nodes(*node_allocator_, arg_list); 33852 | 33853 | return error_node(); 33854 | } 33855 | } 33856 | 33857 | #ifndef exprtk_disable_string_capabilities 33858 | inline expression_node_ptr string_function_call(igeneric_function_t* gf, 33859 | std::vector<expression_node_ptr>& arg_list, 33860 | const std::size_t& param_seq_index = std::numeric_limits<std::size_t>::max()) 33861 | { 33862 | if (!all_nodes_valid(arg_list)) 33863 | { 33864 | details::free_all_nodes(*node_allocator_,arg_list); 33865 | return error_node(); 33866 | } 33867 | 33868 | typedef details::string_function_node <Type,igeneric_function_t> alloc_type1; 33869 | typedef details::multimode_strfunction_node<Type,igeneric_function_t> alloc_type2; 33870 | 33871 | const std::size_t no_psi = std::numeric_limits<std::size_t>::max(); 33872 | 33873 | expression_node_ptr result = error_node(); 33874 | std::string node_name = "Unknown" 33875 | 33876 | if (no_psi == param_seq_index) 33877 | { 33878 | result = node_allocator_->allocate<alloc_type1>(gf,arg_list); 33879 | node_name = "string_function_node<igeneric_function_t>" 33880 | } 33881 | else 33882 | { 33883 | result = node_allocator_->allocate<alloc_type2>(gf, param_seq_index, arg_list); 33884 | node_name = "multimode_strfunction_node<igeneric_function_t>" 33885 | } 33886 | 33887 | alloc_type1* strfunc_node_ptr = static_cast<alloc_type1*>(result); 33888 | 33889 | assert(strfunc_node_ptr); 33890 | 33891 | if ( 33892 | !arg_list.empty() && 33893 | !gf->has_side_effects() && 33894 | is_constant_foldable(arg_list) 33895 | ) 33896 | { 33897 | strfunc_node_ptr->init_branches(); 33898 | 33899 | const Type v = result->value(); 33900 | 33901 | details::free_node(*node_allocator_,result); 33902 | 33903 | return node_allocator_->allocate<literal_node_t>(v); 33904 | } 33905 | else if (strfunc_node_ptr->init_branches()) 33906 | { 33907 | if (result && result->valid()) 33908 | { 33909 | parser_->state_.activate_side_effect("string_function_call()"); 33910 | return result; 33911 | } 33912 | 33913 | parser_->set_error(parser_error::make_error( 33914 | parser_error::e_synthesis, 33915 | token_t(), 33916 | "ERR267 - Failed to synthesize node: " + node_name, 33917 | exprtk_error_location)); 33918 | 33919 | details::free_node(*node_allocator_, result); 33920 | return error_node(); 33921 | } 33922 | else 33923 | { 33924 | details::free_node (*node_allocator_,result ); 33925 | details::free_all_nodes(*node_allocator_,arg_list); 33926 | 33927 | return error_node(); 33928 | } 33929 | } 33930 | #endif 33931 | 33932 | #ifndef exprtk_disable_return_statement 33933 | inline expression_node_ptr return_call(std::vector<expression_node_ptr>& arg_list) 33934 | { 33935 | if (!all_nodes_valid(arg_list)) 33936 | { 33937 | details::free_all_nodes(*node_allocator_,arg_list); 33938 | return error_node(); 33939 | } 33940 | 33941 | typedef details::return_node<Type> alloc_type; 33942 | 33943 | expression_node_ptr result = node_allocator_-> 33944 | allocate_rr<alloc_type>(arg_list,parser_->results_ctx()); 33945 | 33946 | alloc_type* return_node_ptr = static_cast<alloc_type*>(result); 33947 | 33948 | assert(return_node_ptr); 33949 | 33950 | if (return_node_ptr->init_branches()) 33951 | { 33952 | if (result && result->valid()) 33953 | { 33954 | parser_->state_.activate_side_effect("return_call()"); 33955 | return result; 33956 | } 33957 | 33958 | parser_->set_error(parser_error::make_error( 33959 | parser_error::e_synthesis, 33960 | token_t(), 33961 | "ERR268 - Failed to synthesize node: return_node", 33962 | exprtk_error_location)); 33963 | 33964 | details::free_node(*node_allocator_, result); 33965 | return error_node(); 33966 | } 33967 | else 33968 | { 33969 | details::free_node (*node_allocator_, result ); 33970 | details::free_all_nodes(*node_allocator_, arg_list); 33971 | 33972 | return error_node(); 33973 | } 33974 | } 33975 | 33976 | inline expression_node_ptr return_envelope(expression_node_ptr body, 33977 | results_context_t* rc, 33978 | bool*& return_invoked) 33979 | { 33980 | typedef details::return_envelope_node<Type> alloc_type; 33981 | 33982 | expression_node_ptr result = node_allocator_-> 33983 | allocate_cr<alloc_type>(body,(*rc)); 33984 | 33985 | return_invoked = static_cast<alloc_type*>(result)->retinvk_ptr(); 33986 | 33987 | return result; 33988 | } 33989 | #else 33990 | inline expression_node_ptr return_call(std::vector<expression_node_ptr>&) 33991 | { 33992 | return error_node(); 33993 | } 33994 | 33995 | inline expression_node_ptr return_envelope(expression_node_ptr, 33996 | results_context_t*, 33997 | bool*&) 33998 | { 33999 | return error_node(); 34000 | } 34001 | #endif 34002 | 34003 | inline expression_node_ptr vector_element(const std::string& symbol, 34004 | vector_holder_ptr vector_base, 34005 | expression_node_ptr vec_node, 34006 | expression_node_ptr index) 34007 | { 34008 | expression_node_ptr result = error_node(); 34009 | std::string node_name = "Unknown" 34010 | 34011 | if (details::is_constant_node(index)) 34012 | { 34013 | const std::size_t vec_index = static_cast<std::size_t>(details::numeric::to_int64(index->value())); 34014 | 34015 | details::free_node(*node_allocator_,index); 34016 | 34017 | if (vec_index >= vector_base->size()) 34018 | { 34019 | parser_->set_error(parser_error::make_error( 34020 | parser_error::e_parser, 34021 | token_t(), 34022 | "ERR269 - Index of " + details::to_str(vec_index) + " out of range for " 34023 | "vector '" + symbol + "' of size " + details::to_str(vector_base->size()), 34024 | exprtk_error_location)); 34025 | 34026 | details::free_node(*node_allocator_,vec_node); 34027 | 34028 | return error_node(); 34029 | } 34030 | 34031 | if (vector_base->rebaseable()) 34032 | { 34033 | vector_access_runtime_check_ptr rtc = get_vector_access_runtime_check(); 34034 | 34035 | result = (rtc) ? 34036 | node_allocator_->allocate<rebasevector_celem_rtc_node_t>(vec_node, vec_index, vector_base, rtc) : 34037 | node_allocator_->allocate<rebasevector_celem_node_t >(vec_node, vec_index, vector_base ) ; 34038 | 34039 | node_name = (rtc) ? 34040 | "rebasevector_elem_rtc_node_t" : 34041 | "rebasevector_elem_node_t" ; 34042 | 34043 | if (result && result->valid()) 34044 | { 34045 | return result; 34046 | } 34047 | 34048 | parser_->set_error(parser_error::make_error( 34049 | parser_error::e_synthesis, 34050 | token_t(), 34051 | "ERR270 - Failed to synthesize node: " + node_name + " for vector: " + symbol, 34052 | exprtk_error_location)); 34053 | 34054 | details::free_node(*node_allocator_, result); 34055 | return error_node(); 34056 | } 34057 | else if (details::is_ivector_node(vec_node) && !details::is_vector_node(vec_node)) 34058 | { 34059 | vector_access_runtime_check_ptr rtc = get_vector_access_runtime_check(); 34060 | 34061 | result = (rtc) ? 34062 | node_allocator_->allocate<vector_celem_rtc_node_t>(vec_node, vec_index, vector_base, rtc) : 34063 | node_allocator_->allocate<vector_celem_node_t >(vec_node, vec_index, vector_base ) ; 34064 | 34065 | node_name = (rtc) ? 34066 | "vector_elem_rtc_node_t" : 34067 | "vector_elem_node_t" ; 34068 | 34069 | if (result && result->valid()) 34070 | { 34071 | return result; 34072 | } 34073 | 34074 | parser_->set_error(parser_error::make_error( 34075 | parser_error::e_synthesis, 34076 | token_t(), 34077 | "ERR271 - Failed to synthesize node: " + node_name + " for vector: " + symbol, 34078 | exprtk_error_location)); 34079 | 34080 | details::free_node(*node_allocator_, result); 34081 | return error_node(); 34082 | } 34083 | 34084 | const scope_element& se = parser_->sem_.get_element(symbol,vec_index); 34085 | 34086 | if (se.index == vec_index) 34087 | { 34088 | result = se.var_node; 34089 | details::free_node(*node_allocator_,vec_node); 34090 | } 34091 | else 34092 | { 34093 | scope_element nse; 34094 | nse.name = symbol; 34095 | nse.active = true; 34096 | nse.ref_count = 1; 34097 | nse.type = scope_element::e_vecelem; 34098 | nse.index = vec_index; 34099 | nse.depth = parser_->state_.scope_depth; 34100 | nse.data = 0; 34101 | nse.var_node = node_allocator_->allocate<variable_node_t>((*(*vector_base)[vec_index])); 34102 | 34103 | if (!parser_->sem_.add_element(nse)) 34104 | { 34105 | parser_->set_synthesis_error("Failed to add new local vector element to SEM [1]"); 34106 | 34107 | parser_->sem_.free_element(nse); 34108 | 34109 | result = error_node(); 34110 | } 34111 | 34112 | assert(parser_->sem_.total_local_symb_size_bytes() <= parser_->settings().max_total_local_symbol_size_bytes()); 34113 | 34114 | details::free_node(*node_allocator_,vec_node); 34115 | 34116 | exprtk_debug(("vector_element() - INFO - Added new local vector element: %s\n", nse.name.c_str())); 34117 | 34118 | parser_->state_.activate_side_effect("vector_element()"); 34119 | 34120 | result = nse.var_node; 34121 | node_name = "variable_node_t" 34122 | } 34123 | } 34124 | else 34125 | { 34126 | vector_access_runtime_check_ptr rtc = get_vector_access_runtime_check(); 34127 | 34128 | if (vector_base->rebaseable()) 34129 | { 34130 | result = (rtc) ? 34131 | node_allocator_->allocate<rebasevector_elem_rtc_node_t>(vec_node, index, vector_base, rtc) : 34132 | node_allocator_->allocate<rebasevector_elem_node_t >(vec_node, index, vector_base ) ; 34133 | 34134 | node_name = (rtc) ? 34135 | "rebasevector_elem_rtc_node_t" : 34136 | "rebasevector_elem_node_t" ; 34137 | } 34138 | else 34139 | { 34140 | result = rtc ? 34141 | node_allocator_->allocate<vector_elem_rtc_node_t>(vec_node, index, vector_base, rtc) : 34142 | node_allocator_->allocate<vector_elem_node_t >(vec_node, index, vector_base ) ; 34143 | 34144 | node_name = (rtc) ? 34145 | "vector_elem_rtc_node_t" : 34146 | "vector_elem_node_t" ; 34147 | } 34148 | } 34149 | 34150 | if (result && result->valid()) 34151 | { 34152 | return result; 34153 | } 34154 | 34155 | parser_->set_error(parser_error::make_error( 34156 | parser_error::e_synthesis, 34157 | token_t(), 34158 | "ERR272 - Failed to synthesize node: " + node_name, 34159 | exprtk_error_location)); 34160 | 34161 | details::free_node(*node_allocator_, result); 34162 | return error_node(); 34163 | } 34164 | 34165 | private: 34166 | 34167 | template <std::size_t N, typename NodePtr> 34168 | inline bool is_constant_foldable(NodePtr (&b)[N]) const 34169 | { 34170 | for (std::size_t i = 0; i < N; ++i) 34171 | { 34172 | if (0 == b[i]) 34173 | return false; 34174 | else if (!details::is_constant_node(b[i])) 34175 | return false; 34176 | } 34177 | 34178 | return true; 34179 | } 34180 | 34181 | template <typename NodePtr, 34182 | typename Allocator, 34183 | template <typename, typename> class Sequence> 34184 | inline bool is_constant_foldable(const Sequence<NodePtr,Allocator>& b) const 34185 | { 34186 | for (std::size_t i = 0; i < b.size(); ++i) 34187 | { 34188 | if (0 == b[i]) 34189 | return false; 34190 | else if (!details::is_constant_node(b[i])) 34191 | return false; 34192 | } 34193 | 34194 | return true; 34195 | } 34196 | 34197 | void lodge_assignment(symbol_type cst, expression_node_ptr node) 34198 | { 34199 | parser_->state_.activate_side_effect("lodge_assignment()"); 34200 | 34201 | if (!parser_->dec_.collect_assignments()) 34202 | return; 34203 | 34204 | std::string symbol_name; 34205 | 34206 | switch (cst) 34207 | { 34208 | case e_st_variable : symbol_name = parser_->symtab_store_ 34209 | .get_variable_name(node); 34210 | break; 34211 | 34212 | #ifndef exprtk_disable_string_capabilities 34213 | case e_st_string : symbol_name = parser_->symtab_store_ 34214 | .get_stringvar_name(node); 34215 | break; 34216 | #endif 34217 | 34218 | case e_st_vector : { 34219 | typedef details::vector_holder<T> vector_holder_t; 34220 | 34221 | vector_holder_t& vh = static_cast<vector_node_t*>(node)->vec_holder(); 34222 | 34223 | symbol_name = parser_->symtab_store_.get_vector_name(&vh); 34224 | } 34225 | break; 34226 | 34227 | case e_st_vecelem : { 34228 | typedef details::vector_holder<T> vector_holder_t; 34229 | 34230 | vector_holder_t& vh = static_cast<vector_elem_node_t*>(node)->vec_holder(); 34231 | 34232 | symbol_name = parser_->symtab_store_.get_vector_name(&vh); 34233 | 34234 | cst = e_st_vector; 34235 | } 34236 | break; 34237 | 34238 | default : return; 34239 | } 34240 | 34241 | if (!symbol_name.empty()) 34242 | { 34243 | parser_->dec_.add_assignment(symbol_name,cst); 34244 | } 34245 | } 34246 | 34247 | const void* base_ptr(expression_node_ptr node) 34248 | { 34249 | if (node) 34250 | { 34251 | switch (node->type()) 34252 | { 34253 | case details::expression_node<T>::e_variable: 34254 | return reinterpret_cast<const void*>(&static_cast<variable_node_t*>(node)->ref()); 34255 | 34256 | case details::expression_node<T>::e_vecelem: 34257 | return reinterpret_cast<const void*>(&static_cast<vector_elem_node_t*>(node)->ref()); 34258 | 34259 | case details::expression_node<T>::e_veccelem: 34260 | return reinterpret_cast<const void*>(&static_cast<vector_celem_node_t*>(node)->ref()); 34261 | 34262 | case details::expression_node<T>::e_vecelemrtc: 34263 | return reinterpret_cast<const void*>(&static_cast<vector_elem_rtc_node_t*>(node)->ref()); 34264 | 34265 | case details::expression_node<T>::e_veccelemrtc: 34266 | return reinterpret_cast<const void*>(&static_cast<vector_celem_rtc_node_t*>(node)->ref()); 34267 | 34268 | case details::expression_node<T>::e_rbvecelem: 34269 | return reinterpret_cast<const void*>(&static_cast<rebasevector_elem_node_t*>(node)->ref()); 34270 | 34271 | case details::expression_node<T>::e_rbvecelemrtc: 34272 | return reinterpret_cast<const void*>(&static_cast<rebasevector_elem_rtc_node_t*>(node)->ref()); 34273 | 34274 | case details::expression_node<T>::e_rbveccelem: 34275 | return reinterpret_cast<const void*>(&static_cast<rebasevector_celem_node_t*>(node)->ref()); 34276 | 34277 | case details::expression_node<T>::e_rbveccelemrtc: 34278 | return reinterpret_cast<const void*>(&static_cast<rebasevector_celem_rtc_node_t*>(node)->ref()); 34279 | 34280 | case details::expression_node<T>::e_vector: 34281 | return reinterpret_cast<const void*>(static_cast<vector_node_t*>(node)->vec_holder().data()); 34282 | 34283 | #ifndef exprtk_disable_string_capabilities 34284 | case details::expression_node<T>::e_stringvar: 34285 | return reinterpret_cast<const void*>((static_cast<stringvar_node_t*>(node)->base())); 34286 | 34287 | case details::expression_node<T>::e_stringvarrng: 34288 | return reinterpret_cast<const void*>((static_cast<string_range_node_t*>(node)->base())); 34289 | #endif 34290 | default : return reinterpret_cast<const void*>(0); 34291 | } 34292 | } 34293 | 34294 | return reinterpret_cast<const void*>(0); 34295 | } 34296 | 34297 | bool assign_immutable_symbol(expression_node_ptr node) 34298 | { 34299 | interval_t interval; 34300 | const void* baseptr_addr = base_ptr(node); 34301 | 34302 | exprtk_debug(("assign_immutable_symbol - base ptr addr: %p\n", baseptr_addr)); 34303 | 34304 | if (parser_->immutable_memory_map_.in_interval(baseptr_addr,interval)) 34305 | { 34306 | typename immutable_symtok_map_t::iterator itr = parser_->immutable_symtok_map_.find(interval); 34307 | 34308 | if (parser_->immutable_symtok_map_.end() != itr) 34309 | { 34310 | token_t& token = itr->second; 34311 | parser_->set_error(parser_error::make_error( 34312 | parser_error::e_parser, 34313 | token, 34314 | "ERR273 - Symbol '" + token.value + "' cannot be assigned-to as it is immutable.", 34315 | exprtk_error_location)); 34316 | } 34317 | else 34318 | parser_->set_synthesis_error("Unable to assign symbol is immutable."); 34319 | 34320 | return true; 34321 | } 34322 | 34323 | return false; 34324 | } 34325 | 34326 | inline expression_node_ptr synthesize_assignment_expression(const details::operator_type& operation, expression_node_ptr (&branch)[2]) 34327 | { 34328 | if (assign_immutable_symbol(branch[0])) 34329 | { 34330 | return error_node(); 34331 | } 34332 | else if (details::is_variable_node(branch[0])) 34333 | { 34334 | lodge_assignment(e_st_variable,branch[0]); 34335 | return synthesize_expression<assignment_node_t,2>(operation,branch); 34336 | } 34337 | else if (details::is_vector_elem_node(branch[0]) || details::is_vector_celem_node(branch[0])) 34338 | { 34339 | lodge_assignment(e_st_vecelem,branch[0]); 34340 | return synthesize_expression<assignment_vec_elem_node_t, 2>(operation, branch); 34341 | } 34342 | else if (details::is_vector_elem_rtc_node(branch[0]) || details::is_vector_celem_rtc_node(branch[0])) 34343 | { 34344 | lodge_assignment(e_st_vecelem,branch[0]); 34345 | return synthesize_expression<assignment_vec_elem_rtc_node_t, 2>(operation, branch); 34346 | } 34347 | else if (details::is_rebasevector_elem_node(branch[0])) 34348 | { 34349 | lodge_assignment(e_st_vecelem,branch[0]); 34350 | return synthesize_expression<assignment_rebasevec_elem_node_t, 2>(operation, branch); 34351 | } 34352 | else if (details::is_rebasevector_elem_rtc_node(branch[0])) 34353 | { 34354 | lodge_assignment(e_st_vecelem,branch[0]); 34355 | return synthesize_expression<assignment_rebasevec_elem_rtc_node_t, 2>(operation, branch); 34356 | } 34357 | else if (details::is_rebasevector_celem_node(branch[0])) 34358 | { 34359 | lodge_assignment(e_st_vecelem,branch[0]); 34360 | return synthesize_expression<assignment_rebasevec_celem_node_t, 2>(operation, branch); 34361 | } 34362 | #ifndef exprtk_disable_string_capabilities 34363 | else if (details::is_string_node(branch[0])) 34364 | { 34365 | lodge_assignment(e_st_string,branch[0]); 34366 | return synthesize_expression<assignment_string_node_t,2>(operation, branch); 34367 | } 34368 | else if (details::is_string_range_node(branch[0])) 34369 | { 34370 | lodge_assignment(e_st_string,branch[0]); 34371 | return synthesize_expression<assignment_string_range_node_t,2>(operation, branch); 34372 | } 34373 | #endif 34374 | else if (details::is_vector_node(branch[0])) 34375 | { 34376 | lodge_assignment(e_st_vector,branch[0]); 34377 | 34378 | if (details::is_ivector_node(branch[1])) 34379 | return synthesize_expression<assignment_vecvec_node_t,2>(operation, branch); 34380 | else 34381 | return synthesize_expression<assignment_vec_node_t,2>(operation, branch); 34382 | } 34383 | else if (details::is_literal_node(branch[0])) 34384 | { 34385 | parser_->set_error(parser_error::make_error( 34386 | parser_error::e_syntax, 34387 | parser_->current_state().token, 34388 | "ERR274 - Cannot assign value to const variable", 34389 | exprtk_error_location)); 34390 | 34391 | return error_node(); 34392 | } 34393 | else 34394 | { 34395 | parser_->set_error(parser_error::make_error( 34396 | parser_error::e_syntax, 34397 | parser_->current_state().token, 34398 | "ERR275 - Invalid branches for assignment operator '" + details::to_str(operation) + "'", 34399 | exprtk_error_location)); 34400 | 34401 | return error_node(); 34402 | } 34403 | } 34404 | 34405 | inline expression_node_ptr synthesize_assignment_operation_expression(const details::operator_type& operation, 34406 | expression_node_ptr (&branch)[2]) 34407 | { 34408 | if (assign_immutable_symbol(branch[0])) 34409 | { 34410 | return error_node(); 34411 | } 34412 | 34413 | expression_node_ptr result = error_node(); 34414 | std::string node_name = "Unknown" 34415 | 34416 | if (details::is_variable_node(branch[0])) 34417 | { 34418 | lodge_assignment(e_st_variable,branch[0]); 34419 | 34420 | switch (operation) 34421 | { 34422 | #define case_stmt(op0, op1) \ 34423 | case op0 : result = node_allocator_-> \ 34424 | template allocate_rrr<typename details::assignment_op_node<Type,op1<Type> > > \ 34425 | (operation, branch[0], branch[1]); \ 34426 | node_name = "assignment_op_node" \ 34427 | break; \ 34428 | 34429 | case_stmt(details::e_addass , details::add_op) 34430 | case_stmt(details::e_subass , details::sub_op) 34431 | case_stmt(details::e_mulass , details::mul_op) 34432 | case_stmt(details::e_divass , details::div_op) 34433 | case_stmt(details::e_modass , details::mod_op) 34434 | #undef case_stmt 34435 | default : return error_node(); 34436 | } 34437 | } 34438 | else if (details::is_vector_elem_node(branch[0])) 34439 | { 34440 | lodge_assignment(e_st_vecelem,branch[0]); 34441 | 34442 | switch (operation) 34443 | { 34444 | #define case_stmt(op0, op1) \ 34445 | case op0 : result = node_allocator_-> \ 34446 | template allocate_rrr<typename details::assignment_vec_elem_op_node<Type,op1<Type> > > \ 34447 | (operation, branch[0], branch[1]); \ 34448 | node_name = "assignment_vec_elem_op_node" \ 34449 | break; \ 34450 | 34451 | case_stmt(details::e_addass , details::add_op) 34452 | case_stmt(details::e_subass , details::sub_op) 34453 | case_stmt(details::e_mulass , details::mul_op) 34454 | case_stmt(details::e_divass , details::div_op) 34455 | case_stmt(details::e_modass , details::mod_op) 34456 | #undef case_stmt 34457 | default : return error_node(); 34458 | } 34459 | } 34460 | else if (details::is_vector_elem_rtc_node(branch[0])) 34461 | { 34462 | lodge_assignment(e_st_vecelem,branch[0]); 34463 | 34464 | switch (operation) 34465 | { 34466 | #define case_stmt(op0, op1) \ 34467 | case op0 : result = node_allocator_-> \ 34468 | template allocate_rrr<typename details::assignment_vec_elem_op_rtc_node<Type,op1<Type> > > \ 34469 | (operation, branch[0], branch[1]); \ 34470 | node_name = "assignment_vec_elem_op_rtc_node" \ 34471 | break; \ 34472 | 34473 | case_stmt(details::e_addass , details::add_op) 34474 | case_stmt(details::e_subass , details::sub_op) 34475 | case_stmt(details::e_mulass , details::mul_op) 34476 | case_stmt(details::e_divass , details::div_op) 34477 | case_stmt(details::e_modass , details::mod_op) 34478 | #undef case_stmt 34479 | default : return error_node(); 34480 | } 34481 | } 34482 | else if (details::is_vector_celem_rtc_node(branch[0])) 34483 | { 34484 | lodge_assignment(e_st_vecelem,branch[0]); 34485 | 34486 | switch (operation) 34487 | { 34488 | #define case_stmt(op0, op1) \ 34489 | case op0 : result = node_allocator_-> \ 34490 | template allocate_rrr<typename details::assignment_vec_celem_op_rtc_node<Type,op1<Type> > > \ 34491 | (operation, branch[0], branch[1]); \ 34492 | node_name = "assignment_vec_celem_op_rtc_node" \ 34493 | break; \ 34494 | 34495 | case_stmt(details::e_addass , details::add_op) 34496 | case_stmt(details::e_subass , details::sub_op) 34497 | case_stmt(details::e_mulass , details::mul_op) 34498 | case_stmt(details::e_divass , details::div_op) 34499 | case_stmt(details::e_modass , details::mod_op) 34500 | #undef case_stmt 34501 | default : return error_node(); 34502 | } 34503 | } 34504 | else if (details::is_rebasevector_elem_node(branch[0])) 34505 | { 34506 | lodge_assignment(e_st_vecelem,branch[0]); 34507 | 34508 | switch (operation) 34509 | { 34510 | #define case_stmt(op0, op1) \ 34511 | case op0 : result = node_allocator_-> \ 34512 | template allocate_rrr<typename details::assignment_rebasevec_elem_op_node<Type,op1<Type> > > \ 34513 | (operation, branch[0], branch[1]); \ 34514 | node_name = "assignment_rebasevec_elem_op_node" \ 34515 | break; \ 34516 | 34517 | case_stmt(details::e_addass , details::add_op) 34518 | case_stmt(details::e_subass , details::sub_op) 34519 | case_stmt(details::e_mulass , details::mul_op) 34520 | case_stmt(details::e_divass , details::div_op) 34521 | case_stmt(details::e_modass , details::mod_op) 34522 | #undef case_stmt 34523 | default : return error_node(); 34524 | } 34525 | } 34526 | else if (details::is_rebasevector_celem_node(branch[0])) 34527 | { 34528 | lodge_assignment(e_st_vecelem,branch[0]); 34529 | 34530 | switch (operation) 34531 | { 34532 | #define case_stmt(op0, op1) \ 34533 | case op0 : result = node_allocator_-> \ 34534 | template allocate_rrr<typename details::assignment_rebasevec_celem_op_node<Type,op1<Type> > > \ 34535 | (operation, branch[0], branch[1]); \ 34536 | node_name = "assignment_rebasevec_celem_op_node" \ 34537 | break; \ 34538 | 34539 | case_stmt(details::e_addass , details::add_op) 34540 | case_stmt(details::e_subass , details::sub_op) 34541 | case_stmt(details::e_mulass , details::mul_op) 34542 | case_stmt(details::e_divass , details::div_op) 34543 | case_stmt(details::e_modass , details::mod_op) 34544 | #undef case_stmt 34545 | default : return error_node(); 34546 | } 34547 | } 34548 | else if (details::is_rebasevector_elem_rtc_node(branch[0])) 34549 | { 34550 | lodge_assignment(e_st_vecelem,branch[0]); 34551 | 34552 | switch (operation) 34553 | { 34554 | #define case_stmt(op0, op1) \ 34555 | case op0 : result = node_allocator_-> \ 34556 | template allocate_rrr<typename details::assignment_rebasevec_elem_op_rtc_node<Type,op1<Type> > > \ 34557 | (operation, branch[0], branch[1]); \ 34558 | node_name = "assignment_rebasevec_elem_op_rtc_node" \ 34559 | break; \ 34560 | 34561 | case_stmt(details::e_addass , details::add_op) 34562 | case_stmt(details::e_subass , details::sub_op) 34563 | case_stmt(details::e_mulass , details::mul_op) 34564 | case_stmt(details::e_divass , details::div_op) 34565 | case_stmt(details::e_modass , details::mod_op) 34566 | #undef case_stmt 34567 | default : return error_node(); 34568 | } 34569 | } 34570 | else if (details::is_rebasevector_celem_rtc_node(branch[0])) 34571 | { 34572 | lodge_assignment(e_st_vecelem,branch[0]); 34573 | 34574 | switch (operation) 34575 | { 34576 | #define case_stmt(op0, op1) \ 34577 | case op0 : result = node_allocator_-> \ 34578 | template allocate_rrr<typename details::assignment_rebasevec_celem_op_rtc_node<Type,op1<Type> > > \ 34579 | (operation, branch[0], branch[1]); \ 34580 | node_name = "assignment_rebasevec_celem_op_rtc_node" \ 34581 | break; \ 34582 | 34583 | case_stmt(details::e_addass , details::add_op) 34584 | case_stmt(details::e_subass , details::sub_op) 34585 | case_stmt(details::e_mulass , details::mul_op) 34586 | case_stmt(details::e_divass , details::div_op) 34587 | case_stmt(details::e_modass , details::mod_op) 34588 | #undef case_stmt 34589 | default : return error_node(); 34590 | } 34591 | } 34592 | else if (details::is_vector_node(branch[0])) 34593 | { 34594 | lodge_assignment(e_st_vector,branch[0]); 34595 | 34596 | if (details::is_ivector_node(branch[1])) 34597 | { 34598 | switch (operation) 34599 | { 34600 | #define case_stmt(op0, op1) \ 34601 | case op0 : result = node_allocator_-> \ 34602 | template allocate_rrr<typename details::assignment_vecvec_op_node<Type,op1<Type> > > \ 34603 | (operation, branch[0], branch[1]); \ 34604 | node_name = "assignment_rebasevec_celem_op_node" \ 34605 | break; \ 34606 | 34607 | case_stmt(details::e_addass , details::add_op) 34608 | case_stmt(details::e_subass , details::sub_op) 34609 | case_stmt(details::e_mulass , details::mul_op) 34610 | case_stmt(details::e_divass , details::div_op) 34611 | case_stmt(details::e_modass , details::mod_op) 34612 | #undef case_stmt 34613 | default : return error_node(); 34614 | } 34615 | } 34616 | else 34617 | { 34618 | switch (operation) 34619 | { 34620 | #define case_stmt(op0, op1) \ 34621 | case op0 : result = node_allocator_-> \ 34622 | template allocate_rrr<typename details::assignment_vec_op_node<Type,op1<Type> > > \ 34623 | (operation, branch[0], branch[1]); \ 34624 | node_name = "assignment_vec_op_node" \ 34625 | break; \ 34626 | 34627 | case_stmt(details::e_addass , details::add_op) 34628 | case_stmt(details::e_subass , details::sub_op) 34629 | case_stmt(details::e_mulass , details::mul_op) 34630 | case_stmt(details::e_divass , details::div_op) 34631 | case_stmt(details::e_modass , details::mod_op) 34632 | #undef case_stmt 34633 | default : return error_node(); 34634 | } 34635 | } 34636 | } 34637 | #ifndef exprtk_disable_string_capabilities 34638 | else if ( 34639 | (details::e_addass == operation) && 34640 | details::is_string_node(branch[0]) 34641 | ) 34642 | { 34643 | typedef details::assignment_string_node<T,details::asn_addassignment> addass_t; 34644 | 34645 | lodge_assignment(e_st_string,branch[0]); 34646 | 34647 | result = synthesize_expression<addass_t,2>(operation,branch); 34648 | node_name = "assignment_string_node<T,details::asn_addassignment>" 34649 | } 34650 | #endif 34651 | else 34652 | { 34653 | parser_->set_error(parser_error::make_error( 34654 | parser_error::e_syntax, 34655 | parser_->current_state().token, 34656 | "ERR276 - Invalid branches for assignment operator '" + details::to_str(operation) + "'", 34657 | exprtk_error_location)); 34658 | 34659 | return error_node(); 34660 | } 34661 | 34662 | if (result && result->valid()) 34663 | { 34664 | return result; 34665 | } 34666 | 34667 | parser_->set_error(parser_error::make_error( 34668 | parser_error::e_synthesis, 34669 | token_t(), 34670 | "ERR277 - Failed to synthesize node: " + node_name, 34671 | exprtk_error_location)); 34672 | 34673 | details::free_node(*node_allocator_, result); 34674 | return error_node(); 34675 | } 34676 | 34677 | inline expression_node_ptr synthesize_veceqineqlogic_operation_expression(const details::operator_type& operation, 34678 | expression_node_ptr (&branch)[2]) 34679 | { 34680 | const bool is_b0_ivec = details::is_ivector_node(branch[0]); 34681 | const bool is_b1_ivec = details::is_ivector_node(branch[1]); 34682 | 34683 | #define batch_eqineq_logic_case \ 34684 | case_stmt(details::e_lt , details::lt_op ) \ 34685 | case_stmt(details::e_lte , details::lte_op ) \ 34686 | case_stmt(details::e_gt , details::gt_op ) \ 34687 | case_stmt(details::e_gte , details::gte_op ) \ 34688 | case_stmt(details::e_eq , details::eq_op ) \ 34689 | case_stmt(details::e_ne , details::ne_op ) \ 34690 | case_stmt(details::e_equal , details::equal_op) \ 34691 | case_stmt(details::e_and , details::and_op ) \ 34692 | case_stmt(details::e_nand , details::nand_op ) \ 34693 | case_stmt(details::e_or , details::or_op ) \ 34694 | case_stmt(details::e_nor , details::nor_op ) \ 34695 | case_stmt(details::e_xor , details::xor_op ) \ 34696 | case_stmt(details::e_xnor , details::xnor_op ) \ 34697 | 34698 | expression_node_ptr result = error_node(); 34699 | std::string node_name = "Unknown" 34700 | 34701 | if (is_b0_ivec && is_b1_ivec) 34702 | { 34703 | switch (operation) 34704 | { 34705 | #define case_stmt(op0, op1) \ 34706 | case op0 : result = node_allocator_-> \ 34707 | template allocate_rrr<typename details::vec_binop_vecvec_node<Type,op1<Type> > > \ 34708 | (operation, branch[0], branch[1]); \ 34709 | node_name = "vec_binop_vecvec_node" \ 34710 | break; \ 34711 | 34712 | batch_eqineq_logic_case 34713 | #undef case_stmt 34714 | default : return error_node(); 34715 | } 34716 | } 34717 | else if (is_b0_ivec && !is_b1_ivec) 34718 | { 34719 | switch (operation) 34720 | { 34721 | #define case_stmt(op0, op1) \ 34722 | case op0 : result = node_allocator_-> \ 34723 | template allocate_rrr<typename details::vec_binop_vecval_node<Type,op1<Type> > > \ 34724 | (operation, branch[0], branch[1]); \ 34725 | node_name = "vec_binop_vecval_node" \ 34726 | break; \ 34727 | 34728 | batch_eqineq_logic_case 34729 | #undef case_stmt 34730 | default : return error_node(); 34731 | } 34732 | } 34733 | else if (!is_b0_ivec && is_b1_ivec) 34734 | { 34735 | switch (operation) 34736 | { 34737 | #define case_stmt(op0, op1) \ 34738 | case op0 : result = node_allocator_-> \ 34739 | template allocate_rrr<typename details::vec_binop_valvec_node<Type,op1<Type> > > \ 34740 | (operation, branch[0], branch[1]); \ 34741 | node_name = "vec_binop_valvec_node" \ 34742 | break; \ 34743 | 34744 | batch_eqineq_logic_case 34745 | #undef case_stmt 34746 | default : return error_node(); 34747 | } 34748 | } 34749 | else 34750 | return error_node(); 34751 | 34752 | if (result && result->valid()) 34753 | { 34754 | return result; 34755 | } 34756 | 34757 | parser_->set_error(parser_error::make_error( 34758 | parser_error::e_synthesis, 34759 | token_t(), 34760 | "ERR278 - Failed to synthesize node: " + node_name, 34761 | exprtk_error_location)); 34762 | 34763 | details::free_node(*node_allocator_, result); 34764 | return error_node(); 34765 | 34766 | #undef batch_eqineq_logic_case 34767 | } 34768 | 34769 | inline expression_node_ptr synthesize_vecarithmetic_operation_expression(const details::operator_type& operation, 34770 | expression_node_ptr (&branch)[2]) 34771 | { 34772 | const bool is_b0_ivec = details::is_ivector_node(branch[0]); 34773 | const bool is_b1_ivec = details::is_ivector_node(branch[1]); 34774 | 34775 | #define vector_ops \ 34776 | case_stmt(details::e_add , details::add_op) \ 34777 | case_stmt(details::e_sub , details::sub_op) \ 34778 | case_stmt(details::e_mul , details::mul_op) \ 34779 | case_stmt(details::e_div , details::div_op) \ 34780 | case_stmt(details::e_mod , details::mod_op) \ 34781 | 34782 | expression_node_ptr result = error_node(); 34783 | std::string node_name = "Unknown" 34784 | 34785 | if (is_b0_ivec && is_b1_ivec) 34786 | { 34787 | switch (operation) 34788 | { 34789 | #define case_stmt(op0, op1) \ 34790 | case op0 : result = node_allocator_-> \ 34791 | template allocate_rrr<typename details::vec_binop_vecvec_node<Type,op1<Type> > > \ 34792 | (operation, branch[0], branch[1]); \ 34793 | node_name = "vec_binop_vecvec_node" \ 34794 | break; \ 34795 | 34796 | vector_ops 34797 | case_stmt(details::e_pow,details:: pow_op) 34798 | #undef case_stmt 34799 | default : return error_node(); 34800 | } 34801 | } 34802 | else if (is_b0_ivec && !is_b1_ivec) 34803 | { 34804 | switch (operation) 34805 | { 34806 | #define case_stmt(op0, op1) \ 34807 | case op0 : result = node_allocator_-> \ 34808 | template allocate_rrr<typename details::vec_binop_vecval_node<Type,op1<Type> > > \ 34809 | (operation, branch[0], branch[1]); \ 34810 | node_name = "vec_binop_vecval_node(b0ivec,!b1ivec)" \ 34811 | break; \ 34812 | 34813 | vector_ops 34814 | case_stmt(details::e_pow,details:: pow_op) 34815 | #undef case_stmt 34816 | default : return error_node(); 34817 | } 34818 | } 34819 | else if (!is_b0_ivec && is_b1_ivec) 34820 | { 34821 | switch (operation) 34822 | { 34823 | #define case_stmt(op0, op1) \ 34824 | case op0 : result = node_allocator_-> \ 34825 | template allocate_rrr<typename details::vec_binop_valvec_node<Type,op1<Type> > > \ 34826 | (operation, branch[0], branch[1]); \ 34827 | node_name = "vec_binop_vecval_node(!b0ivec,b1ivec)" \ 34828 | break; \ 34829 | 34830 | vector_ops 34831 | #undef case_stmt 34832 | default : return error_node(); 34833 | } 34834 | } 34835 | else 34836 | return error_node(); 34837 | 34838 | if (result && result->valid()) 34839 | { 34840 | return result; 34841 | } 34842 | 34843 | parser_->set_error(parser_error::make_error( 34844 | parser_error::e_synthesis, 34845 | token_t(), 34846 | "ERR279 - Failed to synthesize node: " + node_name, 34847 | exprtk_error_location)); 34848 | 34849 | details::free_node(*node_allocator_, result); 34850 | return error_node(); 34851 | 34852 | #undef vector_ops 34853 | } 34854 | 34855 | inline expression_node_ptr synthesize_swap_expression(expression_node_ptr (&branch)[2]) 34856 | { 34857 | const bool v0_is_ivar = details::is_ivariable_node(branch[0]); 34858 | const bool v1_is_ivar = details::is_ivariable_node(branch[1]); 34859 | 34860 | const bool v0_is_ivec = details::is_ivector_node (branch[0]); 34861 | const bool v1_is_ivec = details::is_ivector_node (branch[1]); 34862 | 34863 | #ifndef exprtk_disable_string_capabilities 34864 | const bool v0_is_str = details::is_generally_string_node(branch[0]); 34865 | const bool v1_is_str = details::is_generally_string_node(branch[1]); 34866 | #endif 34867 | 34868 | expression_node_ptr result = error_node(); 34869 | std::string node_name = "Unknown" 34870 | 34871 | if (v0_is_ivar && v1_is_ivar) 34872 | { 34873 | typedef details::variable_node<T>* variable_node_ptr; 34874 | 34875 | variable_node_ptr v0 = variable_node_ptr(0); 34876 | variable_node_ptr v1 = variable_node_ptr(0); 34877 | 34878 | if ( 34879 | (0 != (v0 = dynamic_cast<variable_node_ptr>(branch[0]))) && 34880 | (0 != (v1 = dynamic_cast<variable_node_ptr>(branch[1]))) 34881 | ) 34882 | { 34883 | result = node_allocator_->allocate<details::swap_node<T> >(v0,v1); 34884 | node_name = "swap_node" 34885 | } 34886 | else 34887 | { 34888 | result = node_allocator_->allocate<details::swap_generic_node<T> >(branch[0],branch[1]); 34889 | node_name = "swap_generic_node" 34890 | } 34891 | } 34892 | else if (v0_is_ivec && v1_is_ivec) 34893 | { 34894 | result = node_allocator_->allocate<details::swap_vecvec_node<T> >(branch[0],branch[1]); 34895 | node_name = "swap_vecvec_node" 34896 | } 34897 | #ifndef exprtk_disable_string_capabilities 34898 | else if (v0_is_str && v1_is_str) 34899 | { 34900 | if (is_string_node(branch[0]) && is_string_node(branch[1])) 34901 | { 34902 | result = node_allocator_->allocate<details::swap_string_node<T> > 34903 | (branch[0], branch[1]); 34904 | node_name = "swap_string_node" 34905 | } 34906 | else 34907 | { 34908 | result = node_allocator_->allocate<details::swap_genstrings_node<T> > 34909 | (branch[0], branch[1]); 34910 | node_name = "swap_genstrings_node" 34911 | } 34912 | } 34913 | #endif 34914 | else 34915 | { 34916 | parser_->set_synthesis_error("Only variables, strings, vectors or vector elements can be swapped"); 34917 | return error_node(); 34918 | } 34919 | 34920 | if (result && result->valid()) 34921 | { 34922 | parser_->state_.activate_side_effect("synthesize_swap_expression()"); 34923 | return result; 34924 | } 34925 | 34926 | parser_->set_error(parser_error::make_error( 34927 | parser_error::e_synthesis, 34928 | token_t(), 34929 | "ERR280 - Failed to synthesize node: " + node_name, 34930 | exprtk_error_location)); 34931 | 34932 | details::free_node(*node_allocator_, result); 34933 | return error_node(); 34934 | } 34935 | 34936 | #ifndef exprtk_disable_sc_andor 34937 | inline expression_node_ptr synthesize_shortcircuit_expression(const details::operator_type& operation, expression_node_ptr (&branch)[2]) 34938 | { 34939 | expression_node_ptr result = error_node(); 34940 | 34941 | if (details::is_constant_node(branch[0])) 34942 | { 34943 | if ( 34944 | (details::e_scand == operation) && 34945 | std::equal_to<T>()(T(0),branch[0]->value()) 34946 | ) 34947 | result = node_allocator_->allocate_c<literal_node_t>(T(0)); 34948 | else if ( 34949 | (details::e_scor == operation) && 34950 | std::not_equal_to<T>()(T(0),branch[0]->value()) 34951 | ) 34952 | result = node_allocator_->allocate_c<literal_node_t>(T(1)); 34953 | } 34954 | 34955 | if (details::is_constant_node(branch[1]) && (0 == result)) 34956 | { 34957 | if ( 34958 | (details::e_scand == operation) && 34959 | std::equal_to<T>()(T(0),branch[1]->value()) 34960 | ) 34961 | result = node_allocator_->allocate_c<literal_node_t>(T(0)); 34962 | else if ( 34963 | (details::e_scor == operation) && 34964 | std::not_equal_to<T>()(T(0),branch[1]->value()) 34965 | ) 34966 | result = node_allocator_->allocate_c<literal_node_t>(T(1)); 34967 | } 34968 | 34969 | if (result) 34970 | { 34971 | details::free_node(*node_allocator_, branch[0]); 34972 | details::free_node(*node_allocator_, branch[1]); 34973 | 34974 | return result; 34975 | } 34976 | else if (details::e_scand == operation) 34977 | { 34978 | return synthesize_expression<scand_node_t,2>(operation, branch); 34979 | } 34980 | else if (details::e_scor == operation) 34981 | { 34982 | return synthesize_expression<scor_node_t,2>(operation, branch); 34983 | } 34984 | else 34985 | return error_node(); 34986 | } 34987 | #else 34988 | inline expression_node_ptr synthesize_shortcircuit_expression(const details::operator_type&, expression_node_ptr (&)[2]) 34989 | { 34990 | return error_node(); 34991 | } 34992 | #endif 34993 | 34994 | #define basic_opr_switch_statements \ 34995 | case_stmt(details::e_add , details::add_op) \ 34996 | case_stmt(details::e_sub , details::sub_op) \ 34997 | case_stmt(details::e_mul , details::mul_op) \ 34998 | case_stmt(details::e_div , details::div_op) \ 34999 | case_stmt(details::e_mod , details::mod_op) \ 35000 | case_stmt(details::e_pow , details::pow_op) \ 35001 | 35002 | #define extended_opr_switch_statements \ 35003 | case_stmt(details::e_lt , details::lt_op ) \ 35004 | case_stmt(details::e_lte , details::lte_op ) \ 35005 | case_stmt(details::e_gt , details::gt_op ) \ 35006 | case_stmt(details::e_gte , details::gte_op ) \ 35007 | case_stmt(details::e_eq , details::eq_op ) \ 35008 | case_stmt(details::e_ne , details::ne_op ) \ 35009 | case_stmt(details::e_and , details::and_op ) \ 35010 | case_stmt(details::e_nand , details::nand_op) \ 35011 | case_stmt(details::e_or , details::or_op ) \ 35012 | case_stmt(details::e_nor , details::nor_op ) \ 35013 | case_stmt(details::e_xor , details::xor_op ) \ 35014 | case_stmt(details::e_xnor , details::xnor_op) \ 35015 | 35016 | #ifndef exprtk_disable_cardinal_pow_optimisation 35017 | template <typename TType, template <typename, typename> class IPowNode> 35018 | inline expression_node_ptr cardinal_pow_optimisation_impl(const TType& v, const unsigned int& p) 35019 | { 35020 | switch (p) 35021 | { 35022 | #define case_stmt(cp) \ 35023 | case cp : return node_allocator_-> \ 35024 | allocate<IPowNode<T,details::numeric::fast_exp<T,cp> > >(v); \ 35025 | 35026 | case_stmt( 1) case_stmt( 2) case_stmt( 3) case_stmt( 4) 35027 | case_stmt( 5) case_stmt( 6) case_stmt( 7) case_stmt( 8) 35028 | case_stmt( 9) case_stmt(10) case_stmt(11) case_stmt(12) 35029 | case_stmt(13) case_stmt(14) case_stmt(15) case_stmt(16) 35030 | case_stmt(17) case_stmt(18) case_stmt(19) case_stmt(20) 35031 | case_stmt(21) case_stmt(22) case_stmt(23) case_stmt(24) 35032 | case_stmt(25) case_stmt(26) case_stmt(27) case_stmt(28) 35033 | case_stmt(29) case_stmt(30) case_stmt(31) case_stmt(32) 35034 | case_stmt(33) case_stmt(34) case_stmt(35) case_stmt(36) 35035 | case_stmt(37) case_stmt(38) case_stmt(39) case_stmt(40) 35036 | case_stmt(41) case_stmt(42) case_stmt(43) case_stmt(44) 35037 | case_stmt(45) case_stmt(46) case_stmt(47) case_stmt(48) 35038 | case_stmt(49) case_stmt(50) case_stmt(51) case_stmt(52) 35039 | case_stmt(53) case_stmt(54) case_stmt(55) case_stmt(56) 35040 | case_stmt(57) case_stmt(58) case_stmt(59) case_stmt(60) 35041 | #undef case_stmt 35042 | default : return error_node(); 35043 | } 35044 | } 35045 | 35046 | inline expression_node_ptr cardinal_pow_optimisation(const T& v, const T& c) 35047 | { 35048 | const bool not_recipricol = (c >= T(0)); 35049 | const unsigned int p = static_cast<unsigned int>(details::numeric::to_int32(details::numeric::abs(c))); 35050 | 35051 | if (0 == p) 35052 | return node_allocator_->allocate_c<literal_node_t>(T(1)); 35053 | else if (std::equal_to<T>()(T(2),c)) 35054 | { 35055 | return node_allocator_-> 35056 | template allocate_rr<typename details::vov_node<Type,details::mul_op<Type> > >(v,v); 35057 | } 35058 | else 35059 | { 35060 | if (not_recipricol) 35061 | return cardinal_pow_optimisation_impl<T,details::ipow_node>(v,p); 35062 | else 35063 | return cardinal_pow_optimisation_impl<T,details::ipowinv_node>(v,p); 35064 | } 35065 | } 35066 | 35067 | inline bool cardinal_pow_optimisable(const details::operator_type& operation, const T& c) const 35068 | { 35069 | return (details::e_pow == operation) && (details::numeric::abs(c) <= T(60)) && details::numeric::is_integer(c); 35070 | } 35071 | 35072 | inline expression_node_ptr cardinal_pow_optimisation(expression_node_ptr (&branch)[2]) 35073 | { 35074 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 35075 | const bool not_recipricol = (c >= T(0)); 35076 | const unsigned int p = static_cast<unsigned int>(details::numeric::to_int32(details::numeric::abs(c))); 35077 | 35078 | node_allocator_->free(branch[1]); 35079 | 35080 | if (0 == p) 35081 | { 35082 | details::free_all_nodes(*node_allocator_, branch); 35083 | 35084 | return node_allocator_->allocate_c<literal_node_t>(T(1)); 35085 | } 35086 | else if (not_recipricol) 35087 | return cardinal_pow_optimisation_impl<expression_node_ptr,details::bipow_node>(branch[0],p); 35088 | else 35089 | return cardinal_pow_optimisation_impl<expression_node_ptr,details::bipowinv_node>(branch[0],p); 35090 | } 35091 | #else 35092 | inline expression_node_ptr cardinal_pow_optimisation(T&, const T&) 35093 | { 35094 | return error_node(); 35095 | } 35096 | 35097 | inline bool cardinal_pow_optimisable(const details::operator_type&, const T&) 35098 | { 35099 | return false; 35100 | } 35101 | 35102 | inline expression_node_ptr cardinal_pow_optimisation(expression_node_ptr(&)[2]) 35103 | { 35104 | return error_node(); 35105 | } 35106 | #endif 35107 | 35108 | struct synthesize_binary_ext_expression 35109 | { 35110 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35111 | const details::operator_type& operation, 35112 | expression_node_ptr (&branch)[2]) 35113 | { 35114 | const bool left_neg = is_neg_unary_node(branch[0]); 35115 | const bool right_neg = is_neg_unary_node(branch[1]); 35116 | 35117 | if (left_neg && right_neg) 35118 | { 35119 | if ( 35120 | (details::e_add == operation) || 35121 | (details::e_sub == operation) || 35122 | (details::e_mul == operation) || 35123 | (details::e_div == operation) 35124 | ) 35125 | { 35126 | if ( 35127 | !expr_gen.parser_->simplify_unary_negation_branch(branch[0]) || 35128 | !expr_gen.parser_->simplify_unary_negation_branch(branch[1]) 35129 | ) 35130 | { 35131 | details::free_all_nodes(*expr_gen.node_allocator_,branch); 35132 | 35133 | return error_node(); 35134 | } 35135 | } 35136 | 35137 | switch (operation) 35138 | { 35139 | // -f(x + 1) + -g(y + 1) --> -(f(x + 1) + g(y + 1)) 35140 | case details::e_add : return expr_gen(details::e_neg, 35141 | expr_gen.node_allocator_-> 35142 | template allocate<typename details::binary_ext_node<Type,details::add_op<Type> > > 35143 | (branch[0],branch[1])); 35144 | 35145 | // -f(x + 1) - -g(y + 1) --> g(y + 1) - f(x + 1) 35146 | case details::e_sub : return expr_gen.node_allocator_-> 35147 | template allocate<typename details::binary_ext_node<Type,details::sub_op<Type> > > 35148 | (branch[1],branch[0]); 35149 | 35150 | default : break; 35151 | } 35152 | } 35153 | else if (left_neg && !right_neg) 35154 | { 35155 | if ( 35156 | (details::e_add == operation) || 35157 | (details::e_sub == operation) || 35158 | (details::e_mul == operation) || 35159 | (details::e_div == operation) 35160 | ) 35161 | { 35162 | if (!expr_gen.parser_->simplify_unary_negation_branch(branch[0])) 35163 | { 35164 | details::free_all_nodes(*expr_gen.node_allocator_,branch); 35165 | 35166 | return error_node(); 35167 | } 35168 | 35169 | switch (operation) 35170 | { 35171 | // -f(x + 1) + g(y + 1) --> g(y + 1) - f(x + 1) 35172 | case details::e_add : return expr_gen.node_allocator_-> 35173 | template allocate<typename details::binary_ext_node<Type,details::sub_op<Type> > > 35174 | (branch[1], branch[0]); 35175 | 35176 | // -f(x + 1) - g(y + 1) --> -(f(x + 1) + g(y + 1)) 35177 | case details::e_sub : return expr_gen(details::e_neg, 35178 | expr_gen.node_allocator_-> 35179 | template allocate<typename details::binary_ext_node<Type,details::add_op<Type> > > 35180 | (branch[0], branch[1])); 35181 | 35182 | // -f(x + 1) * g(y + 1) --> -(f(x + 1) * g(y + 1)) 35183 | case details::e_mul : return expr_gen(details::e_neg, 35184 | expr_gen.node_allocator_-> 35185 | template allocate<typename details::binary_ext_node<Type,details::mul_op<Type> > > 35186 | (branch[0], branch[1])); 35187 | 35188 | // -f(x + 1) / g(y + 1) --> -(f(x + 1) / g(y + 1)) 35189 | case details::e_div : return expr_gen(details::e_neg, 35190 | expr_gen.node_allocator_-> 35191 | template allocate<typename details::binary_ext_node<Type,details::div_op<Type> > > 35192 | (branch[0], branch[1])); 35193 | 35194 | default : return error_node(); 35195 | } 35196 | } 35197 | } 35198 | else if (!left_neg && right_neg) 35199 | { 35200 | if ( 35201 | (details::e_add == operation) || 35202 | (details::e_sub == operation) || 35203 | (details::e_mul == operation) || 35204 | (details::e_div == operation) 35205 | ) 35206 | { 35207 | if (!expr_gen.parser_->simplify_unary_negation_branch(branch[1])) 35208 | { 35209 | details::free_all_nodes(*expr_gen.node_allocator_,branch); 35210 | 35211 | return error_node(); 35212 | } 35213 | 35214 | switch (operation) 35215 | { 35216 | // f(x + 1) + -g(y + 1) --> f(x + 1) - g(y + 1) 35217 | case details::e_add : return expr_gen.node_allocator_-> 35218 | template allocate<typename details::binary_ext_node<Type,details::sub_op<Type> > > 35219 | (branch[0], branch[1]); 35220 | 35221 | // f(x + 1) - - g(y + 1) --> f(x + 1) + g(y + 1) 35222 | case details::e_sub : return expr_gen.node_allocator_-> 35223 | template allocate<typename details::binary_ext_node<Type,details::add_op<Type> > > 35224 | (branch[0], branch[1]); 35225 | 35226 | // f(x + 1) * -g(y + 1) --> -(f(x + 1) * g(y + 1)) 35227 | case details::e_mul : return expr_gen(details::e_neg, 35228 | expr_gen.node_allocator_-> 35229 | template allocate<typename details::binary_ext_node<Type,details::mul_op<Type> > > 35230 | (branch[0], branch[1])); 35231 | 35232 | // f(x + 1) / -g(y + 1) --> -(f(x + 1) / g(y + 1)) 35233 | case details::e_div : return expr_gen(details::e_neg, 35234 | expr_gen.node_allocator_-> 35235 | template allocate<typename details::binary_ext_node<Type,details::div_op<Type> > > 35236 | (branch[0], branch[1])); 35237 | 35238 | default : return error_node(); 35239 | } 35240 | } 35241 | } 35242 | 35243 | switch (operation) 35244 | { 35245 | #define case_stmt(op0, op1) \ 35246 | case op0 : return expr_gen.node_allocator_-> \ 35247 | template allocate<typename details::binary_ext_node<Type,op1<Type> > > \ 35248 | (branch[0], branch[1]); \ 35249 | 35250 | basic_opr_switch_statements 35251 | extended_opr_switch_statements 35252 | #undef case_stmt 35253 | default : return error_node(); 35254 | } 35255 | } 35256 | }; 35257 | 35258 | struct synthesize_vob_expression 35259 | { 35260 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35261 | const details::operator_type& operation, 35262 | expression_node_ptr (&branch)[2]) 35263 | { 35264 | const Type& v = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 35265 | 35266 | #ifndef exprtk_disable_enhanced_features 35267 | if (details::is_sf3ext_node(branch[1])) 35268 | { 35269 | expression_node_ptr result = error_node(); 35270 | 35271 | const bool synthesis_result = 35272 | synthesize_sf4ext_expression::template compile_right<vtype> 35273 | (expr_gen, v, operation, branch[1], result); 35274 | 35275 | if (synthesis_result) 35276 | { 35277 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35278 | return result; 35279 | } 35280 | } 35281 | #endif 35282 | 35283 | if ( 35284 | (details::e_mul == operation) || 35285 | (details::e_div == operation) 35286 | ) 35287 | { 35288 | if (details::is_uv_node(branch[1])) 35289 | { 35290 | typedef details::uv_base_node<Type>* uvbn_ptr_t; 35291 | 35292 | details::operator_type o = static_cast<uvbn_ptr_t>(branch[1])->operation(); 35293 | 35294 | if (details::e_neg == o) 35295 | { 35296 | const Type& v1 = static_cast<uvbn_ptr_t>(branch[1])->v(); 35297 | 35298 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35299 | 35300 | switch (operation) 35301 | { 35302 | case details::e_mul : return expr_gen(details::e_neg, 35303 | expr_gen.node_allocator_-> 35304 | template allocate_rr<typename details:: 35305 | vov_node<Type,details::mul_op<Type> > >(v,v1)); 35306 | 35307 | case details::e_div : return expr_gen(details::e_neg, 35308 | expr_gen.node_allocator_-> 35309 | template allocate_rr<typename details:: 35310 | vov_node<Type,details::div_op<Type> > >(v,v1)); 35311 | 35312 | default : break; 35313 | } 35314 | } 35315 | } 35316 | } 35317 | 35318 | switch (operation) 35319 | { 35320 | #define case_stmt(op0, op1) \ 35321 | case op0 : return expr_gen.node_allocator_-> \ 35322 | template allocate_rc<typename details::vob_node<Type,op1<Type> > > \ 35323 | (v, branch[1]); \ 35324 | 35325 | basic_opr_switch_statements 35326 | extended_opr_switch_statements 35327 | #undef case_stmt 35328 | default : return error_node(); 35329 | } 35330 | } 35331 | }; 35332 | 35333 | struct synthesize_bov_expression 35334 | { 35335 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35336 | const details::operator_type& operation, 35337 | expression_node_ptr (&branch)[2]) 35338 | { 35339 | const Type& v = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 35340 | 35341 | #ifndef exprtk_disable_enhanced_features 35342 | if (details::is_sf3ext_node(branch[0])) 35343 | { 35344 | expression_node_ptr result = error_node(); 35345 | 35346 | const bool synthesis_result = 35347 | synthesize_sf4ext_expression::template compile_left<vtype> 35348 | (expr_gen, v, operation, branch[0], result); 35349 | 35350 | if (synthesis_result) 35351 | { 35352 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35353 | 35354 | return result; 35355 | } 35356 | } 35357 | #endif 35358 | 35359 | if ( 35360 | (details::e_add == operation) || 35361 | (details::e_sub == operation) || 35362 | (details::e_mul == operation) || 35363 | (details::e_div == operation) 35364 | ) 35365 | { 35366 | if (details::is_uv_node(branch[0])) 35367 | { 35368 | typedef details::uv_base_node<Type>* uvbn_ptr_t; 35369 | 35370 | details::operator_type o = static_cast<uvbn_ptr_t>(branch[0])->operation(); 35371 | 35372 | if (details::e_neg == o) 35373 | { 35374 | const Type& v0 = static_cast<uvbn_ptr_t>(branch[0])->v(); 35375 | 35376 | details::free_node(*expr_gen.node_allocator_,branch[0]); 35377 | 35378 | switch (operation) 35379 | { 35380 | case details::e_add : return expr_gen.node_allocator_-> 35381 | template allocate_rr<typename details:: 35382 | vov_node<Type,details::sub_op<Type> > >(v,v0); 35383 | 35384 | case details::e_sub : return expr_gen(details::e_neg, 35385 | expr_gen.node_allocator_-> 35386 | template allocate_rr<typename details:: 35387 | vov_node<Type,details::add_op<Type> > >(v0,v)); 35388 | 35389 | case details::e_mul : return expr_gen(details::e_neg, 35390 | expr_gen.node_allocator_-> 35391 | template allocate_rr<typename details:: 35392 | vov_node<Type,details::mul_op<Type> > >(v0,v)); 35393 | 35394 | case details::e_div : return expr_gen(details::e_neg, 35395 | expr_gen.node_allocator_-> 35396 | template allocate_rr<typename details:: 35397 | vov_node<Type,details::div_op<Type> > >(v0,v)); 35398 | default : break; 35399 | } 35400 | } 35401 | } 35402 | } 35403 | 35404 | switch (operation) 35405 | { 35406 | #define case_stmt(op0, op1) \ 35407 | case op0 : return expr_gen.node_allocator_-> \ 35408 | template allocate_cr<typename details::bov_node<Type,op1<Type> > > \ 35409 | (branch[0], v); \ 35410 | 35411 | basic_opr_switch_statements 35412 | extended_opr_switch_statements 35413 | #undef case_stmt 35414 | default : return error_node(); 35415 | } 35416 | } 35417 | }; 35418 | 35419 | struct synthesize_cob_expression 35420 | { 35421 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35422 | const details::operator_type& operation, 35423 | expression_node_ptr (&branch)[2]) 35424 | { 35425 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 35426 | 35427 | details::free_node(*expr_gen.node_allocator_,branch[0]); 35428 | 35429 | if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 35430 | { 35431 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35432 | 35433 | return expr_gen(T(0)); 35434 | } 35435 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 35436 | { 35437 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35438 | 35439 | return expr_gen(T(0)); 35440 | } 35441 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 35442 | return branch[1]; 35443 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 35444 | return branch[1]; 35445 | 35446 | if (details::is_cob_node(branch[1])) 35447 | { 35448 | // Simplify expressions of the form: 35449 | // 1. (1 * (2 * (3 * (4 * (5 * (6 * (7 * (8 * (9 + x))))))))) --> 40320 * (9 + x) 35450 | // 2. (1 + (2 + (3 + (4 + (5 + (6 + (7 + (8 + (9 + x))))))))) --> 45 + x 35451 | if ( 35452 | (details::e_mul == operation) || 35453 | (details::e_add == operation) 35454 | ) 35455 | { 35456 | details::cob_base_node<Type>* cobnode = static_cast<details::cob_base_node<Type>*>(branch[1]); 35457 | 35458 | if (operation == cobnode->operation()) 35459 | { 35460 | switch (operation) 35461 | { 35462 | case details::e_add : cobnode->set_c(c + cobnode->c()); break; 35463 | case details::e_mul : cobnode->set_c(c * cobnode->c()); break; 35464 | default : return error_node(); 35465 | } 35466 | 35467 | return cobnode; 35468 | } 35469 | } 35470 | 35471 | if (operation == details::e_mul) 35472 | { 35473 | details::cob_base_node<Type>* cobnode = static_cast<details::cob_base_node<Type>*>(branch[1]); 35474 | details::operator_type cob_opr = cobnode->operation(); 35475 | 35476 | if ( 35477 | (details::e_div == cob_opr) || 35478 | (details::e_mul == cob_opr) 35479 | ) 35480 | { 35481 | switch (cob_opr) 35482 | { 35483 | case details::e_div : cobnode->set_c(c * cobnode->c()); break; 35484 | case details::e_mul : cobnode->set_c(cobnode->c() / c); break; 35485 | default : return error_node(); 35486 | } 35487 | 35488 | return cobnode; 35489 | } 35490 | } 35491 | else if (operation == details::e_div) 35492 | { 35493 | details::cob_base_node<Type>* cobnode = static_cast<details::cob_base_node<Type>*>(branch[1]); 35494 | details::operator_type cob_opr = cobnode->operation(); 35495 | 35496 | if ( 35497 | (details::e_div == cob_opr) || 35498 | (details::e_mul == cob_opr) 35499 | ) 35500 | { 35501 | details::expression_node<Type>* new_cobnode = error_node(); 35502 | 35503 | switch (cob_opr) 35504 | { 35505 | case details::e_div : new_cobnode = expr_gen.node_allocator_-> 35506 | template allocate_tt<typename details::cob_node<Type,details::mul_op<Type> > > 35507 | (c / cobnode->c(), cobnode->move_branch(0)); 35508 | break; 35509 | 35510 | case details::e_mul : new_cobnode = expr_gen.node_allocator_-> 35511 | template allocate_tt<typename details::cob_node<Type,details::div_op<Type> > > 35512 | (c / cobnode->c(), cobnode->move_branch(0)); 35513 | break; 35514 | 35515 | default : return error_node(); 35516 | } 35517 | 35518 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35519 | 35520 | return new_cobnode; 35521 | } 35522 | } 35523 | } 35524 | #ifndef exprtk_disable_enhanced_features 35525 | else if (details::is_sf3ext_node(branch[1])) 35526 | { 35527 | expression_node_ptr result = error_node(); 35528 | 35529 | const bool synthesis_result = 35530 | synthesize_sf4ext_expression::template compile_right<ctype> 35531 | (expr_gen, c, operation, branch[1], result); 35532 | 35533 | if (synthesis_result) 35534 | { 35535 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35536 | 35537 | return result; 35538 | } 35539 | } 35540 | #endif 35541 | 35542 | switch (operation) 35543 | { 35544 | #define case_stmt(op0, op1) \ 35545 | case op0 : return expr_gen.node_allocator_-> \ 35546 | template allocate_tt<typename details::cob_node<Type,op1<Type> > > \ 35547 | (c, branch[1]); \ 35548 | 35549 | basic_opr_switch_statements 35550 | extended_opr_switch_statements 35551 | #undef case_stmt 35552 | default : return error_node(); 35553 | } 35554 | } 35555 | }; 35556 | 35557 | struct synthesize_boc_expression 35558 | { 35559 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35560 | const details::operator_type& operation, 35561 | expression_node_ptr (&branch)[2]) 35562 | { 35563 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 35564 | 35565 | details::free_node(*(expr_gen.node_allocator_), branch[1]); 35566 | 35567 | if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 35568 | { 35569 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35570 | 35571 | return expr_gen(T(0)); 35572 | } 35573 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 35574 | { 35575 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35576 | 35577 | return expr_gen(std::numeric_limits<T>::quiet_NaN()); 35578 | } 35579 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 35580 | return branch[0]; 35581 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 35582 | return branch[0]; 35583 | 35584 | if (details::is_boc_node(branch[0])) 35585 | { 35586 | // Simplify expressions of the form: 35587 | // 1. (((((((((x + 9) * 8) * 7) * 6) * 5) * 4) * 3) * 2) * 1) --> (x + 9) * 40320 35588 | // 2. (((((((((x + 9) + 8) + 7) + 6) + 5) + 4) + 3) + 2) + 1) --> x + 45 35589 | if ( 35590 | (details::e_mul == operation) || 35591 | (details::e_add == operation) 35592 | ) 35593 | { 35594 | details::boc_base_node<Type>* bocnode = static_cast<details::boc_base_node<Type>*>(branch[0]); 35595 | 35596 | if (operation == bocnode->operation()) 35597 | { 35598 | switch (operation) 35599 | { 35600 | case details::e_add : bocnode->set_c(c + bocnode->c()); break; 35601 | case details::e_mul : bocnode->set_c(c * bocnode->c()); break; 35602 | default : return error_node(); 35603 | } 35604 | 35605 | return bocnode; 35606 | } 35607 | } 35608 | else if (operation == details::e_div) 35609 | { 35610 | details::boc_base_node<Type>* bocnode = static_cast<details::boc_base_node<Type>*>(branch[0]); 35611 | details::operator_type boc_opr = bocnode->operation(); 35612 | 35613 | if ( 35614 | (details::e_div == boc_opr) || 35615 | (details::e_mul == boc_opr) 35616 | ) 35617 | { 35618 | switch (boc_opr) 35619 | { 35620 | case details::e_div : bocnode->set_c(c * bocnode->c()); break; 35621 | case details::e_mul : bocnode->set_c(bocnode->c() / c); break; 35622 | default : return error_node(); 35623 | } 35624 | 35625 | return bocnode; 35626 | } 35627 | } 35628 | else if (operation == details::e_pow) 35629 | { 35630 | // (v ^ c0) ^ c1 --> v ^(c0 * c1) 35631 | details::boc_base_node<Type>* bocnode = static_cast<details::boc_base_node<Type>*>(branch[0]); 35632 | details::operator_type boc_opr = bocnode->operation(); 35633 | 35634 | if (details::e_pow == boc_opr) 35635 | { 35636 | bocnode->set_c(bocnode->c() * c); 35637 | 35638 | return bocnode; 35639 | } 35640 | } 35641 | } 35642 | 35643 | #ifndef exprtk_disable_enhanced_features 35644 | if (details::is_sf3ext_node(branch[0])) 35645 | { 35646 | expression_node_ptr result = error_node(); 35647 | 35648 | const bool synthesis_result = 35649 | synthesize_sf4ext_expression::template compile_left<ctype> 35650 | (expr_gen, c, operation, branch[0], result); 35651 | 35652 | if (synthesis_result) 35653 | { 35654 | free_node(*expr_gen.node_allocator_, branch[0]); 35655 | 35656 | return result; 35657 | } 35658 | } 35659 | #endif 35660 | 35661 | switch (operation) 35662 | { 35663 | #define case_stmt(op0, op1) \ 35664 | case op0 : return expr_gen.node_allocator_-> \ 35665 | template allocate_cr<typename details::boc_node<Type,op1<Type> > > \ 35666 | (branch[0], c); \ 35667 | 35668 | basic_opr_switch_statements 35669 | extended_opr_switch_statements 35670 | #undef case_stmt 35671 | default : return error_node(); 35672 | } 35673 | } 35674 | }; 35675 | 35676 | struct synthesize_cocob_expression 35677 | { 35678 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35679 | const details::operator_type& operation, 35680 | expression_node_ptr (&branch)[2]) 35681 | { 35682 | expression_node_ptr result = error_node(); 35683 | 35684 | // (cob) o c --> cob 35685 | if (details::is_cob_node(branch[0])) 35686 | { 35687 | details::cob_base_node<Type>* cobnode = static_cast<details::cob_base_node<Type>*>(branch[0]); 35688 | 35689 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 35690 | 35691 | if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 35692 | { 35693 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35694 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35695 | 35696 | return expr_gen(T(0)); 35697 | } 35698 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 35699 | { 35700 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35701 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35702 | 35703 | return expr_gen(T(std::numeric_limits<T>::quiet_NaN())); 35704 | } 35705 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 35706 | { 35707 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35708 | 35709 | return branch[0]; 35710 | } 35711 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 35712 | { 35713 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35714 | 35715 | return branch[0]; 35716 | } 35717 | else if (std::equal_to<T>()(T(1),c) && (details::e_div == operation)) 35718 | { 35719 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35720 | 35721 | return branch[0]; 35722 | } 35723 | 35724 | const bool op_addsub = (details::e_add == cobnode->operation()) || 35725 | (details::e_sub == cobnode->operation()) ; 35726 | 35727 | if (op_addsub) 35728 | { 35729 | switch (operation) 35730 | { 35731 | case details::e_add : cobnode->set_c(cobnode->c() + c); break; 35732 | case details::e_sub : cobnode->set_c(cobnode->c() - c); break; 35733 | default : return error_node(); 35734 | } 35735 | 35736 | result = cobnode; 35737 | } 35738 | else if (details::e_mul == cobnode->operation()) 35739 | { 35740 | switch (operation) 35741 | { 35742 | case details::e_mul : cobnode->set_c(cobnode->c() * c); break; 35743 | case details::e_div : cobnode->set_c(cobnode->c() / c); break; 35744 | default : return error_node(); 35745 | } 35746 | 35747 | result = cobnode; 35748 | } 35749 | else if (details::e_div == cobnode->operation()) 35750 | { 35751 | if (details::e_mul == operation) 35752 | { 35753 | cobnode->set_c(cobnode->c() * c); 35754 | result = cobnode; 35755 | } 35756 | else if (details::e_div == operation) 35757 | { 35758 | result = expr_gen.node_allocator_-> 35759 | template allocate_tt<typename details::cob_node<Type,details::div_op<Type> > > 35760 | (cobnode->c() / c, cobnode->move_branch(0)); 35761 | 35762 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35763 | } 35764 | } 35765 | 35766 | if (result) 35767 | { 35768 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35769 | } 35770 | } 35771 | 35772 | // c o (cob) --> cob 35773 | else if (details::is_cob_node(branch[1])) 35774 | { 35775 | details::cob_base_node<Type>* cobnode = static_cast<details::cob_base_node<Type>*>(branch[1]); 35776 | 35777 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 35778 | 35779 | if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 35780 | { 35781 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35782 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35783 | 35784 | return expr_gen(T(0)); 35785 | } 35786 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 35787 | { 35788 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35789 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35790 | 35791 | return expr_gen(T(0)); 35792 | } 35793 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 35794 | { 35795 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35796 | 35797 | return branch[1]; 35798 | } 35799 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 35800 | { 35801 | details::free_node(*expr_gen.node_allocator_, branch[0]); 35802 | 35803 | return branch[1]; 35804 | } 35805 | 35806 | if (details::e_add == cobnode->operation()) 35807 | { 35808 | if (details::e_add == operation) 35809 | { 35810 | cobnode->set_c(c + cobnode->c()); 35811 | result = cobnode; 35812 | } 35813 | else if (details::e_sub == operation) 35814 | { 35815 | result = expr_gen.node_allocator_-> 35816 | template allocate_tt<typename details::cob_node<Type,details::sub_op<Type> > > 35817 | (c - cobnode->c(), cobnode->move_branch(0)); 35818 | 35819 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35820 | } 35821 | } 35822 | else if (details::e_sub == cobnode->operation()) 35823 | { 35824 | if (details::e_add == operation) 35825 | { 35826 | cobnode->set_c(c + cobnode->c()); 35827 | result = cobnode; 35828 | } 35829 | else if (details::e_sub == operation) 35830 | { 35831 | result = expr_gen.node_allocator_-> 35832 | template allocate_tt<typename details::cob_node<Type,details::add_op<Type> > > 35833 | (c - cobnode->c(), cobnode->move_branch(0)); 35834 | 35835 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35836 | } 35837 | } 35838 | else if (details::e_mul == cobnode->operation()) 35839 | { 35840 | if (details::e_mul == operation) 35841 | { 35842 | cobnode->set_c(c * cobnode->c()); 35843 | result = cobnode; 35844 | } 35845 | else if (details::e_div == operation) 35846 | { 35847 | result = expr_gen.node_allocator_-> 35848 | template allocate_tt<typename details::cob_node<Type,details::div_op<Type> > > 35849 | (c / cobnode->c(), cobnode->move_branch(0)); 35850 | 35851 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35852 | } 35853 | } 35854 | else if (details::e_div == cobnode->operation()) 35855 | { 35856 | if (details::e_mul == operation) 35857 | { 35858 | cobnode->set_c(c * cobnode->c()); 35859 | result = cobnode; 35860 | } 35861 | else if (details::e_div == operation) 35862 | { 35863 | result = expr_gen.node_allocator_-> 35864 | template allocate_tt<typename details::cob_node<Type,details::mul_op<Type> > > 35865 | (c / cobnode->c(), cobnode->move_branch(0)); 35866 | 35867 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35868 | } 35869 | } 35870 | 35871 | if (result) 35872 | { 35873 | details::free_node(*expr_gen.node_allocator_,branch[0]); 35874 | } 35875 | } 35876 | 35877 | return result; 35878 | } 35879 | }; 35880 | 35881 | struct synthesize_coboc_expression 35882 | { 35883 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 35884 | const details::operator_type& operation, 35885 | expression_node_ptr (&branch)[2]) 35886 | { 35887 | expression_node_ptr result = error_node(); 35888 | 35889 | // (boc) o c --> boc 35890 | if (details::is_boc_node(branch[0])) 35891 | { 35892 | details::boc_base_node<Type>* bocnode = static_cast<details::boc_base_node<Type>*>(branch[0]); 35893 | 35894 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 35895 | 35896 | if (details::e_add == bocnode->operation()) 35897 | { 35898 | switch (operation) 35899 | { 35900 | case details::e_add : bocnode->set_c(bocnode->c() + c); break; 35901 | case details::e_sub : bocnode->set_c(bocnode->c() - c); break; 35902 | default : return error_node(); 35903 | } 35904 | 35905 | result = bocnode; 35906 | } 35907 | else if (details::e_mul == bocnode->operation()) 35908 | { 35909 | switch (operation) 35910 | { 35911 | case details::e_mul : bocnode->set_c(bocnode->c() * c); break; 35912 | case details::e_div : bocnode->set_c(bocnode->c() / c); break; 35913 | default : return error_node(); 35914 | } 35915 | 35916 | result = bocnode; 35917 | } 35918 | else if (details::e_sub == bocnode->operation()) 35919 | { 35920 | if (details::e_add == operation) 35921 | { 35922 | result = expr_gen.node_allocator_-> 35923 | template allocate_tt<typename details::boc_node<Type,details::add_op<Type> > > 35924 | (bocnode->move_branch(0), c - bocnode->c()); 35925 | 35926 | details::free_node(*expr_gen.node_allocator_,branch[0]); 35927 | } 35928 | else if (details::e_sub == operation) 35929 | { 35930 | bocnode->set_c(bocnode->c() + c); 35931 | result = bocnode; 35932 | } 35933 | } 35934 | else if (details::e_div == bocnode->operation()) 35935 | { 35936 | switch (operation) 35937 | { 35938 | case details::e_div : bocnode->set_c(bocnode->c() * c); break; 35939 | case details::e_mul : bocnode->set_c(bocnode->c() / c); break; 35940 | default : return error_node(); 35941 | } 35942 | 35943 | result = bocnode; 35944 | } 35945 | 35946 | if (result) 35947 | { 35948 | details::free_node(*expr_gen.node_allocator_, branch[1]); 35949 | } 35950 | } 35951 | 35952 | // c o (boc) --> boc 35953 | else if (details::is_boc_node(branch[1])) 35954 | { 35955 | details::boc_base_node<Type>* bocnode = static_cast<details::boc_base_node<Type>*>(branch[1]); 35956 | 35957 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 35958 | 35959 | if (details::e_add == bocnode->operation()) 35960 | { 35961 | if (details::e_add == operation) 35962 | { 35963 | bocnode->set_c(c + bocnode->c()); 35964 | result = bocnode; 35965 | } 35966 | else if (details::e_sub == operation) 35967 | { 35968 | result = expr_gen.node_allocator_-> 35969 | template allocate_tt<typename details::cob_node<Type,details::sub_op<Type> > > 35970 | (c - bocnode->c(), bocnode->move_branch(0)); 35971 | 35972 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35973 | } 35974 | } 35975 | else if (details::e_sub == bocnode->operation()) 35976 | { 35977 | if (details::e_add == operation) 35978 | { 35979 | result = expr_gen.node_allocator_-> 35980 | template allocate_tt<typename details::boc_node<Type,details::add_op<Type> > > 35981 | (bocnode->move_branch(0), c - bocnode->c()); 35982 | 35983 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35984 | } 35985 | else if (details::e_sub == operation) 35986 | { 35987 | result = expr_gen.node_allocator_-> 35988 | template allocate_tt<typename details::cob_node<Type,details::sub_op<Type> > > 35989 | (c + bocnode->c(), bocnode->move_branch(0)); 35990 | 35991 | details::free_node(*expr_gen.node_allocator_,branch[1]); 35992 | } 35993 | } 35994 | else if (details::e_mul == bocnode->operation()) 35995 | { 35996 | if (details::e_mul == operation) 35997 | { 35998 | bocnode->set_c(c * bocnode->c()); 35999 | result = bocnode; 36000 | } 36001 | else if (details::e_div == operation) 36002 | { 36003 | result = expr_gen.node_allocator_-> 36004 | template allocate_tt<typename details::cob_node<Type,details::div_op<Type> > > 36005 | (c / bocnode->c(), bocnode->move_branch(0)); 36006 | 36007 | details::free_node(*expr_gen.node_allocator_,branch[1]); 36008 | } 36009 | } 36010 | else if (details::e_div == bocnode->operation()) 36011 | { 36012 | if (details::e_mul == operation) 36013 | { 36014 | bocnode->set_c(bocnode->c() / c); 36015 | result = bocnode; 36016 | } 36017 | else if (details::e_div == operation) 36018 | { 36019 | result = expr_gen.node_allocator_-> 36020 | template allocate_tt<typename details::cob_node<Type,details::div_op<Type> > > 36021 | (c * bocnode->c(), bocnode->move_branch(0)); 36022 | 36023 | details::free_node(*expr_gen.node_allocator_,branch[1]); 36024 | } 36025 | } 36026 | 36027 | if (result) 36028 | { 36029 | details::free_node(*expr_gen.node_allocator_,branch[0]); 36030 | } 36031 | } 36032 | 36033 | return result; 36034 | } 36035 | }; 36036 | 36037 | #ifndef exprtk_disable_enhanced_features 36038 | inline bool synthesize_expression(const details::operator_type& operation, 36039 | expression_node_ptr (&branch)[2], 36040 | expression_node_ptr& result) 36041 | { 36042 | result = error_node(); 36043 | 36044 | if (!operation_optimisable(operation)) 36045 | return false; 36046 | 36047 | const std::string node_id = branch_to_id(branch); 36048 | 36049 | const typename synthesize_map_t::iterator itr = synthesize_map_.find(node_id); 36050 | 36051 | if (synthesize_map_.end() != itr) 36052 | { 36053 | result = itr->second((*this), operation, branch); 36054 | 36055 | return true; 36056 | } 36057 | else 36058 | return false; 36059 | } 36060 | 36061 | struct synthesize_vov_expression 36062 | { 36063 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36064 | const details::operator_type& operation, 36065 | expression_node_ptr (&branch)[2]) 36066 | { 36067 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 36068 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 36069 | 36070 | switch (operation) 36071 | { 36072 | #define case_stmt(op0, op1) \ 36073 | case op0 : return expr_gen.node_allocator_-> \ 36074 | template allocate_rr<typename details::vov_node<Type,op1<Type> > > \ 36075 | (v1, v2); \ 36076 | 36077 | basic_opr_switch_statements 36078 | extended_opr_switch_statements 36079 | #undef case_stmt 36080 | default : return error_node(); 36081 | } 36082 | } 36083 | }; 36084 | 36085 | struct synthesize_cov_expression 36086 | { 36087 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36088 | const details::operator_type& operation, 36089 | expression_node_ptr (&branch)[2]) 36090 | { 36091 | const Type c = static_cast<details::literal_node<Type>*> (branch[0])->value(); 36092 | const Type& v = static_cast<details::variable_node<Type>*>(branch[1])->ref (); 36093 | 36094 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36095 | 36096 | if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 36097 | return expr_gen(T(0)); 36098 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 36099 | return expr_gen(T(0)); 36100 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 36101 | return static_cast<details::variable_node<Type>*>(branch[1]); 36102 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 36103 | return static_cast<details::variable_node<Type>*>(branch[1]); 36104 | 36105 | switch (operation) 36106 | { 36107 | #define case_stmt(op0, op1) \ 36108 | case op0 : return expr_gen.node_allocator_-> \ 36109 | template allocate_cr<typename details::cov_node<Type,op1<Type> > > \ 36110 | (c, v); \ 36111 | 36112 | basic_opr_switch_statements 36113 | extended_opr_switch_statements 36114 | #undef case_stmt 36115 | default : return error_node(); 36116 | } 36117 | } 36118 | }; 36119 | 36120 | struct synthesize_voc_expression 36121 | { 36122 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36123 | const details::operator_type& operation, 36124 | expression_node_ptr (&branch)[2]) 36125 | { 36126 | const Type& v = static_cast<details::variable_node<Type>*>(branch[0])->ref (); 36127 | const Type c = static_cast<details::literal_node<Type>*> (branch[1])->value(); 36128 | 36129 | details::free_node(*(expr_gen.node_allocator_), branch[1]); 36130 | 36131 | if (expr_gen.cardinal_pow_optimisable(operation,c)) 36132 | { 36133 | if (std::equal_to<T>()(T(1),c)) 36134 | return branch[0]; 36135 | else 36136 | return expr_gen.cardinal_pow_optimisation(v,c); 36137 | } 36138 | else if (std::equal_to<T>()(T(0),c) && (details::e_mul == operation)) 36139 | return expr_gen(T(0)); 36140 | else if (std::equal_to<T>()(T(0),c) && (details::e_div == operation)) 36141 | return expr_gen(std::numeric_limits<T>::quiet_NaN()); 36142 | else if (std::equal_to<T>()(T(0),c) && (details::e_add == operation)) 36143 | return static_cast<details::variable_node<Type>*>(branch[0]); 36144 | else if (std::equal_to<T>()(T(1),c) && (details::e_mul == operation)) 36145 | return static_cast<details::variable_node<Type>*>(branch[0]); 36146 | else if (std::equal_to<T>()(T(1),c) && (details::e_div == operation)) 36147 | return static_cast<details::variable_node<Type>*>(branch[0]); 36148 | 36149 | switch (operation) 36150 | { 36151 | #define case_stmt(op0, op1) \ 36152 | case op0 : return expr_gen.node_allocator_-> \ 36153 | template allocate_rc<typename details::voc_node<Type,op1<Type> > > \ 36154 | (v, c); \ 36155 | 36156 | basic_opr_switch_statements 36157 | extended_opr_switch_statements 36158 | #undef case_stmt 36159 | default : return error_node(); 36160 | } 36161 | } 36162 | }; 36163 | 36164 | struct synthesize_sf3ext_expression 36165 | { 36166 | template <typename T0, typename T1, typename T2> 36167 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36168 | const details::operator_type& sf3opr, 36169 | T0 t0, T1 t1, T2 t2) 36170 | { 36171 | switch (sf3opr) 36172 | { 36173 | #define case_stmt(op) \ 36174 | case details::e_sf##op : return details::T0oT1oT2_sf3ext<T,T0,T1,T2,details::sf##op##_op<Type> >:: \ 36175 | allocate(*(expr_gen.node_allocator_), t0, t1, t2); \ 36176 | 36177 | case_stmt(00) case_stmt(01) case_stmt(02) case_stmt(03) 36178 | case_stmt(04) case_stmt(05) case_stmt(06) case_stmt(07) 36179 | case_stmt(08) case_stmt(09) case_stmt(10) case_stmt(11) 36180 | case_stmt(12) case_stmt(13) case_stmt(14) case_stmt(15) 36181 | case_stmt(16) case_stmt(17) case_stmt(18) case_stmt(19) 36182 | case_stmt(20) case_stmt(21) case_stmt(22) case_stmt(23) 36183 | case_stmt(24) case_stmt(25) case_stmt(26) case_stmt(27) 36184 | case_stmt(28) case_stmt(29) case_stmt(30) 36185 | #undef case_stmt 36186 | default : return error_node(); 36187 | } 36188 | } 36189 | 36190 | template <typename T0, typename T1, typename T2> 36191 | static inline bool compile(expression_generator<Type>& expr_gen, const std::string& id, 36192 | T0 t0, T1 t1, T2 t2, 36193 | expression_node_ptr& result) 36194 | { 36195 | details::operator_type sf3opr; 36196 | 36197 | if (!expr_gen.sf3_optimisable(id,sf3opr)) 36198 | return false; 36199 | else 36200 | result = synthesize_sf3ext_expression::template process<T0, T1, T2> 36201 | (expr_gen, sf3opr, t0, t1, t2); 36202 | 36203 | return true; 36204 | } 36205 | }; 36206 | 36207 | struct synthesize_sf4ext_expression 36208 | { 36209 | template <typename T0, typename T1, typename T2, typename T3> 36210 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36211 | const details::operator_type& sf4opr, 36212 | T0 t0, T1 t1, T2 t2, T3 t3) 36213 | { 36214 | switch (sf4opr) 36215 | { 36216 | #define case_stmt0(op) \ 36217 | case details::e_sf##op : return details::T0oT1oT2oT3_sf4ext<Type,T0,T1,T2,T3,details::sf##op##_op<Type> >:: \ 36218 | allocate(*(expr_gen.node_allocator_), t0, t1, t2, t3); \ 36219 | 36220 | #define case_stmt1(op) \ 36221 | case details::e_sf4ext##op : return details::T0oT1oT2oT3_sf4ext<Type,T0,T1,T2,T3,details::sfext##op##_op<Type> >:: \ 36222 | allocate(*(expr_gen.node_allocator_), t0, t1, t2, t3); \ 36223 | 36224 | case_stmt0(48) case_stmt0(49) case_stmt0(50) case_stmt0(51) 36225 | case_stmt0(52) case_stmt0(53) case_stmt0(54) case_stmt0(55) 36226 | case_stmt0(56) case_stmt0(57) case_stmt0(58) case_stmt0(59) 36227 | case_stmt0(60) case_stmt0(61) case_stmt0(62) case_stmt0(63) 36228 | case_stmt0(64) case_stmt0(65) case_stmt0(66) case_stmt0(67) 36229 | case_stmt0(68) case_stmt0(69) case_stmt0(70) case_stmt0(71) 36230 | case_stmt0(72) case_stmt0(73) case_stmt0(74) case_stmt0(75) 36231 | case_stmt0(76) case_stmt0(77) case_stmt0(78) case_stmt0(79) 36232 | case_stmt0(80) case_stmt0(81) case_stmt0(82) case_stmt0(83) 36233 | 36234 | case_stmt1(00) case_stmt1(01) case_stmt1(02) case_stmt1(03) 36235 | case_stmt1(04) case_stmt1(05) case_stmt1(06) case_stmt1(07) 36236 | case_stmt1(08) case_stmt1(09) case_stmt1(10) case_stmt1(11) 36237 | case_stmt1(12) case_stmt1(13) case_stmt1(14) case_stmt1(15) 36238 | case_stmt1(16) case_stmt1(17) case_stmt1(18) case_stmt1(19) 36239 | case_stmt1(20) case_stmt1(21) case_stmt1(22) case_stmt1(23) 36240 | case_stmt1(24) case_stmt1(25) case_stmt1(26) case_stmt1(27) 36241 | case_stmt1(28) case_stmt1(29) case_stmt1(30) case_stmt1(31) 36242 | case_stmt1(32) case_stmt1(33) case_stmt1(34) case_stmt1(35) 36243 | case_stmt1(36) case_stmt1(37) case_stmt1(38) case_stmt1(39) 36244 | case_stmt1(40) case_stmt1(41) case_stmt1(42) case_stmt1(43) 36245 | case_stmt1(44) case_stmt1(45) case_stmt1(46) case_stmt1(47) 36246 | case_stmt1(48) case_stmt1(49) case_stmt1(50) case_stmt1(51) 36247 | case_stmt1(52) case_stmt1(53) case_stmt1(54) case_stmt1(55) 36248 | case_stmt1(56) case_stmt1(57) case_stmt1(58) case_stmt1(59) 36249 | case_stmt1(60) case_stmt1(61) 36250 | 36251 | #undef case_stmt0 36252 | #undef case_stmt1 36253 | default : return error_node(); 36254 | } 36255 | } 36256 | 36257 | template <typename T0, typename T1, typename T2, typename T3> 36258 | static inline bool compile(expression_generator<Type>& expr_gen, const std::string& id, 36259 | T0 t0, T1 t1, T2 t2, T3 t3, 36260 | expression_node_ptr& result) 36261 | { 36262 | details::operator_type sf4opr; 36263 | 36264 | if (!expr_gen.sf4_optimisable(id,sf4opr)) 36265 | return false; 36266 | else 36267 | result = synthesize_sf4ext_expression::template process<T0, T1, T2, T3> 36268 | (expr_gen, sf4opr, t0, t1, t2, t3); 36269 | 36270 | return true; 36271 | } 36272 | 36273 | // T o (sf3ext) 36274 | template <typename ExternalType> 36275 | static inline bool compile_right(expression_generator<Type>& expr_gen, 36276 | ExternalType t, 36277 | const details::operator_type& operation, 36278 | expression_node_ptr& sf3node, 36279 | expression_node_ptr& result) 36280 | { 36281 | if (!details::is_sf3ext_node(sf3node)) 36282 | return false; 36283 | 36284 | typedef details::T0oT1oT2_base_node<Type>* sf3ext_base_ptr; 36285 | 36286 | sf3ext_base_ptr n = static_cast<sf3ext_base_ptr>(sf3node); 36287 | const std::string id = "t" + expr_gen.to_str(operation) + "(" + n->type_id() + ")" 36288 | 36289 | switch (n->type()) 36290 | { 36291 | case details::expression_node<Type>::e_covoc : return compile_right_impl 36292 | <typename covoc_t::sf3_type_node,ExternalType, ctype, vtype, ctype> 36293 | (expr_gen, id, t, sf3node, result); 36294 | 36295 | case details::expression_node<Type>::e_covov : return compile_right_impl 36296 | <typename covov_t::sf3_type_node,ExternalType, ctype, vtype, vtype> 36297 | (expr_gen, id, t, sf3node, result); 36298 | 36299 | case details::expression_node<Type>::e_vocov : return compile_right_impl 36300 | <typename vocov_t::sf3_type_node,ExternalType, vtype, ctype, vtype> 36301 | (expr_gen, id, t, sf3node, result); 36302 | 36303 | case details::expression_node<Type>::e_vovoc : return compile_right_impl 36304 | <typename vovoc_t::sf3_type_node,ExternalType, vtype, vtype, ctype> 36305 | (expr_gen, id, t, sf3node, result); 36306 | 36307 | case details::expression_node<Type>::e_vovov : return compile_right_impl 36308 | <typename vovov_t::sf3_type_node,ExternalType, vtype, vtype, vtype> 36309 | (expr_gen, id, t, sf3node, result); 36310 | 36311 | default : return false; 36312 | } 36313 | } 36314 | 36315 | // (sf3ext) o T 36316 | template <typename ExternalType> 36317 | static inline bool compile_left(expression_generator<Type>& expr_gen, 36318 | ExternalType t, 36319 | const details::operator_type& operation, 36320 | expression_node_ptr& sf3node, 36321 | expression_node_ptr& result) 36322 | { 36323 | if (!details::is_sf3ext_node(sf3node)) 36324 | return false; 36325 | 36326 | typedef details::T0oT1oT2_base_node<Type>* sf3ext_base_ptr; 36327 | 36328 | sf3ext_base_ptr n = static_cast<sf3ext_base_ptr>(sf3node); 36329 | 36330 | const std::string id = "(" + n->type_id() + ")" + expr_gen.to_str(operation) + "t" 36331 | 36332 | switch (n->type()) 36333 | { 36334 | case details::expression_node<Type>::e_covoc : return compile_left_impl 36335 | <typename covoc_t::sf3_type_node,ExternalType, ctype, vtype, ctype> 36336 | (expr_gen, id, t, sf3node, result); 36337 | 36338 | case details::expression_node<Type>::e_covov : return compile_left_impl 36339 | <typename covov_t::sf3_type_node,ExternalType, ctype, vtype, vtype> 36340 | (expr_gen, id, t, sf3node, result); 36341 | 36342 | case details::expression_node<Type>::e_vocov : return compile_left_impl 36343 | <typename vocov_t::sf3_type_node,ExternalType, vtype, ctype, vtype> 36344 | (expr_gen, id, t, sf3node, result); 36345 | 36346 | case details::expression_node<Type>::e_vovoc : return compile_left_impl 36347 | <typename vovoc_t::sf3_type_node,ExternalType, vtype, vtype, ctype> 36348 | (expr_gen, id, t, sf3node, result); 36349 | 36350 | case details::expression_node<Type>::e_vovov : return compile_left_impl 36351 | <typename vovov_t::sf3_type_node,ExternalType, vtype, vtype, vtype> 36352 | (expr_gen, id, t, sf3node, result); 36353 | 36354 | default : return false; 36355 | } 36356 | } 36357 | 36358 | template <typename SF3TypeNode, typename ExternalType, typename T0, typename T1, typename T2> 36359 | static inline bool compile_right_impl(expression_generator<Type>& expr_gen, 36360 | const std::string& id, 36361 | ExternalType t, 36362 | expression_node_ptr& node, 36363 | expression_node_ptr& result) 36364 | { 36365 | SF3TypeNode* n = dynamic_cast<SF3TypeNode*>(node); 36366 | 36367 | if (n) 36368 | { 36369 | T0 t0 = n->t0(); 36370 | T1 t1 = n->t1(); 36371 | T2 t2 = n->t2(); 36372 | 36373 | return synthesize_sf4ext_expression::template compile<ExternalType, T0, T1, T2> 36374 | (expr_gen, id, t, t0, t1, t2, result); 36375 | } 36376 | else 36377 | return false; 36378 | } 36379 | 36380 | template <typename SF3TypeNode, typename ExternalType, typename T0, typename T1, typename T2> 36381 | static inline bool compile_left_impl(expression_generator<Type>& expr_gen, 36382 | const std::string& id, 36383 | ExternalType t, 36384 | expression_node_ptr& node, 36385 | expression_node_ptr& result) 36386 | { 36387 | SF3TypeNode* n = dynamic_cast<SF3TypeNode*>(node); 36388 | 36389 | if (n) 36390 | { 36391 | T0 t0 = n->t0(); 36392 | T1 t1 = n->t1(); 36393 | T2 t2 = n->t2(); 36394 | 36395 | return synthesize_sf4ext_expression::template compile<T0, T1, T2, ExternalType> 36396 | (expr_gen, id, t0, t1, t2, t, result); 36397 | } 36398 | else 36399 | return false; 36400 | } 36401 | }; 36402 | 36403 | struct synthesize_vovov_expression0 36404 | { 36405 | typedef typename vovov_t::type0 node_type; 36406 | typedef typename vovov_t::sf3_type sf3_type; 36407 | 36408 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36409 | const details::operator_type& operation, 36410 | expression_node_ptr (&branch)[2]) 36411 | { 36412 | // (v0 o0 v1) o1 (v2) 36413 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[0]); 36414 | const Type& v0 = vov->v0(); 36415 | const Type& v1 = vov->v1(); 36416 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 36417 | const details::operator_type o0 = vov->operation(); 36418 | const details::operator_type o1 = operation; 36419 | 36420 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36421 | 36422 | expression_node_ptr result = error_node(); 36423 | 36424 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36425 | { 36426 | // (v0 / v1) / v2 --> (vovov) v0 / (v1 * v2) 36427 | if ((details::e_div == o0) && (details::e_div == o1)) 36428 | { 36429 | const bool synthesis_result = 36430 | synthesize_sf3ext_expression:: 36431 | template compile<vtype, vtype, vtype>(expr_gen, "t/(t*t)", v0, v1, v2, result); 36432 | 36433 | exprtk_debug(("(v0 / v1) / v2 --> (vovov) v0 / (v1 * v2)\n")); 36434 | 36435 | return (synthesis_result) ? result : error_node(); 36436 | } 36437 | } 36438 | 36439 | const bool synthesis_result = 36440 | synthesize_sf3ext_expression::template compile<vtype, vtype, vtype> 36441 | (expr_gen, id(expr_gen, o0, o1), v0, v1, v2, result); 36442 | 36443 | if (synthesis_result) 36444 | return result; 36445 | 36446 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36447 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36448 | 36449 | if (!expr_gen.valid_operator(o0,f0)) 36450 | return error_node(); 36451 | else if (!expr_gen.valid_operator(o1,f1)) 36452 | return error_node(); 36453 | else 36454 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, f0, f1); 36455 | } 36456 | 36457 | static inline std::string id(expression_generator<Type>& expr_gen, 36458 | const details::operator_type o0, 36459 | const details::operator_type o1) 36460 | { 36461 | return details::build_string() 36462 | << "(t" << expr_gen.to_str(o0) 36463 | << "t)" << expr_gen.to_str(o1) 36464 | << "t" 36465 | } 36466 | }; 36467 | 36468 | struct synthesize_vovov_expression1 36469 | { 36470 | typedef typename vovov_t::type1 node_type; 36471 | typedef typename vovov_t::sf3_type sf3_type; 36472 | 36473 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36474 | const details::operator_type& operation, 36475 | expression_node_ptr (&branch)[2]) 36476 | { 36477 | // (v0) o0 (v1 o1 v2) 36478 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[1]); 36479 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 36480 | const Type& v1 = vov->v0(); 36481 | const Type& v2 = vov->v1(); 36482 | const details::operator_type o0 = operation; 36483 | const details::operator_type o1 = vov->operation(); 36484 | 36485 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36486 | 36487 | expression_node_ptr result = error_node(); 36488 | 36489 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36490 | { 36491 | // v0 / (v1 / v2) --> (vovov) (v0 * v2) / v1 36492 | if ((details::e_div == o0) && (details::e_div == o1)) 36493 | { 36494 | const bool synthesis_result = 36495 | synthesize_sf3ext_expression:: 36496 | template compile<vtype, vtype, vtype>(expr_gen, "(t*t)/t", v0, v2, v1, result); 36497 | 36498 | exprtk_debug(("v0 / (v1 / v2) --> (vovov) (v0 * v2) / v1\n")); 36499 | 36500 | return (synthesis_result) ? result : error_node(); 36501 | } 36502 | } 36503 | 36504 | const bool synthesis_result = 36505 | synthesize_sf3ext_expression::template compile<vtype, vtype, vtype> 36506 | (expr_gen, id(expr_gen, o0, o1), v0, v1, v2, result); 36507 | 36508 | if (synthesis_result) 36509 | return result; 36510 | 36511 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36512 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36513 | 36514 | if (!expr_gen.valid_operator(o0,f0)) 36515 | return error_node(); 36516 | else if (!expr_gen.valid_operator(o1,f1)) 36517 | return error_node(); 36518 | else 36519 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, f0, f1); 36520 | } 36521 | 36522 | static inline std::string id(expression_generator<Type>& expr_gen, 36523 | const details::operator_type o0, 36524 | const details::operator_type o1) 36525 | { 36526 | return details::build_string() 36527 | << "t" << expr_gen.to_str(o0) 36528 | << "(t" << expr_gen.to_str(o1) 36529 | << "t)" 36530 | } 36531 | }; 36532 | 36533 | struct synthesize_vovoc_expression0 36534 | { 36535 | typedef typename vovoc_t::type0 node_type; 36536 | typedef typename vovoc_t::sf3_type sf3_type; 36537 | 36538 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36539 | const details::operator_type& operation, 36540 | expression_node_ptr (&branch)[2]) 36541 | { 36542 | // (v0 o0 v1) o1 (c) 36543 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[0]); 36544 | const Type& v0 = vov->v0(); 36545 | const Type& v1 = vov->v1(); 36546 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 36547 | const details::operator_type o0 = vov->operation(); 36548 | const details::operator_type o1 = operation; 36549 | 36550 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36551 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36552 | 36553 | expression_node_ptr result = error_node(); 36554 | 36555 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36556 | { 36557 | // (v0 / v1) / c --> (vovoc) v0 / (v1 * c) 36558 | if ((details::e_div == o0) && (details::e_div == o1)) 36559 | { 36560 | const bool synthesis_result = 36561 | synthesize_sf3ext_expression:: 36562 | template compile<vtype, vtype, ctype>(expr_gen, "t/(t*t)", v0, v1, c, result); 36563 | 36564 | exprtk_debug(("(v0 / v1) / c --> (vovoc) v0 / (v1 * c)\n")); 36565 | 36566 | return (synthesis_result) ? result : error_node(); 36567 | } 36568 | } 36569 | 36570 | const bool synthesis_result = 36571 | synthesize_sf3ext_expression::template compile<vtype, vtype, ctype> 36572 | (expr_gen, id(expr_gen, o0, o1), v0, v1, c, result); 36573 | 36574 | if (synthesis_result) 36575 | return result; 36576 | 36577 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36578 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36579 | 36580 | if (!expr_gen.valid_operator(o0,f0)) 36581 | return error_node(); 36582 | else if (!expr_gen.valid_operator(o1,f1)) 36583 | return error_node(); 36584 | else 36585 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, f0, f1); 36586 | } 36587 | 36588 | static inline std::string id(expression_generator<Type>& expr_gen, 36589 | const details::operator_type o0, 36590 | const details::operator_type o1) 36591 | { 36592 | return details::build_string() 36593 | << "(t" << expr_gen.to_str(o0) 36594 | << "t)" << expr_gen.to_str(o1) 36595 | << "t" 36596 | } 36597 | }; 36598 | 36599 | struct synthesize_vovoc_expression1 36600 | { 36601 | typedef typename vovoc_t::type1 node_type; 36602 | typedef typename vovoc_t::sf3_type sf3_type; 36603 | 36604 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36605 | const details::operator_type& operation, 36606 | expression_node_ptr (&branch)[2]) 36607 | { 36608 | // (v0) o0 (v1 o1 c) 36609 | const details::voc_base_node<Type>* voc = static_cast<const details::voc_base_node<Type>*>(branch[1]); 36610 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 36611 | const Type& v1 = voc->v(); 36612 | const Type c = voc->c(); 36613 | const details::operator_type o0 = operation; 36614 | const details::operator_type o1 = voc->operation(); 36615 | 36616 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36617 | 36618 | expression_node_ptr result = error_node(); 36619 | 36620 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36621 | { 36622 | // v0 / (v1 / c) --> (vocov) (v0 * c) / v1 36623 | if ((details::e_div == o0) && (details::e_div == o1)) 36624 | { 36625 | const bool synthesis_result = 36626 | synthesize_sf3ext_expression:: 36627 | template compile<vtype, ctype, vtype>(expr_gen, "(t*t)/t", v0, c, v1, result); 36628 | 36629 | exprtk_debug(("v0 / (v1 / c) --> (vocov) (v0 * c) / v1\n")); 36630 | 36631 | return (synthesis_result) ? result : error_node(); 36632 | } 36633 | } 36634 | 36635 | const bool synthesis_result = 36636 | synthesize_sf3ext_expression::template compile<vtype, vtype, ctype> 36637 | (expr_gen, id(expr_gen, o0, o1), v0, v1, c, result); 36638 | 36639 | if (synthesis_result) 36640 | return result; 36641 | 36642 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36643 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36644 | 36645 | if (!expr_gen.valid_operator(o0,f0)) 36646 | return error_node(); 36647 | else if (!expr_gen.valid_operator(o1,f1)) 36648 | return error_node(); 36649 | else 36650 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, f0, f1); 36651 | } 36652 | 36653 | static inline std::string id(expression_generator<Type>& expr_gen, 36654 | const details::operator_type o0, 36655 | const details::operator_type o1) 36656 | { 36657 | return details::build_string() 36658 | << "t" << expr_gen.to_str(o0) 36659 | << "(t" << expr_gen.to_str(o1) 36660 | << "t)" 36661 | } 36662 | }; 36663 | 36664 | struct synthesize_vocov_expression0 36665 | { 36666 | typedef typename vocov_t::type0 node_type; 36667 | typedef typename vocov_t::sf3_type sf3_type; 36668 | 36669 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36670 | const details::operator_type& operation, 36671 | expression_node_ptr (&branch)[2]) 36672 | { 36673 | // (v0 o0 c) o1 (v1) 36674 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[0]); 36675 | const Type& v0 = voc->v(); 36676 | const Type c = voc->c(); 36677 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 36678 | const details::operator_type o0 = voc->operation(); 36679 | const details::operator_type o1 = operation; 36680 | 36681 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36682 | 36683 | expression_node_ptr result = error_node(); 36684 | 36685 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36686 | { 36687 | // (v0 / c) / v1 --> (vovoc) v0 / (v1 * c) 36688 | if ((details::e_div == o0) && (details::e_div == o1)) 36689 | { 36690 | const bool synthesis_result = 36691 | synthesize_sf3ext_expression:: 36692 | template compile<vtype, vtype, ctype>(expr_gen, "t/(t*t)", v0, v1, c, result); 36693 | 36694 | exprtk_debug(("(v0 / c) / v1 --> (vovoc) v0 / (v1 * c)\n")); 36695 | 36696 | return (synthesis_result) ? result : error_node(); 36697 | } 36698 | } 36699 | 36700 | const bool synthesis_result = 36701 | synthesize_sf3ext_expression::template compile<vtype, ctype, vtype> 36702 | (expr_gen, id(expr_gen, o0, o1), v0, c, v1, result); 36703 | 36704 | if (synthesis_result) 36705 | return result; 36706 | 36707 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36708 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36709 | 36710 | if (!expr_gen.valid_operator(o0,f0)) 36711 | return error_node(); 36712 | else if (!expr_gen.valid_operator(o1,f1)) 36713 | return error_node(); 36714 | else 36715 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, f0, f1); 36716 | } 36717 | 36718 | static inline std::string id(expression_generator<Type>& expr_gen, 36719 | const details::operator_type o0, 36720 | const details::operator_type o1) 36721 | { 36722 | return details::build_string() 36723 | << "(t" << expr_gen.to_str(o0) 36724 | << "t)" << expr_gen.to_str(o1) 36725 | << "t" 36726 | } 36727 | }; 36728 | 36729 | struct synthesize_vocov_expression1 36730 | { 36731 | typedef typename vocov_t::type1 node_type; 36732 | typedef typename vocov_t::sf3_type sf3_type; 36733 | 36734 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36735 | const details::operator_type& operation, 36736 | expression_node_ptr (&branch)[2]) 36737 | { 36738 | // (v0) o0 (c o1 v1) 36739 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[1]); 36740 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 36741 | const Type c = cov->c(); 36742 | const Type& v1 = cov->v(); 36743 | const details::operator_type o0 = operation; 36744 | const details::operator_type o1 = cov->operation(); 36745 | 36746 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36747 | 36748 | expression_node_ptr result = error_node(); 36749 | 36750 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36751 | { 36752 | // v0 / (c / v1) --> (vovoc) (v0 * v1) / c 36753 | if ((details::e_div == o0) && (details::e_div == o1)) 36754 | { 36755 | const bool synthesis_result = 36756 | synthesize_sf3ext_expression:: 36757 | template compile<vtype, vtype, ctype>(expr_gen, "(t*t)/t", v0, v1, c, result); 36758 | 36759 | exprtk_debug(("v0 / (c / v1) --> (vovoc) (v0 * v1) / c\n")); 36760 | 36761 | return (synthesis_result) ? result : error_node(); 36762 | } 36763 | } 36764 | 36765 | const bool synthesis_result = 36766 | synthesize_sf3ext_expression::template compile<vtype, ctype, vtype> 36767 | (expr_gen, id(expr_gen, o0, o1), v0, c, v1, result); 36768 | 36769 | if (synthesis_result) 36770 | return result; 36771 | 36772 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36773 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36774 | 36775 | if (!expr_gen.valid_operator(o0,f0)) 36776 | return error_node(); 36777 | else if (!expr_gen.valid_operator(o1,f1)) 36778 | return error_node(); 36779 | else 36780 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, f0, f1); 36781 | } 36782 | 36783 | static inline std::string id(expression_generator<Type>& expr_gen, 36784 | const details::operator_type o0, 36785 | const details::operator_type o1) 36786 | { 36787 | return details::build_string() 36788 | << "t" << expr_gen.to_str(o0) 36789 | << "(t" << expr_gen.to_str(o1) 36790 | << "t)" 36791 | } 36792 | }; 36793 | 36794 | struct synthesize_covov_expression0 36795 | { 36796 | typedef typename covov_t::type0 node_type; 36797 | typedef typename covov_t::sf3_type sf3_type; 36798 | 36799 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36800 | const details::operator_type& operation, 36801 | expression_node_ptr (&branch)[2]) 36802 | { 36803 | // (c o0 v0) o1 (v1) 36804 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[0]); 36805 | const Type c = cov->c(); 36806 | const Type& v0 = cov->v(); 36807 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 36808 | const details::operator_type o0 = cov->operation(); 36809 | const details::operator_type o1 = operation; 36810 | 36811 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36812 | 36813 | expression_node_ptr result = error_node(); 36814 | 36815 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36816 | { 36817 | // (c / v0) / v1 --> (covov) c / (v0 * v1) 36818 | if ((details::e_div == o0) && (details::e_div == o1)) 36819 | { 36820 | const bool synthesis_result = 36821 | synthesize_sf3ext_expression:: 36822 | template compile<ctype, vtype, vtype>(expr_gen, "t/(t*t)", c, v0, v1, result); 36823 | 36824 | exprtk_debug(("(c / v0) / v1 --> (covov) c / (v0 * v1)\n")); 36825 | 36826 | return (synthesis_result) ? result : error_node(); 36827 | } 36828 | } 36829 | 36830 | const bool synthesis_result = 36831 | synthesize_sf3ext_expression::template compile<ctype, vtype, vtype> 36832 | (expr_gen, id(expr_gen, o0, o1), c, v0, v1, result); 36833 | 36834 | if (synthesis_result) 36835 | return result; 36836 | 36837 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36838 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36839 | 36840 | if (!expr_gen.valid_operator(o0,f0)) 36841 | return error_node(); 36842 | else if (!expr_gen.valid_operator(o1,f1)) 36843 | return error_node(); 36844 | else 36845 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, f0, f1); 36846 | } 36847 | 36848 | static inline std::string id(expression_generator<Type>& expr_gen, 36849 | const details::operator_type o0, 36850 | const details::operator_type o1) 36851 | { 36852 | return details::build_string() 36853 | << "(t" << expr_gen.to_str(o0) 36854 | << "t)" << expr_gen.to_str(o1) 36855 | << "t" 36856 | } 36857 | }; 36858 | 36859 | struct synthesize_covov_expression1 36860 | { 36861 | typedef typename covov_t::type1 node_type; 36862 | typedef typename covov_t::sf3_type sf3_type; 36863 | 36864 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36865 | const details::operator_type& operation, 36866 | expression_node_ptr (&branch)[2]) 36867 | { 36868 | // (c) o0 (v0 o1 v1) 36869 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[1]); 36870 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 36871 | const Type& v0 = vov->v0(); 36872 | const Type& v1 = vov->v1(); 36873 | const details::operator_type o0 = operation; 36874 | const details::operator_type o1 = vov->operation(); 36875 | 36876 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36877 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36878 | 36879 | expression_node_ptr result = error_node(); 36880 | 36881 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36882 | { 36883 | // c / (v0 / v1) --> (covov) (c * v1) / v0 36884 | if ((details::e_div == o0) && (details::e_div == o1)) 36885 | { 36886 | const bool synthesis_result = 36887 | synthesize_sf3ext_expression:: 36888 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", c, v1, v0, result); 36889 | 36890 | exprtk_debug(("c / (v0 / v1) --> (covov) (c * v1) / v0\n")); 36891 | 36892 | return (synthesis_result) ? result : error_node(); 36893 | } 36894 | } 36895 | 36896 | const bool synthesis_result = 36897 | synthesize_sf3ext_expression::template compile<ctype, vtype, vtype> 36898 | (expr_gen, id(expr_gen, o0, o1), c, v0, v1, result); 36899 | 36900 | if (synthesis_result) 36901 | return result; 36902 | 36903 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 36904 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 36905 | 36906 | if (!expr_gen.valid_operator(o0,f0)) 36907 | return error_node(); 36908 | else if (!expr_gen.valid_operator(o1,f1)) 36909 | return error_node(); 36910 | else 36911 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, f0, f1); 36912 | } 36913 | 36914 | static inline std::string id(expression_generator<Type>& expr_gen, 36915 | const details::operator_type o0, 36916 | const details::operator_type o1) 36917 | { 36918 | return details::build_string() 36919 | << "t" << expr_gen.to_str(o0) 36920 | << "(t" << expr_gen.to_str(o1) 36921 | << "t)" 36922 | } 36923 | }; 36924 | 36925 | struct synthesize_covoc_expression0 36926 | { 36927 | typedef typename covoc_t::type0 node_type; 36928 | typedef typename covoc_t::sf3_type sf3_type; 36929 | 36930 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 36931 | const details::operator_type& operation, 36932 | expression_node_ptr (&branch)[2]) 36933 | { 36934 | // (c0 o0 v) o1 (c1) 36935 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[0]); 36936 | const Type c0 = cov->c(); 36937 | const Type& v = cov->v(); 36938 | const Type c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 36939 | const details::operator_type o0 = cov->operation(); 36940 | const details::operator_type o1 = operation; 36941 | 36942 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 36943 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 36944 | 36945 | expression_node_ptr result = error_node(); 36946 | 36947 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 36948 | { 36949 | // (c0 + v) + c1 --> (cov) (c0 + c1) + v 36950 | if ((details::e_add == o0) && (details::e_add == o1)) 36951 | { 36952 | exprtk_debug(("(c0 + v) + c1 --> (cov) (c0 + c1) + v\n")); 36953 | 36954 | return expr_gen.node_allocator_-> 36955 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 + c1, v); 36956 | } 36957 | // (c0 + v) - c1 --> (cov) (c0 - c1) + v 36958 | else if ((details::e_add == o0) && (details::e_sub == o1)) 36959 | { 36960 | exprtk_debug(("(c0 + v) - c1 --> (cov) (c0 - c1) + v\n")); 36961 | 36962 | return expr_gen.node_allocator_-> 36963 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 - c1, v); 36964 | } 36965 | // (c0 - v) + c1 --> (cov) (c0 + c1) - v 36966 | else if ((details::e_sub == o0) && (details::e_add == o1)) 36967 | { 36968 | exprtk_debug(("(c0 - v) + c1 --> (cov) (c0 + c1) - v\n")); 36969 | 36970 | return expr_gen.node_allocator_-> 36971 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 + c1, v); 36972 | } 36973 | // (c0 - v) - c1 --> (cov) (c0 - c1) - v 36974 | else if ((details::e_sub == o0) && (details::e_sub == o1)) 36975 | { 36976 | exprtk_debug(("(c0 - v) - c1 --> (cov) (c0 - c1) - v\n")); 36977 | 36978 | return expr_gen.node_allocator_-> 36979 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 - c1, v); 36980 | } 36981 | // (c0 * v) * c1 --> (cov) (c0 * c1) * v 36982 | else if ((details::e_mul == o0) && (details::e_mul == o1)) 36983 | { 36984 | exprtk_debug(("(c0 * v) * c1 --> (cov) (c0 * c1) * v\n")); 36985 | 36986 | return expr_gen.node_allocator_-> 36987 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 * c1, v); 36988 | } 36989 | // (c0 * v) / c1 --> (cov) (c0 / c1) * v 36990 | else if ((details::e_mul == o0) && (details::e_div == o1)) 36991 | { 36992 | exprtk_debug(("(c0 * v) / c1 --> (cov) (c0 / c1) * v\n")); 36993 | 36994 | return expr_gen.node_allocator_-> 36995 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 / c1, v); 36996 | } 36997 | // (c0 / v) * c1 --> (cov) (c0 * c1) / v 36998 | else if ((details::e_div == o0) && (details::e_mul == o1)) 36999 | { 37000 | exprtk_debug(("(c0 / v) * c1 --> (cov) (c0 * c1) / v\n")); 37001 | 37002 | return expr_gen.node_allocator_-> 37003 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 * c1, v); 37004 | } 37005 | // (c0 / v) / c1 --> (cov) (c0 / c1) / v 37006 | else if ((details::e_div == o0) && (details::e_div == o1)) 37007 | { 37008 | exprtk_debug(("(c0 / v) / c1 --> (cov) (c0 / c1) / v\n")); 37009 | 37010 | return expr_gen.node_allocator_-> 37011 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 / c1, v); 37012 | } 37013 | } 37014 | 37015 | const bool synthesis_result = 37016 | synthesize_sf3ext_expression::template compile<ctype, vtype, ctype> 37017 | (expr_gen, id(expr_gen, o0, o1), c0, v, c1, result); 37018 | 37019 | if (synthesis_result) 37020 | return result; 37021 | 37022 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37023 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37024 | 37025 | if (!expr_gen.valid_operator(o0,f0)) 37026 | return error_node(); 37027 | else if (!expr_gen.valid_operator(o1,f1)) 37028 | return error_node(); 37029 | else 37030 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v, c1, f0, f1); 37031 | } 37032 | 37033 | static inline std::string id(expression_generator<Type>& expr_gen, 37034 | const details::operator_type o0, 37035 | const details::operator_type o1) 37036 | { 37037 | return details::build_string() 37038 | << "(t" << expr_gen.to_str(o0) 37039 | << "t)" << expr_gen.to_str(o1) 37040 | << "t" 37041 | } 37042 | }; 37043 | 37044 | struct synthesize_covoc_expression1 37045 | { 37046 | typedef typename covoc_t::type1 node_type; 37047 | typedef typename covoc_t::sf3_type sf3_type; 37048 | 37049 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37050 | const details::operator_type& operation, 37051 | expression_node_ptr (&branch)[2]) 37052 | { 37053 | // (c0) o0 (v o1 c1) 37054 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[1]); 37055 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 37056 | const Type& v = voc->v(); 37057 | const Type c1 = voc->c(); 37058 | const details::operator_type o0 = operation; 37059 | const details::operator_type o1 = voc->operation(); 37060 | 37061 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37062 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37063 | 37064 | expression_node_ptr result = error_node(); 37065 | 37066 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37067 | { 37068 | // (c0) + (v + c1) --> (cov) (c0 + c1) + v 37069 | if ((details::e_add == o0) && (details::e_add == o1)) 37070 | { 37071 | exprtk_debug(("(c0) + (v + c1) --> (cov) (c0 + c1) + v\n")); 37072 | 37073 | return expr_gen.node_allocator_-> 37074 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 + c1, v); 37075 | } 37076 | // (c0) + (v - c1) --> (cov) (c0 - c1) + v 37077 | else if ((details::e_add == o0) && (details::e_sub == o1)) 37078 | { 37079 | exprtk_debug(("(c0) + (v - c1) --> (cov) (c0 - c1) + v\n")); 37080 | 37081 | return expr_gen.node_allocator_-> 37082 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 - c1, v); 37083 | } 37084 | // (c0) - (v + c1) --> (cov) (c0 - c1) - v 37085 | else if ((details::e_sub == o0) && (details::e_add == o1)) 37086 | { 37087 | exprtk_debug(("(c0) - (v + c1) --> (cov) (c0 - c1) - v\n")); 37088 | 37089 | return expr_gen.node_allocator_-> 37090 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 - c1, v); 37091 | } 37092 | // (c0) - (v - c1) --> (cov) (c0 + c1) - v 37093 | else if ((details::e_sub == o0) && (details::e_sub == o1)) 37094 | { 37095 | exprtk_debug(("(c0) - (v - c1) --> (cov) (c0 + c1) - v\n")); 37096 | 37097 | return expr_gen.node_allocator_-> 37098 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 + c1, v); 37099 | } 37100 | // (c0) * (v * c1) --> (voc) v * (c0 * c1) 37101 | else if ((details::e_mul == o0) && (details::e_mul == o1)) 37102 | { 37103 | exprtk_debug(("(c0) * (v * c1) --> (voc) v * (c0 * c1)\n")); 37104 | 37105 | return expr_gen.node_allocator_-> 37106 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 * c1, v); 37107 | } 37108 | // (c0) * (v / c1) --> (cov) (c0 / c1) * v 37109 | else if ((details::e_mul == o0) && (details::e_div == o1)) 37110 | { 37111 | exprtk_debug(("(c0) * (v / c1) --> (cov) (c0 / c1) * v\n")); 37112 | 37113 | return expr_gen.node_allocator_-> 37114 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 / c1, v); 37115 | } 37116 | // (c0) / (v * c1) --> (cov) (c0 / c1) / v 37117 | else if ((details::e_div == o0) && (details::e_mul == o1)) 37118 | { 37119 | exprtk_debug(("(c0) / (v * c1) --> (cov) (c0 / c1) / v\n")); 37120 | 37121 | return expr_gen.node_allocator_-> 37122 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 / c1, v); 37123 | } 37124 | // (c0) / (v / c1) --> (cov) (c0 * c1) / v 37125 | else if ((details::e_div == o0) && (details::e_div == o1)) 37126 | { 37127 | exprtk_debug(("(c0) / (v / c1) --> (cov) (c0 * c1) / v\n")); 37128 | 37129 | return expr_gen.node_allocator_-> 37130 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 * c1, v); 37131 | } 37132 | } 37133 | 37134 | const bool synthesis_result = 37135 | synthesize_sf3ext_expression::template compile<ctype, vtype, ctype> 37136 | (expr_gen, id(expr_gen, o0, o1), c0, v, c1, result); 37137 | 37138 | if (synthesis_result) 37139 | return result; 37140 | 37141 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37142 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37143 | 37144 | if (!expr_gen.valid_operator(o0,f0)) 37145 | return error_node(); 37146 | else if (!expr_gen.valid_operator(o1,f1)) 37147 | return error_node(); 37148 | else 37149 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v, c1, f0, f1); 37150 | } 37151 | 37152 | static inline std::string id(expression_generator<Type>& expr_gen, 37153 | const details::operator_type o0, 37154 | const details::operator_type o1) 37155 | { 37156 | return details::build_string() 37157 | << "t" << expr_gen.to_str(o0) 37158 | << "(t" << expr_gen.to_str(o1) 37159 | << "t)" 37160 | } 37161 | }; 37162 | 37163 | struct synthesize_cocov_expression0 37164 | { 37165 | typedef typename cocov_t::type0 node_type; 37166 | static inline expression_node_ptr process(expression_generator<Type>&, 37167 | const details::operator_type&, 37168 | expression_node_ptr (&)[2]) 37169 | { 37170 | // (c0 o0 c1) o1 (v) - Not possible. 37171 | return error_node(); 37172 | } 37173 | }; 37174 | 37175 | struct synthesize_cocov_expression1 37176 | { 37177 | typedef typename cocov_t::type1 node_type; 37178 | typedef typename cocov_t::sf3_type sf3_type; 37179 | 37180 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37181 | const details::operator_type& operation, 37182 | expression_node_ptr (&branch)[2]) 37183 | { 37184 | // (c0) o0 (c1 o1 v) 37185 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[1]); 37186 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 37187 | const Type c1 = cov->c(); 37188 | const Type& v = cov->v(); 37189 | const details::operator_type o0 = operation; 37190 | const details::operator_type o1 = cov->operation(); 37191 | 37192 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37193 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37194 | 37195 | expression_node_ptr result = error_node(); 37196 | 37197 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37198 | { 37199 | // (c0) + (c1 + v) --> (cov) (c0 + c1) + v 37200 | if ((details::e_add == o0) && (details::e_add == o1)) 37201 | { 37202 | exprtk_debug(("(c0) + (c1 + v) --> (cov) (c0 + c1) + v\n")); 37203 | 37204 | return expr_gen.node_allocator_-> 37205 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 + c1, v); 37206 | } 37207 | // (c0) + (c1 - v) --> (cov) (c0 + c1) - v 37208 | else if ((details::e_add == o0) && (details::e_sub == o1)) 37209 | { 37210 | exprtk_debug(("(c0) + (c1 - v) --> (cov) (c0 + c1) - v\n")); 37211 | 37212 | return expr_gen.node_allocator_-> 37213 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 + c1, v); 37214 | } 37215 | // (c0) - (c1 + v) --> (cov) (c0 - c1) - v 37216 | else if ((details::e_sub == o0) && (details::e_add == o1)) 37217 | { 37218 | exprtk_debug(("(c0) - (c1 + v) --> (cov) (c0 - c1) - v\n")); 37219 | 37220 | return expr_gen.node_allocator_-> 37221 | template allocate_cr<typename details::cov_node<Type,details::sub_op<Type> > >(c0 - c1, v); 37222 | } 37223 | // (c0) - (c1 - v) --> (cov) (c0 - c1) + v 37224 | else if ((details::e_sub == o0) && (details::e_sub == o1)) 37225 | { 37226 | exprtk_debug(("(c0) - (c1 - v) --> (cov) (c0 - c1) + v\n")); 37227 | 37228 | return expr_gen.node_allocator_-> 37229 | template allocate_cr<typename details::cov_node<Type,details::add_op<Type> > >(c0 - c1, v); 37230 | } 37231 | // (c0) * (c1 * v) --> (cov) (c0 * c1) * v 37232 | else if ((details::e_mul == o0) && (details::e_mul == o1)) 37233 | { 37234 | exprtk_debug(("(c0) * (c1 * v) --> (cov) (c0 * c1) * v\n")); 37235 | 37236 | return expr_gen.node_allocator_-> 37237 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 * c1, v); 37238 | } 37239 | // (c0) * (c1 / v) --> (cov) (c0 * c1) / v 37240 | else if ((details::e_mul == o0) && (details::e_div == o1)) 37241 | { 37242 | exprtk_debug(("(c0) * (c1 / v) --> (cov) (c0 * c1) / v\n")); 37243 | 37244 | return expr_gen.node_allocator_-> 37245 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 * c1, v); 37246 | } 37247 | // (c0) / (c1 * v) --> (cov) (c0 / c1) / v 37248 | else if ((details::e_div == o0) && (details::e_mul == o1)) 37249 | { 37250 | exprtk_debug(("(c0) / (c1 * v) --> (cov) (c0 / c1) / v\n")); 37251 | 37252 | return expr_gen.node_allocator_-> 37253 | template allocate_cr<typename details::cov_node<Type,details::div_op<Type> > >(c0 / c1, v); 37254 | } 37255 | // (c0) / (c1 / v) --> (cov) (c0 / c1) * v 37256 | else if ((details::e_div == o0) && (details::e_div == o1)) 37257 | { 37258 | exprtk_debug(("(c0) / (c1 / v) --> (cov) (c0 / c1) * v\n")); 37259 | 37260 | return expr_gen.node_allocator_-> 37261 | template allocate_cr<typename details::cov_node<Type,details::mul_op<Type> > >(c0 / c1, v); 37262 | } 37263 | } 37264 | 37265 | const bool synthesis_result = 37266 | synthesize_sf3ext_expression::template compile<ctype, ctype, vtype> 37267 | (expr_gen, id(expr_gen, o0, o1), c0, c1, v, result); 37268 | 37269 | if (synthesis_result) 37270 | return result; 37271 | 37272 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37273 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37274 | 37275 | if (!expr_gen.valid_operator(o0,f0)) 37276 | return error_node(); 37277 | else if (!expr_gen.valid_operator(o1,f1)) 37278 | return error_node(); 37279 | else 37280 | return node_type::allocate(*(expr_gen.node_allocator_), c0, c1, v, f0, f1); 37281 | } 37282 | 37283 | static inline std::string id(expression_generator<Type>& expr_gen, 37284 | const details::operator_type o0, 37285 | const details::operator_type o1) 37286 | { 37287 | return details::build_string() 37288 | << "t" << expr_gen.to_str(o0) 37289 | << "(t" << expr_gen.to_str(o1) 37290 | << "t)" 37291 | } 37292 | }; 37293 | 37294 | struct synthesize_vococ_expression0 37295 | { 37296 | typedef typename vococ_t::type0 node_type; 37297 | typedef typename vococ_t::sf3_type sf3_type; 37298 | 37299 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37300 | const details::operator_type& operation, 37301 | expression_node_ptr (&branch)[2]) 37302 | { 37303 | // (v o0 c0) o1 (c1) 37304 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[0]); 37305 | const Type& v = voc->v(); 37306 | const Type& c0 = voc->c(); 37307 | const Type& c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 37308 | const details::operator_type o0 = voc->operation(); 37309 | const details::operator_type o1 = operation; 37310 | 37311 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37312 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37313 | 37314 | expression_node_ptr result = error_node(); 37315 | 37316 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37317 | { 37318 | // (v + c0) + c1 --> (voc) v + (c0 + c1) 37319 | if ((details::e_add == o0) && (details::e_add == o1)) 37320 | { 37321 | exprtk_debug(("(v + c0) + c1 --> (voc) v + (c0 + c1)\n")); 37322 | 37323 | return expr_gen.node_allocator_-> 37324 | template allocate_rc<typename details::voc_node<Type,details::add_op<Type> > >(v, c0 + c1); 37325 | } 37326 | // (v + c0) - c1 --> (voc) v + (c0 - c1) 37327 | else if ((details::e_add == o0) && (details::e_sub == o1)) 37328 | { 37329 | exprtk_debug(("(v + c0) - c1 --> (voc) v + (c0 - c1)\n")); 37330 | 37331 | return expr_gen.node_allocator_-> 37332 | template allocate_rc<typename details::voc_node<Type,details::add_op<Type> > >(v, c0 - c1); 37333 | } 37334 | // (v - c0) + c1 --> (voc) v - (c0 + c1) 37335 | else if ((details::e_sub == o0) && (details::e_add == o1)) 37336 | { 37337 | exprtk_debug(("(v - c0) + c1 --> (voc) v - (c0 + c1)\n")); 37338 | 37339 | return expr_gen.node_allocator_-> 37340 | template allocate_rc<typename details::voc_node<Type,details::add_op<Type> > >(v, c1 - c0); 37341 | } 37342 | // (v - c0) - c1 --> (voc) v - (c0 + c1) 37343 | else if ((details::e_sub == o0) && (details::e_sub == o1)) 37344 | { 37345 | exprtk_debug(("(v - c0) - c1 --> (voc) v - (c0 + c1)\n")); 37346 | 37347 | return expr_gen.node_allocator_-> 37348 | template allocate_rc<typename details::voc_node<Type,details::sub_op<Type> > >(v, c0 + c1); 37349 | } 37350 | // (v * c0) * c1 --> (voc) v * (c0 * c1) 37351 | else if ((details::e_mul == o0) && (details::e_mul == o1)) 37352 | { 37353 | exprtk_debug(("(v * c0) * c1 --> (voc) v * (c0 * c1)\n")); 37354 | 37355 | return expr_gen.node_allocator_-> 37356 | template allocate_rc<typename details::voc_node<Type,details::mul_op<Type> > >(v, c0 * c1); 37357 | } 37358 | // (v * c0) / c1 --> (voc) v * (c0 / c1) 37359 | else if ((details::e_mul == o0) && (details::e_div == o1)) 37360 | { 37361 | exprtk_debug(("(v * c0) / c1 --> (voc) v * (c0 / c1)\n")); 37362 | 37363 | return expr_gen.node_allocator_-> 37364 | template allocate_rc<typename details::voc_node<Type,details::mul_op<Type> > >(v, c0 / c1); 37365 | } 37366 | // (v / c0) * c1 --> (voc) v * (c1 / c0) 37367 | else if ((details::e_div == o0) && (details::e_mul == o1)) 37368 | { 37369 | exprtk_debug(("(v / c0) * c1 --> (voc) v * (c1 / c0)\n")); 37370 | 37371 | return expr_gen.node_allocator_-> 37372 | template allocate_rc<typename details::voc_node<Type,details::mul_op<Type> > >(v, c1 / c0); 37373 | } 37374 | // (v / c0) / c1 --> (voc) v / (c0 * c1) 37375 | else if ((details::e_div == o0) && (details::e_div == o1)) 37376 | { 37377 | exprtk_debug(("(v / c0) / c1 --> (voc) v / (c0 * c1)\n")); 37378 | 37379 | return expr_gen.node_allocator_-> 37380 | template allocate_rc<typename details::voc_node<Type,details::div_op<Type> > >(v, c0 * c1); 37381 | } 37382 | // (v ^ c0) ^ c1 --> (voc) v ^ (c0 * c1) 37383 | else if ((details::e_pow == o0) && (details::e_pow == o1)) 37384 | { 37385 | exprtk_debug(("(v ^ c0) ^ c1 --> (voc) v ^ (c0 * c1)\n")); 37386 | 37387 | return expr_gen.node_allocator_-> 37388 | template allocate_rc<typename details::voc_node<Type,details::pow_op<Type> > >(v, c0 * c1); 37389 | } 37390 | } 37391 | 37392 | const bool synthesis_result = 37393 | synthesize_sf3ext_expression::template compile<vtype, ctype, ctype> 37394 | (expr_gen, id(expr_gen, o0, o1), v, c0, c1, result); 37395 | 37396 | if (synthesis_result) 37397 | return result; 37398 | 37399 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37400 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37401 | 37402 | if (!expr_gen.valid_operator(o0,f0)) 37403 | return error_node(); 37404 | else if (!expr_gen.valid_operator(o1,f1)) 37405 | return error_node(); 37406 | else 37407 | return node_type::allocate(*(expr_gen.node_allocator_), v, c0, c1, f0, f1); 37408 | } 37409 | 37410 | static inline std::string id(expression_generator<Type>& expr_gen, 37411 | const details::operator_type o0, 37412 | const details::operator_type o1) 37413 | { 37414 | return details::build_string() 37415 | << "(t" << expr_gen.to_str(o0) 37416 | << "t)" << expr_gen.to_str(o1) 37417 | << "t" 37418 | } 37419 | }; 37420 | 37421 | struct synthesize_vococ_expression1 37422 | { 37423 | typedef typename vococ_t::type0 node_type; 37424 | 37425 | static inline expression_node_ptr process(expression_generator<Type>&, 37426 | const details::operator_type&, 37427 | expression_node_ptr (&)[2]) 37428 | { 37429 | // (v) o0 (c0 o1 c1) - Not possible. 37430 | exprtk_debug(("(v) o0 (c0 o1 c1) - Not possible.\n")); 37431 | return error_node(); 37432 | } 37433 | }; 37434 | 37435 | struct synthesize_vovovov_expression0 37436 | { 37437 | typedef typename vovovov_t::type0 node_type; 37438 | typedef typename vovovov_t::sf4_type sf4_type; 37439 | typedef typename node_type::T0 T0; 37440 | typedef typename node_type::T1 T1; 37441 | typedef typename node_type::T2 T2; 37442 | typedef typename node_type::T3 T3; 37443 | 37444 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37445 | const details::operator_type& operation, 37446 | expression_node_ptr (&branch)[2]) 37447 | { 37448 | // (v0 o0 v1) o1 (v2 o2 v3) 37449 | const details::vov_base_node<Type>* vov0 = static_cast<details::vov_base_node<Type>*>(branch[0]); 37450 | const details::vov_base_node<Type>* vov1 = static_cast<details::vov_base_node<Type>*>(branch[1]); 37451 | const Type& v0 = vov0->v0(); 37452 | const Type& v1 = vov0->v1(); 37453 | const Type& v2 = vov1->v0(); 37454 | const Type& v3 = vov1->v1(); 37455 | const details::operator_type o0 = vov0->operation(); 37456 | const details::operator_type o1 = operation; 37457 | const details::operator_type o2 = vov1->operation(); 37458 | 37459 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37460 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37461 | 37462 | expression_node_ptr result = error_node(); 37463 | 37464 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37465 | { 37466 | // (v0 / v1) * (v2 / v3) --> (vovovov) (v0 * v2) / (v1 * v3) 37467 | if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 37468 | { 37469 | const bool synthesis_result = 37470 | synthesize_sf4ext_expression:: 37471 | template compile<vtype, vtype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", v0, v2, v1, v3, result); 37472 | 37473 | exprtk_debug(("(v0 / v1) * (v2 / v3) --> (vovovov) (v0 * v2) / (v1 * v3)\n")); 37474 | 37475 | return (synthesis_result) ? result : error_node(); 37476 | } 37477 | // (v0 / v1) / (v2 / v3) --> (vovovov) (v0 * v3) / (v1 * v2) 37478 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 37479 | { 37480 | const bool synthesis_result = 37481 | synthesize_sf4ext_expression:: 37482 | template compile<vtype, vtype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", v0, v3, v1, v2, result); 37483 | 37484 | exprtk_debug(("(v0 / v1) / (v2 / v3) --> (vovovov) (v0 * v3) / (v1 * v2)\n")); 37485 | 37486 | return (synthesis_result) ? result : error_node(); 37487 | } 37488 | // (v0 + v1) / (v2 / v3) --> (vovovov) (v0 + v1) * (v3 / v2) 37489 | else if ((details::e_add == o0) && (details::e_div == o1) && (details::e_div == o2)) 37490 | { 37491 | const bool synthesis_result = 37492 | synthesize_sf4ext_expression:: 37493 | template compile<vtype, vtype, vtype, vtype>(expr_gen, "(t+t)*(t/t)", v0, v1, v3, v2, result); 37494 | 37495 | exprtk_debug(("(v0 + v1) / (v2 / v3) --> (vovovov) (v0 + v1) * (v3 / v2)\n")); 37496 | 37497 | return (synthesis_result) ? result : error_node(); 37498 | } 37499 | // (v0 - v1) / (v2 / v3) --> (vovovov) (v0 + v1) * (v3 / v2) 37500 | else if ((details::e_sub == o0) && (details::e_div == o1) && (details::e_div == o2)) 37501 | { 37502 | const bool synthesis_result = 37503 | synthesize_sf4ext_expression:: 37504 | template compile<vtype, vtype, vtype, vtype>(expr_gen, "(t-t)*(t/t)", v0, v1, v3, v2, result); 37505 | 37506 | exprtk_debug(("(v0 - v1) / (v2 / v3) --> (vovovov) (v0 - v1) * (v3 / v2)\n")); 37507 | 37508 | return (synthesis_result) ? result : error_node(); 37509 | } 37510 | // (v0 * v1) / (v2 / v3) --> (vovovov) ((v0 * v1) * v3) / v2 37511 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_div == o2)) 37512 | { 37513 | const bool synthesis_result = 37514 | synthesize_sf4ext_expression:: 37515 | template compile<vtype, vtype, vtype, vtype>(expr_gen, "((t*t)*t)/t", v0, v1, v3, v2, result); 37516 | 37517 | exprtk_debug(("(v0 * v1) / (v2 / v3) --> (vovovov) ((v0 * v1) * v3) / v2\n")); 37518 | 37519 | return (synthesis_result) ? result : error_node(); 37520 | } 37521 | } 37522 | 37523 | const bool synthesis_result = 37524 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 37525 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, v3, result); 37526 | 37527 | if (synthesis_result) 37528 | return result; 37529 | 37530 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37531 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37532 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 37533 | 37534 | if (!expr_gen.valid_operator(o0,f0)) 37535 | return error_node(); 37536 | else if (!expr_gen.valid_operator(o1,f1)) 37537 | return error_node(); 37538 | else if (!expr_gen.valid_operator(o2,f2)) 37539 | return error_node(); 37540 | else 37541 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, v3, f0, f1, f2); 37542 | } 37543 | 37544 | static inline std::string id(expression_generator<Type>& expr_gen, 37545 | const details::operator_type o0, 37546 | const details::operator_type o1, 37547 | const details::operator_type o2) 37548 | { 37549 | return details::build_string() 37550 | << "(t" << expr_gen.to_str(o0) 37551 | << "t)" << expr_gen.to_str(o1) 37552 | << "(t" << expr_gen.to_str(o2) 37553 | << "t)" 37554 | } 37555 | }; 37556 | 37557 | struct synthesize_vovovoc_expression0 37558 | { 37559 | typedef typename vovovoc_t::type0 node_type; 37560 | typedef typename vovovoc_t::sf4_type sf4_type; 37561 | typedef typename node_type::T0 T0; 37562 | typedef typename node_type::T1 T1; 37563 | typedef typename node_type::T2 T2; 37564 | typedef typename node_type::T3 T3; 37565 | 37566 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37567 | const details::operator_type& operation, 37568 | expression_node_ptr (&branch)[2]) 37569 | { 37570 | // (v0 o0 v1) o1 (v2 o2 c) 37571 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[0]); 37572 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[1]); 37573 | const Type& v0 = vov->v0(); 37574 | const Type& v1 = vov->v1(); 37575 | const Type& v2 = voc->v (); 37576 | const Type c = voc->c (); 37577 | const details::operator_type o0 = vov->operation(); 37578 | const details::operator_type o1 = operation; 37579 | const details::operator_type o2 = voc->operation(); 37580 | 37581 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37582 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37583 | 37584 | expression_node_ptr result = error_node(); 37585 | 37586 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37587 | { 37588 | // (v0 / v1) * (v2 / c) --> (vovovoc) (v0 * v2) / (v1 * c) 37589 | if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 37590 | { 37591 | const bool synthesis_result = 37592 | synthesize_sf4ext_expression:: 37593 | template compile<vtype, vtype, vtype, ctype>(expr_gen, "(t*t)/(t*t)", v0, v2, v1, c, result); 37594 | 37595 | exprtk_debug(("(v0 / v1) * (v2 / c) --> (vovovoc) (v0 * v2) / (v1 * c)\n")); 37596 | 37597 | return (synthesis_result) ? result : error_node(); 37598 | } 37599 | // (v0 / v1) / (v2 / c) --> (vocovov) (v0 * c) / (v1 * v2) 37600 | if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 37601 | { 37602 | const bool synthesis_result = 37603 | synthesize_sf4ext_expression:: 37604 | template compile<vtype, ctype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", v0, c, v1, v2, result); 37605 | 37606 | exprtk_debug(("(v0 / v1) / (v2 / c) --> (vocovov) (v0 * c) / (v1 * v2)\n")); 37607 | 37608 | return (synthesis_result) ? result : error_node(); 37609 | } 37610 | } 37611 | 37612 | const bool synthesis_result = 37613 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 37614 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, c, result); 37615 | 37616 | if (synthesis_result) 37617 | return result; 37618 | 37619 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37620 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37621 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 37622 | 37623 | if (!expr_gen.valid_operator(o0,f0)) 37624 | return error_node(); 37625 | else if (!expr_gen.valid_operator(o1,f1)) 37626 | return error_node(); 37627 | else if (!expr_gen.valid_operator(o2,f2)) 37628 | return error_node(); 37629 | else 37630 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, c, f0, f1, f2); 37631 | } 37632 | 37633 | static inline std::string id(expression_generator<Type>& expr_gen, 37634 | const details::operator_type o0, 37635 | const details::operator_type o1, 37636 | const details::operator_type o2) 37637 | { 37638 | return details::build_string() 37639 | << "(t" << expr_gen.to_str(o0) 37640 | << "t)" << expr_gen.to_str(o1) 37641 | << "(t" << expr_gen.to_str(o2) 37642 | << "t)" 37643 | } 37644 | }; 37645 | 37646 | struct synthesize_vovocov_expression0 37647 | { 37648 | typedef typename vovocov_t::type0 node_type; 37649 | typedef typename vovocov_t::sf4_type sf4_type; 37650 | typedef typename node_type::T0 T0; 37651 | typedef typename node_type::T1 T1; 37652 | typedef typename node_type::T2 T2; 37653 | typedef typename node_type::T3 T3; 37654 | 37655 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37656 | const details::operator_type& operation, 37657 | expression_node_ptr (&branch)[2]) 37658 | { 37659 | // (v0 o0 v1) o1 (c o2 v2) 37660 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[0]); 37661 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[1]); 37662 | const Type& v0 = vov->v0(); 37663 | const Type& v1 = vov->v1(); 37664 | const Type& v2 = cov->v (); 37665 | const Type c = cov->c (); 37666 | const details::operator_type o0 = vov->operation(); 37667 | const details::operator_type o1 = operation; 37668 | const details::operator_type o2 = cov->operation(); 37669 | 37670 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37671 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37672 | 37673 | expression_node_ptr result = error_node(); 37674 | 37675 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37676 | { 37677 | // (v0 / v1) * (c / v2) --> (vocovov) (v0 * c) / (v1 * v2) 37678 | if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 37679 | { 37680 | const bool synthesis_result = 37681 | synthesize_sf4ext_expression:: 37682 | template compile<vtype, ctype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", v0, c, v1, v2, result); 37683 | 37684 | exprtk_debug(("(v0 / v1) * (c / v2) --> (vocovov) (v0 * c) / (v1 * v2)\n")); 37685 | 37686 | return (synthesis_result) ? result : error_node(); 37687 | } 37688 | // (v0 / v1) / (c / v2) --> (vovovoc) (v0 * v2) / (v1 * c) 37689 | if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 37690 | { 37691 | const bool synthesis_result = 37692 | synthesize_sf4ext_expression:: 37693 | template compile<vtype, vtype, vtype, ctype>(expr_gen, "(t*t)/(t*t)", v0, v2, v1, c, result); 37694 | 37695 | exprtk_debug(("(v0 / v1) / (c / v2) --> (vovovoc) (v0 * v2) / (v1 * c)\n")); 37696 | 37697 | return (synthesis_result) ? result : error_node(); 37698 | } 37699 | } 37700 | 37701 | const bool synthesis_result = 37702 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 37703 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, c, v2, result); 37704 | 37705 | if (synthesis_result) 37706 | return result; 37707 | 37708 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37709 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37710 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 37711 | 37712 | if (!expr_gen.valid_operator(o0,f0)) 37713 | return error_node(); 37714 | else if (!expr_gen.valid_operator(o1,f1)) 37715 | return error_node(); 37716 | else if (!expr_gen.valid_operator(o2,f2)) 37717 | return error_node(); 37718 | else 37719 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, v2, f0, f1, f2); 37720 | } 37721 | 37722 | static inline std::string id(expression_generator<Type>& expr_gen, 37723 | const details::operator_type o0, 37724 | const details::operator_type o1, 37725 | const details::operator_type o2) 37726 | { 37727 | return details::build_string() 37728 | << "(t" << expr_gen.to_str(o0) 37729 | << "t)" << expr_gen.to_str(o1) 37730 | << "(t" << expr_gen.to_str(o2) 37731 | << "t)" 37732 | } 37733 | }; 37734 | 37735 | struct synthesize_vocovov_expression0 37736 | { 37737 | typedef typename vocovov_t::type0 node_type; 37738 | typedef typename vocovov_t::sf4_type sf4_type; 37739 | typedef typename node_type::T0 T0; 37740 | typedef typename node_type::T1 T1; 37741 | typedef typename node_type::T2 T2; 37742 | typedef typename node_type::T3 T3; 37743 | 37744 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37745 | const details::operator_type& operation, 37746 | expression_node_ptr (&branch)[2]) 37747 | { 37748 | // (v0 o0 c) o1 (v1 o2 v2) 37749 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[0]); 37750 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[1]); 37751 | const Type c = voc->c (); 37752 | const Type& v0 = voc->v (); 37753 | const Type& v1 = vov->v0(); 37754 | const Type& v2 = vov->v1(); 37755 | const details::operator_type o0 = voc->operation(); 37756 | const details::operator_type o1 = operation; 37757 | const details::operator_type o2 = vov->operation(); 37758 | 37759 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37760 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37761 | 37762 | expression_node_ptr result = error_node(); 37763 | 37764 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37765 | { 37766 | // (v0 / c) * (v1 / v2) --> (vovocov) (v0 * v1) / (c * v2) 37767 | if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 37768 | { 37769 | const bool synthesis_result = 37770 | synthesize_sf4ext_expression:: 37771 | template compile<vtype, vtype, ctype, vtype>(expr_gen, "(t*t)/(t*t)", v0, v1, c, v2, result); 37772 | 37773 | exprtk_debug(("(v0 / c) * (v1 / v2) --> (vovocov) (v0 * v1) / (c * v2)\n")); 37774 | 37775 | return (synthesis_result) ? result : error_node(); 37776 | } 37777 | // (v0 / c) / (v1 / v2) --> (vovocov) (v0 * v2) / (c * v1) 37778 | if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 37779 | { 37780 | const bool synthesis_result = 37781 | synthesize_sf4ext_expression:: 37782 | template compile<vtype, vtype, ctype, vtype>(expr_gen, "(t*t)/(t*t)", v0, v2, c, v1, result); 37783 | 37784 | exprtk_debug(("(v0 / c) / (v1 / v2) --> (vovocov) (v0 * v2) / (c * v1)\n")); 37785 | 37786 | return (synthesis_result) ? result : error_node(); 37787 | } 37788 | } 37789 | 37790 | const bool synthesis_result = 37791 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 37792 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c, v1, v2, result); 37793 | 37794 | if (synthesis_result) 37795 | return result; 37796 | 37797 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37798 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37799 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 37800 | 37801 | if (!expr_gen.valid_operator(o0,f0)) 37802 | return error_node(); 37803 | else if (!expr_gen.valid_operator(o1,f1)) 37804 | return error_node(); 37805 | else if (!expr_gen.valid_operator(o2,f2)) 37806 | return error_node(); 37807 | else 37808 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, v2, f0, f1, f2); 37809 | } 37810 | 37811 | static inline std::string id(expression_generator<Type>& expr_gen, 37812 | const details::operator_type o0, 37813 | const details::operator_type o1, 37814 | const details::operator_type o2) 37815 | { 37816 | return details::build_string() 37817 | << "(t" << expr_gen.to_str(o0) 37818 | << "t)" << expr_gen.to_str(o1) 37819 | << "(t" << expr_gen.to_str(o2) 37820 | << "t)" 37821 | } 37822 | }; 37823 | 37824 | struct synthesize_covovov_expression0 37825 | { 37826 | typedef typename covovov_t::type0 node_type; 37827 | typedef typename covovov_t::sf4_type sf4_type; 37828 | typedef typename node_type::T0 T0; 37829 | typedef typename node_type::T1 T1; 37830 | typedef typename node_type::T2 T2; 37831 | typedef typename node_type::T3 T3; 37832 | 37833 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37834 | const details::operator_type& operation, 37835 | expression_node_ptr (&branch)[2]) 37836 | { 37837 | // (c o0 v0) o1 (v1 o2 v2) 37838 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[0]); 37839 | const details::vov_base_node<Type>* vov = static_cast<details::vov_base_node<Type>*>(branch[1]); 37840 | const Type c = cov->c (); 37841 | const Type& v0 = cov->v (); 37842 | const Type& v1 = vov->v0(); 37843 | const Type& v2 = vov->v1(); 37844 | const details::operator_type o0 = cov->operation(); 37845 | const details::operator_type o1 = operation; 37846 | const details::operator_type o2 = vov->operation(); 37847 | 37848 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37849 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37850 | 37851 | expression_node_ptr result = error_node(); 37852 | 37853 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37854 | { 37855 | // (c / v0) * (v1 / v2) --> (covovov) (c * v1) / (v0 * v2) 37856 | if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 37857 | { 37858 | const bool synthesis_result = 37859 | synthesize_sf4ext_expression:: 37860 | template compile<ctype, vtype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", c, v1, v0, v2, result); 37861 | 37862 | exprtk_debug(("(c / v0) * (v1 / v2) --> (covovov) (c * v1) / (v0 * v2)\n")); 37863 | 37864 | return (synthesis_result) ? result : error_node(); 37865 | } 37866 | // (c / v0) / (v1 / v2) --> (covovov) (c * v2) / (v0 * v1) 37867 | if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 37868 | { 37869 | const bool synthesis_result = 37870 | synthesize_sf4ext_expression:: 37871 | template compile<ctype, vtype, vtype, vtype>(expr_gen, "(t*t)/(t*t)", c, v2, v0, v1, result); 37872 | 37873 | exprtk_debug(("(c / v0) / (v1 / v2) --> (covovov) (c * v2) / (v0 * v1)\n")); 37874 | 37875 | return (synthesis_result) ? result : error_node(); 37876 | } 37877 | } 37878 | 37879 | const bool synthesis_result = 37880 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 37881 | (expr_gen, id(expr_gen, o0, o1, o2), c, v0, v1, v2, result); 37882 | 37883 | if (synthesis_result) 37884 | return result; 37885 | 37886 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 37887 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 37888 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 37889 | 37890 | if (!expr_gen.valid_operator(o0,f0)) 37891 | return error_node(); 37892 | else if (!expr_gen.valid_operator(o1,f1)) 37893 | return error_node(); 37894 | else if (!expr_gen.valid_operator(o2,f2)) 37895 | return error_node(); 37896 | else 37897 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, v2, f0, f1, f2); 37898 | } 37899 | 37900 | static inline std::string id(expression_generator<Type>& expr_gen, 37901 | const details::operator_type o0, 37902 | const details::operator_type o1, 37903 | const details::operator_type o2) 37904 | { 37905 | return details::build_string() 37906 | << "(t" << expr_gen.to_str(o0) 37907 | << "t)" << expr_gen.to_str(o1) 37908 | << "(t" << expr_gen.to_str(o2) 37909 | << "t)" 37910 | } 37911 | }; 37912 | 37913 | struct synthesize_covocov_expression0 37914 | { 37915 | typedef typename covocov_t::type0 node_type; 37916 | typedef typename covocov_t::sf4_type sf4_type; 37917 | typedef typename node_type::T0 T0; 37918 | typedef typename node_type::T1 T1; 37919 | typedef typename node_type::T2 T2; 37920 | typedef typename node_type::T3 T3; 37921 | 37922 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 37923 | const details::operator_type& operation, 37924 | expression_node_ptr (&branch)[2]) 37925 | { 37926 | // (c0 o0 v0) o1 (c1 o2 v1) 37927 | const details::cov_base_node<Type>* cov0 = static_cast<details::cov_base_node<Type>*>(branch[0]); 37928 | const details::cov_base_node<Type>* cov1 = static_cast<details::cov_base_node<Type>*>(branch[1]); 37929 | const Type c0 = cov0->c(); 37930 | const Type& v0 = cov0->v(); 37931 | const Type c1 = cov1->c(); 37932 | const Type& v1 = cov1->v(); 37933 | const details::operator_type o0 = cov0->operation(); 37934 | const details::operator_type o1 = operation; 37935 | const details::operator_type o2 = cov1->operation(); 37936 | 37937 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 37938 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 37939 | 37940 | expression_node_ptr result = error_node(); 37941 | 37942 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 37943 | { 37944 | // (c0 + v0) + (c1 + v1) --> (covov) (c0 + c1) + v0 + v1 37945 | if ((details::e_add == o0) && (details::e_add == o1) && (details::e_add == o2)) 37946 | { 37947 | const bool synthesis_result = 37948 | synthesize_sf3ext_expression:: 37949 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)+t", (c0 + c1), v0, v1, result); 37950 | 37951 | exprtk_debug(("(c0 + v0) + (c1 + v1) --> (covov) (c0 + c1) + v0 + v1\n")); 37952 | 37953 | return (synthesis_result) ? result : error_node(); 37954 | } 37955 | // (c0 + v0) - (c1 + v1) --> (covov) (c0 - c1) + v0 - v1 37956 | else if ((details::e_add == o0) && (details::e_sub == o1) && (details::e_add == o2)) 37957 | { 37958 | const bool synthesis_result = 37959 | synthesize_sf3ext_expression:: 37960 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)-t", (c0 - c1), v0, v1, result); 37961 | 37962 | exprtk_debug(("(c0 + v0) - (c1 + v1) --> (covov) (c0 - c1) + v0 - v1\n")); 37963 | 37964 | return (synthesis_result) ? result : error_node(); 37965 | } 37966 | // (c0 - v0) - (c1 - v1) --> (covov) (c0 - c1) - v0 + v1 37967 | else if ((details::e_sub == o0) && (details::e_sub == o1) && (details::e_sub == o2)) 37968 | { 37969 | const bool synthesis_result = 37970 | synthesize_sf3ext_expression:: 37971 | template compile<ctype, vtype, vtype>(expr_gen, "(t-t)+t", (c0 - c1), v0, v1, result); 37972 | 37973 | exprtk_debug(("(c0 - v0) - (c1 - v1) --> (covov) (c0 - c1) - v0 + v1\n")); 37974 | 37975 | return (synthesis_result) ? result : error_node(); 37976 | } 37977 | // (c0 * v0) * (c1 * v1) --> (covov) (c0 * c1) * v0 * v1 37978 | else if ((details::e_mul == o0) && (details::e_mul == o1) && (details::e_mul == o2)) 37979 | { 37980 | const bool synthesis_result = 37981 | synthesize_sf3ext_expression:: 37982 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", (c0 * c1), v0, v1, result); 37983 | 37984 | exprtk_debug(("(c0 * v0) * (c1 * v1) --> (covov) (c0 * c1) * v0 * v1\n")); 37985 | 37986 | return (synthesis_result) ? result : error_node(); 37987 | } 37988 | // (c0 * v0) / (c1 * v1) --> (covov) (c0 / c1) * (v0 / v1) 37989 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_mul == o2)) 37990 | { 37991 | const bool synthesis_result = 37992 | synthesize_sf3ext_expression:: 37993 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 / c1), v0, v1, result); 37994 | 37995 | exprtk_debug(("(c0 * v0) / (c1 * v1) --> (covov) (c0 / c1) * (v0 / v1)\n")); 37996 | 37997 | return (synthesis_result) ? result : error_node(); 37998 | } 37999 | // (c0 / v0) * (c1 / v1) --> (covov) (c0 * c1) / (v0 * v1) 38000 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 38001 | { 38002 | const bool synthesis_result = 38003 | synthesize_sf3ext_expression:: 38004 | template compile<ctype, vtype, vtype>(expr_gen, "t/(t*t)", (c0 * c1), v0, v1, result); 38005 | 38006 | exprtk_debug(("(c0 / v0) * (c1 / v1) --> (covov) (c0 * c1) / (v0 * v1)\n")); 38007 | 38008 | return (synthesis_result) ? result : error_node(); 38009 | } 38010 | // (c0 / v0) / (c1 / v1) --> (covov) ((c0 / c1) * v1) / v0 38011 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 38012 | { 38013 | const bool synthesis_result = 38014 | synthesize_sf3ext_expression:: 38015 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 / c1), v1, v0, result); 38016 | 38017 | exprtk_debug(("(c0 / v0) / (c1 / v1) --> (covov) ((c0 / c1) * v1) / v0\n")); 38018 | 38019 | return (synthesis_result) ? result : error_node(); 38020 | } 38021 | // (c0 * v0) / (c1 / v1) --> (covov) (c0 / c1) * (v0 * v1) 38022 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_div == o2)) 38023 | { 38024 | const bool synthesis_result = 38025 | synthesize_sf3ext_expression:: 38026 | template compile<ctype, vtype, vtype>(expr_gen, "t*(t*t)", (c0 / c1), v0, v1, result); 38027 | 38028 | exprtk_debug(("(c0 * v0) / (c1 / v1) --> (covov) (c0 / c1) * (v0 * v1)\n")); 38029 | 38030 | return (synthesis_result) ? result : error_node(); 38031 | } 38032 | // (c0 / v0) / (c1 * v1) --> (covov) (c0 / c1) / (v0 * v1) 38033 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38034 | { 38035 | const bool synthesis_result = 38036 | synthesize_sf3ext_expression:: 38037 | template compile<ctype, vtype, vtype>(expr_gen, "t/(t*t)", (c0 / c1), v0, v1, result); 38038 | 38039 | exprtk_debug(("(c0 / v0) / (c1 * v1) --> (covov) (c0 / c1) / (v0 * v1)\n")); 38040 | 38041 | return (synthesis_result) ? result : error_node(); 38042 | } 38043 | // (c * v0) +/- (c * v1) --> (covov) c * (v0 +/- v1) 38044 | else if ( 38045 | (std::equal_to<T>()(c0,c1)) && 38046 | (details::e_mul == o0) && 38047 | (details::e_mul == o2) && 38048 | ( 38049 | (details::e_add == o1) || 38050 | (details::e_sub == o1) 38051 | ) 38052 | ) 38053 | { 38054 | std::string specfunc; 38055 | 38056 | switch (o1) 38057 | { 38058 | case details::e_add : specfunc = "t*(t+t)" break; 38059 | case details::e_sub : specfunc = "t*(t-t)" break; 38060 | default : return error_node(); 38061 | } 38062 | 38063 | const bool synthesis_result = 38064 | synthesize_sf3ext_expression:: 38065 | template compile<ctype, vtype, vtype>(expr_gen, specfunc, c0, v0, v1, result); 38066 | 38067 | exprtk_debug(("(c * v0) +/- (c * v1) --> (covov) c * (v0 +/- v1)\n")); 38068 | 38069 | return (synthesis_result) ? result : error_node(); 38070 | } 38071 | } 38072 | 38073 | const bool synthesis_result = 38074 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38075 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, c1, v1, result); 38076 | 38077 | if (synthesis_result) 38078 | return result; 38079 | 38080 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38081 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 38082 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 38083 | 38084 | if (!expr_gen.valid_operator(o0,f0)) 38085 | return error_node(); 38086 | else if (!expr_gen.valid_operator(o1,f1)) 38087 | return error_node(); 38088 | else if (!expr_gen.valid_operator(o2,f2)) 38089 | return error_node(); 38090 | else 38091 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, c1, v1, f0, f1, f2); 38092 | } 38093 | 38094 | static inline std::string id(expression_generator<Type>& expr_gen, 38095 | const details::operator_type o0, 38096 | const details::operator_type o1, 38097 | const details::operator_type o2) 38098 | { 38099 | return details::build_string() 38100 | << "(t" << expr_gen.to_str(o0) 38101 | << "t)" << expr_gen.to_str(o1) 38102 | << "(t" << expr_gen.to_str(o2) 38103 | << "t)" 38104 | } 38105 | }; 38106 | 38107 | struct synthesize_vocovoc_expression0 38108 | { 38109 | typedef typename vocovoc_t::type0 node_type; 38110 | typedef typename vocovoc_t::sf4_type sf4_type; 38111 | typedef typename node_type::T0 T0; 38112 | typedef typename node_type::T1 T1; 38113 | typedef typename node_type::T2 T2; 38114 | typedef typename node_type::T3 T3; 38115 | 38116 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38117 | const details::operator_type& operation, 38118 | expression_node_ptr (&branch)[2]) 38119 | { 38120 | // (v0 o0 c0) o1 (v1 o2 c1) 38121 | const details::voc_base_node<Type>* voc0 = static_cast<details::voc_base_node<Type>*>(branch[0]); 38122 | const details::voc_base_node<Type>* voc1 = static_cast<details::voc_base_node<Type>*>(branch[1]); 38123 | const Type c0 = voc0->c(); 38124 | const Type& v0 = voc0->v(); 38125 | const Type c1 = voc1->c(); 38126 | const Type& v1 = voc1->v(); 38127 | const details::operator_type o0 = voc0->operation(); 38128 | const details::operator_type o1 = operation; 38129 | const details::operator_type o2 = voc1->operation(); 38130 | 38131 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 38132 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38133 | 38134 | expression_node_ptr result = error_node(); 38135 | 38136 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 38137 | { 38138 | // (v0 + c0) + (v1 + c1) --> (covov) (c0 + c1) + v0 + v1 38139 | if ((details::e_add == o0) && (details::e_add == o1) && (details::e_add == o2)) 38140 | { 38141 | const bool synthesis_result = 38142 | synthesize_sf3ext_expression:: 38143 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)+t", (c0 + c1), v0, v1, result); 38144 | 38145 | exprtk_debug(("(v0 + c0) + (v1 + c1) --> (covov) (c0 + c1) + v0 + v1\n")); 38146 | 38147 | return (synthesis_result) ? result : error_node(); 38148 | } 38149 | // (v0 + c0) - (v1 + c1) --> (covov) (c0 - c1) + v0 - v1 38150 | else if ((details::e_add == o0) && (details::e_sub == o1) && (details::e_add == o2)) 38151 | { 38152 | const bool synthesis_result = 38153 | synthesize_sf3ext_expression:: 38154 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)-t", (c0 - c1), v0, v1, result); 38155 | 38156 | exprtk_debug(("(v0 + c0) - (v1 + c1) --> (covov) (c0 - c1) + v0 - v1\n")); 38157 | 38158 | return (synthesis_result) ? result : error_node(); 38159 | } 38160 | // (v0 - c0) - (v1 - c1) --> (covov) (c1 - c0) + v0 - v1 38161 | else if ((details::e_sub == o0) && (details::e_sub == o1) && (details::e_sub == o2)) 38162 | { 38163 | const bool synthesis_result = 38164 | synthesize_sf3ext_expression:: 38165 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)-t", (c1 - c0), v0, v1, result); 38166 | 38167 | exprtk_debug(("(v0 - c0) - (v1 - c1) --> (covov) (c1 - c0) + v0 - v1\n")); 38168 | 38169 | return (synthesis_result) ? result : error_node(); 38170 | } 38171 | // (v0 * c0) * (v1 * c1) --> (covov) (c0 * c1) * v0 * v1 38172 | else if ((details::e_mul == o0) && (details::e_mul == o1) && (details::e_mul == o2)) 38173 | { 38174 | const bool synthesis_result = 38175 | synthesize_sf3ext_expression:: 38176 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", (c0 * c1), v0, v1, result); 38177 | 38178 | exprtk_debug(("(v0 * c0) * (v1 * c1) --> (covov) (c0 * c1) * v0 * v1\n")); 38179 | 38180 | return (synthesis_result) ? result : error_node(); 38181 | } 38182 | // (v0 * c0) / (v1 * c1) --> (covov) (c0 / c1) * (v0 / v1) 38183 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38184 | { 38185 | const bool synthesis_result = 38186 | synthesize_sf3ext_expression:: 38187 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 / c1), v0, v1, result); 38188 | 38189 | exprtk_debug(("(v0 * c0) / (v1 * c1) --> (covov) (c0 / c1) * (v0 / v1)\n")); 38190 | 38191 | return (synthesis_result) ? result : error_node(); 38192 | } 38193 | // (v0 / c0) * (v1 / c1) --> (covov) (1 / (c0 * c1)) * v0 * v1 38194 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 38195 | { 38196 | const bool synthesis_result = 38197 | synthesize_sf3ext_expression:: 38198 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", Type(1) / (c0 * c1), v0, v1, result); 38199 | 38200 | exprtk_debug(("(v0 / c0) * (v1 / c1) --> (covov) (1 / (c0 * c1)) * v0 * v1\n")); 38201 | 38202 | return (synthesis_result) ? result : error_node(); 38203 | } 38204 | // (v0 / c0) / (v1 / c1) --> (covov) ((c1 / c0) * v0) / v1 38205 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 38206 | { 38207 | const bool synthesis_result = 38208 | synthesize_sf3ext_expression:: 38209 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c1 / c0), v0, v1, result); 38210 | 38211 | exprtk_debug(("(v0 / c0) / (v1 / c1) --> (covov) ((c1 / c0) * v0) / v1\n")); 38212 | 38213 | return (synthesis_result) ? result : error_node(); 38214 | } 38215 | // (v0 * c0) / (v1 / c1) --> (covov) (c0 * c1) * (v0 / v1) 38216 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_div == o2)) 38217 | { 38218 | const bool synthesis_result = 38219 | synthesize_sf3ext_expression:: 38220 | template compile<ctype, vtype, vtype>(expr_gen, "t*(t/t)", (c0 * c1), v0, v1, result); 38221 | 38222 | exprtk_debug(("(v0 * c0) / (v1 / c1) --> (covov) (c0 * c1) * (v0 / v1)\n")); 38223 | 38224 | return (synthesis_result) ? result : error_node(); 38225 | } 38226 | // (v0 / c0) / (v1 * c1) --> (covov) (1 / (c0 * c1)) * v0 / v1 38227 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38228 | { 38229 | const bool synthesis_result = 38230 | synthesize_sf3ext_expression:: 38231 | template compile<ctype, vtype, vtype>(expr_gen, "t*(t/t)", Type(1) / (c0 * c1), v0, v1, result); 38232 | 38233 | exprtk_debug(("(v0 / c0) / (v1 * c1) --> (covov) (1 / (c0 * c1)) * v0 / v1\n")); 38234 | 38235 | return (synthesis_result) ? result : error_node(); 38236 | } 38237 | // (v0 / c0) * (v1 + c1) --> (vocovoc) (v0 * (1 / c0)) * (v1 + c1) 38238 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_add == o2)) 38239 | { 38240 | const bool synthesis_result = 38241 | synthesize_sf4ext_expression:: 38242 | template compile<vtype, ctype, vtype, ctype>(expr_gen, "(t*t)*(t+t)", v0, T(1) / c0, v1, c1, result); 38243 | 38244 | exprtk_debug(("(v0 / c0) * (v1 + c1) --> (vocovoc) (v0 * (1 / c0)) * (v1 + c1)\n")); 38245 | 38246 | return (synthesis_result) ? result : error_node(); 38247 | } 38248 | // (v0 / c0) * (v1 - c1) --> (vocovoc) (v0 * (1 / c0)) * (v1 - c1) 38249 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_sub == o2)) 38250 | { 38251 | const bool synthesis_result = 38252 | synthesize_sf4ext_expression:: 38253 | template compile<vtype, ctype, vtype, ctype>(expr_gen, "(t*t)*(t-t)", v0, T(1) / c0, v1, c1, result); 38254 | 38255 | exprtk_debug(("(v0 / c0) * (v1 - c1) --> (vocovoc) (v0 * (1 / c0)) * (v1 - c1)\n")); 38256 | 38257 | return (synthesis_result) ? result : error_node(); 38258 | } 38259 | // (v0 * c) +/- (v1 * c) --> (covov) c * (v0 +/- v1) 38260 | else if ( 38261 | (std::equal_to<T>()(c0,c1)) && 38262 | (details::e_mul == o0) && 38263 | (details::e_mul == o2) && 38264 | ( 38265 | (details::e_add == o1) || 38266 | (details::e_sub == o1) 38267 | ) 38268 | ) 38269 | { 38270 | std::string specfunc; 38271 | 38272 | switch (o1) 38273 | { 38274 | case details::e_add : specfunc = "t*(t+t)" break; 38275 | case details::e_sub : specfunc = "t*(t-t)" break; 38276 | default : return error_node(); 38277 | } 38278 | 38279 | const bool synthesis_result = 38280 | synthesize_sf3ext_expression:: 38281 | template compile<ctype, vtype, vtype>(expr_gen, specfunc, c0, v0, v1, result); 38282 | 38283 | exprtk_debug(("(v0 * c) +/- (v1 * c) --> (covov) c * (v0 +/- v1)\n")); 38284 | 38285 | return (synthesis_result) ? result : error_node(); 38286 | } 38287 | // (v0 / c) +/- (v1 / c) --> (vovoc) (v0 +/- v1) / c 38288 | else if ( 38289 | (std::equal_to<T>()(c0,c1)) && 38290 | (details::e_div == o0) && 38291 | (details::e_div == o2) && 38292 | ( 38293 | (details::e_add == o1) || 38294 | (details::e_sub == o1) 38295 | ) 38296 | ) 38297 | { 38298 | std::string specfunc; 38299 | 38300 | switch (o1) 38301 | { 38302 | case details::e_add : specfunc = "(t+t)/t" break; 38303 | case details::e_sub : specfunc = "(t-t)/t" break; 38304 | default : return error_node(); 38305 | } 38306 | 38307 | const bool synthesis_result = 38308 | synthesize_sf3ext_expression:: 38309 | template compile<vtype, vtype, ctype>(expr_gen, specfunc, v0, v1, c0, result); 38310 | 38311 | exprtk_debug(("(v0 / c) +/- (v1 / c) --> (vovoc) (v0 +/- v1) / c\n")); 38312 | 38313 | return (synthesis_result) ? result : error_node(); 38314 | } 38315 | } 38316 | 38317 | const bool synthesis_result = 38318 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38319 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, v1, c1, result); 38320 | 38321 | if (synthesis_result) 38322 | return result; 38323 | 38324 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38325 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 38326 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 38327 | 38328 | if (!expr_gen.valid_operator(o0,f0)) 38329 | return error_node(); 38330 | else if (!expr_gen.valid_operator(o1,f1)) 38331 | return error_node(); 38332 | else if (!expr_gen.valid_operator(o2,f2)) 38333 | return error_node(); 38334 | else 38335 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, v1, c1, f0, f1, f2); 38336 | } 38337 | 38338 | static inline std::string id(expression_generator<Type>& expr_gen, 38339 | const details::operator_type o0, 38340 | const details::operator_type o1, 38341 | const details::operator_type o2) 38342 | { 38343 | return details::build_string() 38344 | << "(t" << expr_gen.to_str(o0) 38345 | << "t)" << expr_gen.to_str(o1) 38346 | << "(t" << expr_gen.to_str(o2) 38347 | << "t)" 38348 | } 38349 | }; 38350 | 38351 | struct synthesize_covovoc_expression0 38352 | { 38353 | typedef typename covovoc_t::type0 node_type; 38354 | typedef typename covovoc_t::sf4_type sf4_type; 38355 | typedef typename node_type::T0 T0; 38356 | typedef typename node_type::T1 T1; 38357 | typedef typename node_type::T2 T2; 38358 | typedef typename node_type::T3 T3; 38359 | 38360 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38361 | const details::operator_type& operation, 38362 | expression_node_ptr (&branch)[2]) 38363 | { 38364 | // (c0 o0 v0) o1 (v1 o2 c1) 38365 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[0]); 38366 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[1]); 38367 | const Type c0 = cov->c(); 38368 | const Type& v0 = cov->v(); 38369 | const Type c1 = voc->c(); 38370 | const Type& v1 = voc->v(); 38371 | const details::operator_type o0 = cov->operation(); 38372 | const details::operator_type o1 = operation; 38373 | const details::operator_type o2 = voc->operation(); 38374 | 38375 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 38376 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38377 | 38378 | expression_node_ptr result = error_node(); 38379 | 38380 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 38381 | { 38382 | // (c0 + v0) + (v1 + c1) --> (covov) (c0 + c1) + v0 + v1 38383 | if ((details::e_add == o0) && (details::e_add == o1) && (details::e_add == o2)) 38384 | { 38385 | const bool synthesis_result = 38386 | synthesize_sf3ext_expression:: 38387 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)+t", (c0 + c1), v0, v1, result); 38388 | 38389 | exprtk_debug(("(c0 + v0) + (v1 + c1) --> (covov) (c0 + c1) + v0 + v1\n")); 38390 | 38391 | return (synthesis_result) ? result : error_node(); 38392 | } 38393 | // (c0 + v0) - (v1 + c1) --> (covov) (c0 - c1) + v0 - v1 38394 | else if ((details::e_add == o0) && (details::e_sub == o1) && (details::e_add == o2)) 38395 | { 38396 | const bool synthesis_result = 38397 | synthesize_sf3ext_expression:: 38398 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)-t", (c0 - c1), v0, v1, result); 38399 | 38400 | exprtk_debug(("(c0 + v0) - (v1 + c1) --> (covov) (c0 - c1) + v0 - v1\n")); 38401 | 38402 | return (synthesis_result) ? result : error_node(); 38403 | } 38404 | // (c0 - v0) - (v1 - c1) --> (covov) (c0 + c1) - v0 - v1 38405 | else if ((details::e_sub == o0) && (details::e_sub == o1) && (details::e_sub == o2)) 38406 | { 38407 | const bool synthesis_result = 38408 | synthesize_sf3ext_expression:: 38409 | template compile<ctype, vtype, vtype>(expr_gen, "t-(t+t)", (c0 + c1), v0, v1, result); 38410 | 38411 | exprtk_debug(("(c0 - v0) - (v1 - c1) --> (covov) (c0 + c1) - v0 - v1\n")); 38412 | 38413 | return (synthesis_result) ? result : error_node(); 38414 | } 38415 | // (c0 * v0) * (v1 * c1) --> (covov) (c0 * c1) * v0 * v1 38416 | else if ((details::e_mul == o0) && (details::e_mul == o1) && (details::e_mul == o2)) 38417 | { 38418 | const bool synthesis_result = 38419 | synthesize_sf3ext_expression:: 38420 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", (c0 * c1), v0, v1, result); 38421 | 38422 | exprtk_debug(("(c0 * v0) * (v1 * c1) --> (covov) (c0 * c1) * v0 * v1\n")); 38423 | 38424 | return (synthesis_result) ? result : error_node(); 38425 | } 38426 | // (c0 * v0) / (v1 * c1) --> (covov) (c0 / c1) * (v0 / v1) 38427 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38428 | { 38429 | const bool synthesis_result = 38430 | synthesize_sf3ext_expression:: 38431 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 / c1), v0, v1, result); 38432 | 38433 | exprtk_debug(("(c0 * v0) / (v1 * c1) --> (covov) (c0 / c1) * (v0 / v1)\n")); 38434 | 38435 | return (synthesis_result) ? result : error_node(); 38436 | } 38437 | // (c0 / v0) * (v1 / c1) --> (covov) (c0 / c1) * (v1 / v0) 38438 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 38439 | { 38440 | const bool synthesis_result = 38441 | synthesize_sf3ext_expression:: 38442 | template compile<ctype, vtype, vtype>(expr_gen, "t*(t/t)", (c0 / c1), v1, v0, result); 38443 | 38444 | exprtk_debug(("(c0 / v0) * (v1 / c1) --> (covov) (c0 / c1) * (v1 / v0)\n")); 38445 | 38446 | return (synthesis_result) ? result : error_node(); 38447 | } 38448 | // (c0 / v0) / (v1 / c1) --> (covov) (c0 * c1) / (v0 * v1) 38449 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 38450 | { 38451 | const bool synthesis_result = 38452 | synthesize_sf3ext_expression:: 38453 | template compile<ctype, vtype, vtype>(expr_gen, "t/(t*t)", (c0 * c1), v0, v1, result); 38454 | 38455 | exprtk_debug(("(c0 / v0) / (v1 / c1) --> (covov) (c0 * c1) / (v0 * v1)\n")); 38456 | 38457 | return (synthesis_result) ? result : error_node(); 38458 | } 38459 | // (c0 * v0) / (v1 / c1) --> (covov) (c0 * c1) * (v0 / v1) 38460 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_div == o2)) 38461 | { 38462 | const bool synthesis_result = 38463 | synthesize_sf3ext_expression:: 38464 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 * c1), v0, v1, result); 38465 | 38466 | exprtk_debug(("(c0 * v0) / (v1 / c1) --> (covov) (c0 * c1) * (v0 / v1)\n")); 38467 | 38468 | return (synthesis_result) ? result : error_node(); 38469 | } 38470 | // (c0 / v0) / (v1 * c1) --> (covov) (c0 / c1) / (v0 * v1) 38471 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38472 | { 38473 | const bool synthesis_result = 38474 | synthesize_sf3ext_expression:: 38475 | template compile<ctype, vtype, vtype>(expr_gen, "t/(t*t)", (c0 / c1), v0, v1, result); 38476 | 38477 | exprtk_debug(("(c0 / v0) / (v1 * c1) --> (covov) (c0 / c1) / (v0 * v1)\n")); 38478 | 38479 | return (synthesis_result) ? result : error_node(); 38480 | } 38481 | // (c * v0) +/- (v1 * c) --> (covov) c * (v0 +/- v1) 38482 | else if ( 38483 | (std::equal_to<T>()(c0,c1)) && 38484 | (details::e_mul == o0) && 38485 | (details::e_mul == o2) && 38486 | ( 38487 | (details::e_add == o1) || 38488 | (details::e_sub == o1) 38489 | ) 38490 | ) 38491 | { 38492 | std::string specfunc; 38493 | 38494 | switch (o1) 38495 | { 38496 | case details::e_add : specfunc = "t*(t+t)" break; 38497 | case details::e_sub : specfunc = "t*(t-t)" break; 38498 | default : return error_node(); 38499 | } 38500 | 38501 | const bool synthesis_result = 38502 | synthesize_sf3ext_expression:: 38503 | template compile<ctype, vtype, vtype>(expr_gen, specfunc, c0, v0, v1, result); 38504 | 38505 | exprtk_debug(("(c * v0) +/- (v1 * c) --> (covov) c * (v0 +/- v1)\n")); 38506 | 38507 | return (synthesis_result) ? result : error_node(); 38508 | } 38509 | } 38510 | 38511 | const bool synthesis_result = 38512 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38513 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, v1, c1, result); 38514 | 38515 | if (synthesis_result) 38516 | return result; 38517 | 38518 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38519 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 38520 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 38521 | 38522 | if (!expr_gen.valid_operator(o0,f0)) 38523 | return error_node(); 38524 | else if (!expr_gen.valid_operator(o1,f1)) 38525 | return error_node(); 38526 | else if (!expr_gen.valid_operator(o2,f2)) 38527 | return error_node(); 38528 | else 38529 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, v1, c1, f0, f1, f2); 38530 | } 38531 | 38532 | static inline std::string id(expression_generator<Type>& expr_gen, 38533 | const details::operator_type o0, 38534 | const details::operator_type o1, 38535 | const details::operator_type o2) 38536 | { 38537 | return details::build_string() 38538 | << "(t" << expr_gen.to_str(o0) 38539 | << "t)" << expr_gen.to_str(o1) 38540 | << "(t" << expr_gen.to_str(o2) 38541 | << "t)" 38542 | } 38543 | }; 38544 | 38545 | struct synthesize_vococov_expression0 38546 | { 38547 | typedef typename vococov_t::type0 node_type; 38548 | typedef typename vococov_t::sf4_type sf4_type; 38549 | typedef typename node_type::T0 T0; 38550 | typedef typename node_type::T1 T1; 38551 | typedef typename node_type::T2 T2; 38552 | typedef typename node_type::T3 T3; 38553 | 38554 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38555 | const details::operator_type& operation, 38556 | expression_node_ptr (&branch)[2]) 38557 | { 38558 | // (v0 o0 c0) o1 (c1 o2 v1) 38559 | const details::voc_base_node<Type>* voc = static_cast<details::voc_base_node<Type>*>(branch[0]); 38560 | const details::cov_base_node<Type>* cov = static_cast<details::cov_base_node<Type>*>(branch[1]); 38561 | const Type c0 = voc->c(); 38562 | const Type& v0 = voc->v(); 38563 | const Type c1 = cov->c(); 38564 | const Type& v1 = cov->v(); 38565 | const details::operator_type o0 = voc->operation(); 38566 | const details::operator_type o1 = operation; 38567 | const details::operator_type o2 = cov->operation(); 38568 | 38569 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 38570 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38571 | 38572 | expression_node_ptr result = error_node(); 38573 | 38574 | if (expr_gen.parser_->settings_.strength_reduction_enabled()) 38575 | { 38576 | // (v0 + c0) + (c1 + v1) --> (covov) (c0 + c1) + v0 + v1 38577 | if ((details::e_add == o0) && (details::e_add == o1) && (details::e_add == o2)) 38578 | { 38579 | const bool synthesis_result = 38580 | synthesize_sf3ext_expression:: 38581 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)+t", (c0 + c1), v0, v1, result); 38582 | 38583 | exprtk_debug(("(v0 + c0) + (c1 + v1) --> (covov) (c0 + c1) + v0 + v1\n")); 38584 | 38585 | return (synthesis_result) ? result : error_node(); 38586 | } 38587 | // (v0 + c0) - (c1 + v1) --> (covov) (c0 - c1) + v0 - v1 38588 | else if ((details::e_add == o0) && (details::e_sub == o1) && (details::e_add == o2)) 38589 | { 38590 | const bool synthesis_result = 38591 | synthesize_sf3ext_expression:: 38592 | template compile<ctype, vtype, vtype>(expr_gen, "(t+t)-t", (c0 - c1), v0, v1, result); 38593 | 38594 | exprtk_debug(("(v0 + c0) - (c1 + v1) --> (covov) (c0 - c1) + v0 - v1\n")); 38595 | 38596 | return (synthesis_result) ? result : error_node(); 38597 | } 38598 | // (v0 - c0) - (c1 - v1) --> (vovoc) v0 + v1 - (c1 + c0) 38599 | else if ((details::e_sub == o0) && (details::e_sub == o1) && (details::e_sub == o2)) 38600 | { 38601 | const bool synthesis_result = 38602 | synthesize_sf3ext_expression:: 38603 | template compile<vtype, vtype, ctype>(expr_gen, "(t+t)-t", v0, v1, (c1 + c0), result); 38604 | 38605 | exprtk_debug(("(v0 - c0) - (c1 - v1) --> (vovoc) v0 + v1 - (c1 + c0)\n")); 38606 | 38607 | return (synthesis_result) ? result : error_node(); 38608 | } 38609 | // (v0 * c0) * (c1 * v1) --> (covov) (c0 * c1) * v0 * v1 38610 | else if ((details::e_mul == o0) && (details::e_mul == o1) && (details::e_mul == o2)) 38611 | { 38612 | const bool synthesis_result = 38613 | synthesize_sf3ext_expression:: 38614 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", (c0 * c1), v0, v1, result); 38615 | 38616 | exprtk_debug(("(v0 * c0) * (c1 * v1) --> (covov) (c0 * c1) * v0 * v1\n")); 38617 | 38618 | return (synthesis_result) ? result : error_node(); 38619 | } 38620 | // (v0 * c0) / (c1 * v1) --> (covov) (c0 / c1) * (v0 * v1) 38621 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38622 | { 38623 | const bool synthesis_result = 38624 | synthesize_sf3ext_expression:: 38625 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c0 / c1), v0, v1, result); 38626 | 38627 | exprtk_debug(("(v0 * c0) / (c1 * v1) --> (covov) (c0 / c1) * (v0 * v1)\n")); 38628 | 38629 | return (synthesis_result) ? result : error_node(); 38630 | } 38631 | // (v0 / c0) * (c1 / v1) --> (covov) (c1 / c0) * (v0 / v1) 38632 | else if ((details::e_div == o0) && (details::e_mul == o1) && (details::e_div == o2)) 38633 | { 38634 | const bool synthesis_result = 38635 | synthesize_sf3ext_expression:: 38636 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", (c1 / c0), v0, v1, result); 38637 | 38638 | exprtk_debug(("(v0 / c0) * (c1 / v1) --> (covov) (c1 / c0) * (v0 / v1)\n")); 38639 | 38640 | return (synthesis_result) ? result : error_node(); 38641 | } 38642 | // (v0 * c0) / (c1 / v1) --> (covov) (c0 / c1) * (v0 * v1) 38643 | else if ((details::e_mul == o0) && (details::e_div == o1) && (details::e_div == o2)) 38644 | { 38645 | const bool synthesis_result = 38646 | synthesize_sf3ext_expression:: 38647 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)*t", (c0 / c1), v0, v1, result); 38648 | 38649 | exprtk_debug(("(v0 * c0) / (c1 / v1) --> (covov) (c0 / c1) * (v0 * v1)\n")); 38650 | 38651 | return (synthesis_result) ? result : error_node(); 38652 | } 38653 | // (v0 / c0) / (c1 * v1) --> (covov) (1 / (c0 * c1)) * (v0 / v1) 38654 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_mul == o2)) 38655 | { 38656 | const bool synthesis_result = 38657 | synthesize_sf3ext_expression:: 38658 | template compile<ctype, vtype, vtype>(expr_gen, "(t*t)/t", Type(1) / (c0 * c1), v0, v1, result); 38659 | 38660 | exprtk_debug(("(v0 / c0) / (c1 * v1) --> (covov) (1 / (c0 * c1)) * (v0 / v1)\n")); 38661 | 38662 | return (synthesis_result) ? result : error_node(); 38663 | } 38664 | // (v0 / c0) / (c1 / v1) --> (vovoc) (v0 * v1) * (1 / (c0 * c1)) 38665 | else if ((details::e_div == o0) && (details::e_div == o1) && (details::e_div == o2)) 38666 | { 38667 | const bool synthesis_result = 38668 | synthesize_sf3ext_expression:: 38669 | template compile<vtype, vtype, ctype>(expr_gen, "(t*t)*t", v0, v1, Type(1) / (c0 * c1), result); 38670 | 38671 | exprtk_debug(("(v0 / c0) / (c1 / v1) --> (vovoc) (v0 * v1) * (1 / (c0 * c1))\n")); 38672 | 38673 | return (synthesis_result) ? result : error_node(); 38674 | } 38675 | // (v0 * c) +/- (c * v1) --> (covov) c * (v0 +/- v1) 38676 | else if ( 38677 | (std::equal_to<T>()(c0,c1)) && 38678 | (details::e_mul == o0) && 38679 | (details::e_mul == o2) && 38680 | ( 38681 | (details::e_add == o1) || (details::e_sub == o1) 38682 | ) 38683 | ) 38684 | { 38685 | std::string specfunc; 38686 | 38687 | switch (o1) 38688 | { 38689 | case details::e_add : specfunc = "t*(t+t)" break; 38690 | case details::e_sub : specfunc = "t*(t-t)" break; 38691 | default : return error_node(); 38692 | } 38693 | 38694 | const bool synthesis_result = 38695 | synthesize_sf3ext_expression:: 38696 | template compile<ctype, vtype, vtype>(expr_gen, specfunc, c0, v0, v1, result); 38697 | 38698 | exprtk_debug(("(v0 * c) +/- (c * v1) --> (covov) c * (v0 +/- v1)\n")); 38699 | 38700 | return (synthesis_result) ? result : error_node(); 38701 | } 38702 | } 38703 | 38704 | const bool synthesis_result = 38705 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38706 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, c1, v1, result); 38707 | 38708 | if (synthesis_result) 38709 | return result; 38710 | 38711 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38712 | binary_functor_t f1 = reinterpret_cast<binary_functor_t>(0); 38713 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 38714 | 38715 | if (!expr_gen.valid_operator(o0,f0)) 38716 | return error_node(); 38717 | else if (!expr_gen.valid_operator(o1,f1)) 38718 | return error_node(); 38719 | else if (!expr_gen.valid_operator(o2,f2)) 38720 | return error_node(); 38721 | else 38722 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, c1, v1, f0, f1, f2); 38723 | } 38724 | 38725 | static inline std::string id(expression_generator<Type>& expr_gen, 38726 | const details::operator_type o0, 38727 | const details::operator_type o1, 38728 | const details::operator_type o2) 38729 | { 38730 | return details::build_string() 38731 | << "(t" << expr_gen.to_str(o0) 38732 | << "t)" << expr_gen.to_str(o1) 38733 | << "(t" << expr_gen.to_str(o2) 38734 | << "t)" 38735 | } 38736 | }; 38737 | 38738 | struct synthesize_vovovov_expression1 38739 | { 38740 | typedef typename vovovov_t::type1 node_type; 38741 | typedef typename vovovov_t::sf4_type sf4_type; 38742 | typedef typename node_type::T0 T0; 38743 | typedef typename node_type::T1 T1; 38744 | typedef typename node_type::T2 T2; 38745 | typedef typename node_type::T3 T3; 38746 | 38747 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38748 | const details::operator_type& operation, 38749 | expression_node_ptr (&branch)[2]) 38750 | { 38751 | // v0 o0 (v1 o1 (v2 o2 v3)) 38752 | typedef typename synthesize_vovov_expression1::node_type lcl_vovov_t; 38753 | 38754 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[1]); 38755 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 38756 | const Type& v1 = vovov->t0(); 38757 | const Type& v2 = vovov->t1(); 38758 | const Type& v3 = vovov->t2(); 38759 | const details::operator_type o0 = operation; 38760 | const details::operator_type o1 = expr_gen.get_operator(vovov->f0()); 38761 | const details::operator_type o2 = expr_gen.get_operator(vovov->f1()); 38762 | 38763 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38764 | binary_functor_t f1 = vovov->f0(); 38765 | binary_functor_t f2 = vovov->f1(); 38766 | 38767 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38768 | 38769 | expression_node_ptr result = error_node(); 38770 | 38771 | const bool synthesis_result = 38772 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38773 | (expr_gen,id(expr_gen, o0, o1, o2), v0, v1, v2, v3, result); 38774 | 38775 | if (synthesis_result) 38776 | return result; 38777 | else if (!expr_gen.valid_operator(o0,f0)) 38778 | return error_node(); 38779 | 38780 | exprtk_debug(("v0 o0 (v1 o1 (v2 o2 v3))\n")); 38781 | 38782 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, v3, f0, f1, f2); 38783 | } 38784 | 38785 | static inline std::string id(expression_generator<Type>& expr_gen, 38786 | const details::operator_type o0, 38787 | const details::operator_type o1, 38788 | const details::operator_type o2) 38789 | { 38790 | return details::build_string() 38791 | << "t" << expr_gen.to_str(o0) 38792 | << "(t" << expr_gen.to_str(o1) 38793 | << "(t" << expr_gen.to_str(o2) 38794 | << "t))" 38795 | } 38796 | }; 38797 | 38798 | struct synthesize_vovovoc_expression1 38799 | { 38800 | typedef typename vovovoc_t::type1 node_type; 38801 | typedef typename vovovoc_t::sf4_type sf4_type; 38802 | typedef typename node_type::T0 T0; 38803 | typedef typename node_type::T1 T1; 38804 | typedef typename node_type::T2 T2; 38805 | typedef typename node_type::T3 T3; 38806 | 38807 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38808 | const details::operator_type& operation, 38809 | expression_node_ptr (&branch)[2]) 38810 | { 38811 | // v0 o0 (v1 o1 (v2 o2 c)) 38812 | typedef typename synthesize_vovoc_expression1::node_type lcl_vovoc_t; 38813 | 38814 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[1]); 38815 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 38816 | const Type& v1 = vovoc->t0(); 38817 | const Type& v2 = vovoc->t1(); 38818 | const Type c = vovoc->t2(); 38819 | const details::operator_type o0 = operation; 38820 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f0()); 38821 | const details::operator_type o2 = expr_gen.get_operator(vovoc->f1()); 38822 | 38823 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38824 | binary_functor_t f1 = vovoc->f0(); 38825 | binary_functor_t f2 = vovoc->f1(); 38826 | 38827 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38828 | 38829 | expression_node_ptr result = error_node(); 38830 | 38831 | const bool synthesis_result = 38832 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38833 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, c, result); 38834 | 38835 | if (synthesis_result) 38836 | return result; 38837 | else if (!expr_gen.valid_operator(o0,f0)) 38838 | return error_node(); 38839 | 38840 | exprtk_debug(("v0 o0 (v1 o1 (v2 o2 c))\n")); 38841 | 38842 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, c, f0, f1, f2); 38843 | } 38844 | 38845 | static inline std::string id(expression_generator<Type>& expr_gen, 38846 | const details::operator_type o0, 38847 | const details::operator_type o1, 38848 | const details::operator_type o2) 38849 | { 38850 | return details::build_string() 38851 | << "t" << expr_gen.to_str(o0) 38852 | << "(t" << expr_gen.to_str(o1) 38853 | << "(t" << expr_gen.to_str(o2) 38854 | << "t))" 38855 | } 38856 | }; 38857 | 38858 | struct synthesize_vovocov_expression1 38859 | { 38860 | typedef typename vovocov_t::type1 node_type; 38861 | typedef typename vovocov_t::sf4_type sf4_type; 38862 | typedef typename node_type::T0 T0; 38863 | typedef typename node_type::T1 T1; 38864 | typedef typename node_type::T2 T2; 38865 | typedef typename node_type::T3 T3; 38866 | 38867 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38868 | const details::operator_type& operation, 38869 | expression_node_ptr (&branch)[2]) 38870 | { 38871 | // v0 o0 (v1 o1 (c o2 v2)) 38872 | typedef typename synthesize_vocov_expression1::node_type lcl_vocov_t; 38873 | 38874 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[1]); 38875 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 38876 | const Type& v1 = vocov->t0(); 38877 | const Type c = vocov->t1(); 38878 | const Type& v2 = vocov->t2(); 38879 | const details::operator_type o0 = operation; 38880 | const details::operator_type o1 = expr_gen.get_operator(vocov->f0()); 38881 | const details::operator_type o2 = expr_gen.get_operator(vocov->f1()); 38882 | 38883 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38884 | binary_functor_t f1 = vocov->f0(); 38885 | binary_functor_t f2 = vocov->f1(); 38886 | 38887 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38888 | 38889 | expression_node_ptr result = error_node(); 38890 | 38891 | const bool synthesis_result = 38892 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38893 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, c, v2, result); 38894 | 38895 | if (synthesis_result) 38896 | return result; 38897 | if (!expr_gen.valid_operator(o0,f0)) 38898 | return error_node(); 38899 | 38900 | exprtk_debug(("v0 o0 (v1 o1 (c o2 v2))\n")); 38901 | 38902 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, v2, f0, f1, f2); 38903 | } 38904 | 38905 | static inline std::string id(expression_generator<Type>& expr_gen, 38906 | const details::operator_type o0, 38907 | const details::operator_type o1, 38908 | const details::operator_type o2) 38909 | { 38910 | return details::build_string() 38911 | << "t" << expr_gen.to_str(o0) 38912 | << "(t" << expr_gen.to_str(o1) 38913 | << "(t" << expr_gen.to_str(o2) 38914 | << "t))" 38915 | } 38916 | }; 38917 | 38918 | struct synthesize_vocovov_expression1 38919 | { 38920 | typedef typename vocovov_t::type1 node_type; 38921 | typedef typename vocovov_t::sf4_type sf4_type; 38922 | typedef typename node_type::T0 T0; 38923 | typedef typename node_type::T1 T1; 38924 | typedef typename node_type::T2 T2; 38925 | typedef typename node_type::T3 T3; 38926 | 38927 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38928 | const details::operator_type& operation, 38929 | expression_node_ptr (&branch)[2]) 38930 | { 38931 | // v0 o0 (c o1 (v1 o2 v2)) 38932 | typedef typename synthesize_covov_expression1::node_type lcl_covov_t; 38933 | 38934 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[1]); 38935 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 38936 | const Type c = covov->t0(); 38937 | const Type& v1 = covov->t1(); 38938 | const Type& v2 = covov->t2(); 38939 | const details::operator_type o0 = operation; 38940 | const details::operator_type o1 = expr_gen.get_operator(covov->f0()); 38941 | const details::operator_type o2 = expr_gen.get_operator(covov->f1()); 38942 | 38943 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 38944 | binary_functor_t f1 = covov->f0(); 38945 | binary_functor_t f2 = covov->f1(); 38946 | 38947 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 38948 | 38949 | expression_node_ptr result = error_node(); 38950 | 38951 | const bool synthesis_result = 38952 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 38953 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c, v1, v2, result); 38954 | 38955 | if (synthesis_result) 38956 | return result; 38957 | else if (!expr_gen.valid_operator(o0,f0)) 38958 | return error_node(); 38959 | 38960 | exprtk_debug(("v0 o0 (c o1 (v1 o2 v2))\n")); 38961 | 38962 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, v2, f0, f1, f2); 38963 | } 38964 | 38965 | static inline std::string id(expression_generator<Type>& expr_gen, 38966 | const details::operator_type o0, 38967 | const details::operator_type o1, 38968 | const details::operator_type o2) 38969 | { 38970 | return details::build_string() 38971 | << "t" << expr_gen.to_str(o0) 38972 | << "(t" << expr_gen.to_str(o1) 38973 | << "(t" << expr_gen.to_str(o2) 38974 | << "t))" 38975 | } 38976 | }; 38977 | 38978 | struct synthesize_covovov_expression1 38979 | { 38980 | typedef typename covovov_t::type1 node_type; 38981 | typedef typename covovov_t::sf4_type sf4_type; 38982 | typedef typename node_type::T0 T0; 38983 | typedef typename node_type::T1 T1; 38984 | typedef typename node_type::T2 T2; 38985 | typedef typename node_type::T3 T3; 38986 | 38987 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 38988 | const details::operator_type& operation, 38989 | expression_node_ptr (&branch)[2]) 38990 | { 38991 | // c o0 (v0 o1 (v1 o2 v2)) 38992 | typedef typename synthesize_vovov_expression1::node_type lcl_vovov_t; 38993 | 38994 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[1]); 38995 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 38996 | const Type& v0 = vovov->t0(); 38997 | const Type& v1 = vovov->t1(); 38998 | const Type& v2 = vovov->t2(); 38999 | const details::operator_type o0 = operation; 39000 | const details::operator_type o1 = expr_gen.get_operator(vovov->f0()); 39001 | const details::operator_type o2 = expr_gen.get_operator(vovov->f1()); 39002 | 39003 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39004 | binary_functor_t f1 = vovov->f0(); 39005 | binary_functor_t f2 = vovov->f1(); 39006 | 39007 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39008 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39009 | 39010 | expression_node_ptr result = error_node(); 39011 | 39012 | const bool synthesis_result = 39013 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39014 | (expr_gen, id(expr_gen, o0, o1, o2), c, v0, v1, v2, result); 39015 | 39016 | if (synthesis_result) 39017 | return result; 39018 | if (!expr_gen.valid_operator(o0,f0)) 39019 | return error_node(); 39020 | 39021 | exprtk_debug(("c o0 (v0 o1 (v1 o2 v2))\n")); 39022 | 39023 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, v2, f0, f1, f2); 39024 | } 39025 | 39026 | static inline std::string id(expression_generator<Type>& expr_gen, 39027 | const details::operator_type o0, 39028 | const details::operator_type o1, 39029 | const details::operator_type o2) 39030 | { 39031 | return details::build_string() 39032 | << "t" << expr_gen.to_str(o0) 39033 | << "(t" << expr_gen.to_str(o1) 39034 | << "(t" << expr_gen.to_str(o2) 39035 | << "t))" 39036 | } 39037 | }; 39038 | 39039 | struct synthesize_covocov_expression1 39040 | { 39041 | typedef typename covocov_t::type1 node_type; 39042 | typedef typename covocov_t::sf4_type sf4_type; 39043 | typedef typename node_type::T0 T0; 39044 | typedef typename node_type::T1 T1; 39045 | typedef typename node_type::T2 T2; 39046 | typedef typename node_type::T3 T3; 39047 | 39048 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39049 | const details::operator_type& operation, 39050 | expression_node_ptr (&branch)[2]) 39051 | { 39052 | // c0 o0 (v0 o1 (c1 o2 v1)) 39053 | typedef typename synthesize_vocov_expression1::node_type lcl_vocov_t; 39054 | 39055 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[1]); 39056 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 39057 | const Type& v0 = vocov->t0(); 39058 | const Type c1 = vocov->t1(); 39059 | const Type& v1 = vocov->t2(); 39060 | const details::operator_type o0 = operation; 39061 | const details::operator_type o1 = expr_gen.get_operator(vocov->f0()); 39062 | const details::operator_type o2 = expr_gen.get_operator(vocov->f1()); 39063 | 39064 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39065 | binary_functor_t f1 = vocov->f0(); 39066 | binary_functor_t f2 = vocov->f1(); 39067 | 39068 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39069 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39070 | 39071 | expression_node_ptr result = error_node(); 39072 | 39073 | const bool synthesis_result = 39074 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39075 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, c1, v1, result); 39076 | 39077 | if (synthesis_result) 39078 | return result; 39079 | else if (!expr_gen.valid_operator(o0,f0)) 39080 | return error_node(); 39081 | 39082 | exprtk_debug(("c0 o0 (v0 o1 (c1 o2 v1))\n")); 39083 | 39084 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, c1, v1, f0, f1, f2); 39085 | } 39086 | 39087 | static inline std::string id(expression_generator<Type>& expr_gen, 39088 | const details::operator_type o0, 39089 | const details::operator_type o1, 39090 | const details::operator_type o2) 39091 | { 39092 | return details::build_string() 39093 | << "t" << expr_gen.to_str(o0) 39094 | << "(t" << expr_gen.to_str(o1) 39095 | << "(t" << expr_gen.to_str(o2) 39096 | << "t))" 39097 | } 39098 | }; 39099 | 39100 | struct synthesize_vocovoc_expression1 39101 | { 39102 | typedef typename vocovoc_t::type1 node_type; 39103 | typedef typename vocovoc_t::sf4_type sf4_type; 39104 | typedef typename node_type::T0 T0; 39105 | typedef typename node_type::T1 T1; 39106 | typedef typename node_type::T2 T2; 39107 | typedef typename node_type::T3 T3; 39108 | 39109 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39110 | const details::operator_type& operation, 39111 | expression_node_ptr (&branch)[2]) 39112 | { 39113 | // v0 o0 (c0 o1 (v1 o2 c2)) 39114 | typedef typename synthesize_covoc_expression1::node_type lcl_covoc_t; 39115 | 39116 | const lcl_covoc_t* covoc = static_cast<const lcl_covoc_t*>(branch[1]); 39117 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39118 | const Type c0 = covoc->t0(); 39119 | const Type& v1 = covoc->t1(); 39120 | const Type c1 = covoc->t2(); 39121 | const details::operator_type o0 = operation; 39122 | const details::operator_type o1 = expr_gen.get_operator(covoc->f0()); 39123 | const details::operator_type o2 = expr_gen.get_operator(covoc->f1()); 39124 | 39125 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39126 | binary_functor_t f1 = covoc->f0(); 39127 | binary_functor_t f2 = covoc->f1(); 39128 | 39129 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39130 | 39131 | expression_node_ptr result = error_node(); 39132 | 39133 | const bool synthesis_result = 39134 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39135 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, v1, c1, result); 39136 | 39137 | if (synthesis_result) 39138 | return result; 39139 | else if (!expr_gen.valid_operator(o0,f0)) 39140 | return error_node(); 39141 | 39142 | exprtk_debug(("v0 o0 (c0 o1 (v1 o2 c2))\n")); 39143 | 39144 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, v1, c1, f0, f1, f2); 39145 | } 39146 | 39147 | static inline std::string id(expression_generator<Type>& expr_gen, 39148 | const details::operator_type o0, 39149 | const details::operator_type o1, 39150 | const details::operator_type o2) 39151 | { 39152 | return details::build_string() 39153 | << "t" << expr_gen.to_str(o0) 39154 | << "(t" << expr_gen.to_str(o1) 39155 | << "(t" << expr_gen.to_str(o2) 39156 | << "t))" 39157 | } 39158 | }; 39159 | 39160 | struct synthesize_covovoc_expression1 39161 | { 39162 | typedef typename covovoc_t::type1 node_type; 39163 | typedef typename covovoc_t::sf4_type sf4_type; 39164 | typedef typename node_type::T0 T0; 39165 | typedef typename node_type::T1 T1; 39166 | typedef typename node_type::T2 T2; 39167 | typedef typename node_type::T3 T3; 39168 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39169 | const details::operator_type& operation, 39170 | expression_node_ptr (&branch)[2]) 39171 | { 39172 | // c0 o0 (v0 o1 (v1 o2 c1)) 39173 | typedef typename synthesize_vovoc_expression1::node_type lcl_vovoc_t; 39174 | 39175 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[1]); 39176 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 39177 | const Type& v0 = vovoc->t0(); 39178 | const Type& v1 = vovoc->t1(); 39179 | const Type c1 = vovoc->t2(); 39180 | const details::operator_type o0 = operation; 39181 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f0()); 39182 | const details::operator_type o2 = expr_gen.get_operator(vovoc->f1()); 39183 | 39184 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39185 | binary_functor_t f1 = vovoc->f0(); 39186 | binary_functor_t f2 = vovoc->f1(); 39187 | 39188 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39189 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39190 | 39191 | expression_node_ptr result = error_node(); 39192 | 39193 | const bool synthesis_result = 39194 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39195 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, v1, c1, result); 39196 | 39197 | if (synthesis_result) 39198 | return result; 39199 | else if (!expr_gen.valid_operator(o0,f0)) 39200 | return error_node(); 39201 | 39202 | exprtk_debug(("c0 o0 (v0 o1 (v1 o2 c1))\n")); 39203 | 39204 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, v1, c1, f0, f1, f2); 39205 | } 39206 | 39207 | static inline std::string id(expression_generator<Type>& expr_gen, 39208 | const details::operator_type o0, 39209 | const details::operator_type o1, 39210 | const details::operator_type o2) 39211 | { 39212 | return details::build_string() 39213 | << "t" << expr_gen.to_str(o0) 39214 | << "(t" << expr_gen.to_str(o1) 39215 | << "(t" << expr_gen.to_str(o2) 39216 | << "t))" 39217 | } 39218 | }; 39219 | 39220 | struct synthesize_vococov_expression1 39221 | { 39222 | typedef typename vococov_t::type1 node_type; 39223 | typedef typename vococov_t::sf4_type sf4_type; 39224 | typedef typename node_type::T0 T0; 39225 | typedef typename node_type::T1 T1; 39226 | typedef typename node_type::T2 T2; 39227 | typedef typename node_type::T3 T3; 39228 | 39229 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39230 | const details::operator_type& operation, 39231 | expression_node_ptr (&branch)[2]) 39232 | { 39233 | // v0 o0 (c0 o1 (c1 o2 v1)) 39234 | typedef typename synthesize_cocov_expression1::node_type lcl_cocov_t; 39235 | 39236 | const lcl_cocov_t* cocov = static_cast<const lcl_cocov_t*>(branch[1]); 39237 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39238 | const Type c0 = cocov->t0(); 39239 | const Type c1 = cocov->t1(); 39240 | const Type& v1 = cocov->t2(); 39241 | const details::operator_type o0 = operation; 39242 | const details::operator_type o1 = expr_gen.get_operator(cocov->f0()); 39243 | const details::operator_type o2 = expr_gen.get_operator(cocov->f1()); 39244 | 39245 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39246 | binary_functor_t f1 = cocov->f0(); 39247 | binary_functor_t f2 = cocov->f1(); 39248 | 39249 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39250 | 39251 | expression_node_ptr result = error_node(); 39252 | 39253 | const bool synthesis_result = 39254 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39255 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, c1, v1, result); 39256 | 39257 | if (synthesis_result) 39258 | return result; 39259 | else if (!expr_gen.valid_operator(o0,f0)) 39260 | return error_node(); 39261 | 39262 | exprtk_debug(("v0 o0 (c0 o1 (c1 o2 v1))\n")); 39263 | 39264 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, c1, v1, f0, f1, f2); 39265 | } 39266 | 39267 | static inline std::string id(expression_generator<Type>& expr_gen, 39268 | const details::operator_type o0, 39269 | const details::operator_type o1, 39270 | const details::operator_type o2) 39271 | { 39272 | return details::build_string() 39273 | << "t" << expr_gen.to_str(o0) 39274 | << "(t" << expr_gen.to_str(o1) 39275 | << "(t" << expr_gen.to_str(o2) 39276 | << "t))" 39277 | } 39278 | }; 39279 | 39280 | struct synthesize_vovovov_expression2 39281 | { 39282 | typedef typename vovovov_t::type2 node_type; 39283 | typedef typename vovovov_t::sf4_type sf4_type; 39284 | typedef typename node_type::T0 T0; 39285 | typedef typename node_type::T1 T1; 39286 | typedef typename node_type::T2 T2; 39287 | typedef typename node_type::T3 T3; 39288 | 39289 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39290 | const details::operator_type& operation, 39291 | expression_node_ptr (&branch)[2]) 39292 | { 39293 | // v0 o0 ((v1 o1 v2) o2 v3) 39294 | typedef typename synthesize_vovov_expression0::node_type lcl_vovov_t; 39295 | 39296 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[1]); 39297 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39298 | const Type& v1 = vovov->t0(); 39299 | const Type& v2 = vovov->t1(); 39300 | const Type& v3 = vovov->t2(); 39301 | const details::operator_type o0 = operation; 39302 | const details::operator_type o1 = expr_gen.get_operator(vovov->f0()); 39303 | const details::operator_type o2 = expr_gen.get_operator(vovov->f1()); 39304 | 39305 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39306 | binary_functor_t f1 = vovov->f0(); 39307 | binary_functor_t f2 = vovov->f1(); 39308 | 39309 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39310 | 39311 | expression_node_ptr result = error_node(); 39312 | 39313 | const bool synthesis_result = 39314 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39315 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, v3, result); 39316 | 39317 | if (synthesis_result) 39318 | return result; 39319 | else if (!expr_gen.valid_operator(o0,f0)) 39320 | return error_node(); 39321 | 39322 | exprtk_debug(("v0 o0 ((v1 o1 v2) o2 v3)\n")); 39323 | 39324 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, v3, f0, f1, f2); 39325 | } 39326 | 39327 | static inline std::string id(expression_generator<Type>& expr_gen, 39328 | const details::operator_type o0, 39329 | const details::operator_type o1, 39330 | const details::operator_type o2) 39331 | { 39332 | return details::build_string() 39333 | << "t" << expr_gen.to_str(o0) 39334 | << "((t" << expr_gen.to_str(o1) 39335 | << "t)" << expr_gen.to_str(o2) 39336 | << "t)" 39337 | } 39338 | }; 39339 | 39340 | struct synthesize_vovovoc_expression2 39341 | { 39342 | typedef typename vovovoc_t::type2 node_type; 39343 | typedef typename vovovoc_t::sf4_type sf4_type; 39344 | typedef typename node_type::T0 T0; 39345 | typedef typename node_type::T1 T1; 39346 | typedef typename node_type::T2 T2; 39347 | typedef typename node_type::T3 T3; 39348 | 39349 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39350 | const details::operator_type& operation, 39351 | expression_node_ptr (&branch)[2]) 39352 | { 39353 | // v0 o0 ((v1 o1 v2) o2 c) 39354 | typedef typename synthesize_vovoc_expression0::node_type lcl_vovoc_t; 39355 | 39356 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[1]); 39357 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39358 | const Type& v1 = vovoc->t0(); 39359 | const Type& v2 = vovoc->t1(); 39360 | const Type c = vovoc->t2(); 39361 | const details::operator_type o0 = operation; 39362 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f0()); 39363 | const details::operator_type o2 = expr_gen.get_operator(vovoc->f1()); 39364 | 39365 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39366 | binary_functor_t f1 = vovoc->f0(); 39367 | binary_functor_t f2 = vovoc->f1(); 39368 | 39369 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39370 | 39371 | expression_node_ptr result = error_node(); 39372 | 39373 | const bool synthesis_result = 39374 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39375 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, c, result); 39376 | 39377 | if (synthesis_result) 39378 | return result; 39379 | else if (!expr_gen.valid_operator(o0,f0)) 39380 | return error_node(); 39381 | 39382 | exprtk_debug(("v0 o0 ((v1 o1 v2) o2 c)\n")); 39383 | 39384 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, c, f0, f1, f2); 39385 | } 39386 | 39387 | static inline std::string id(expression_generator<Type>& expr_gen, 39388 | const details::operator_type o0, 39389 | const details::operator_type o1, 39390 | const details::operator_type o2) 39391 | { 39392 | return details::build_string() 39393 | << "t" << expr_gen.to_str(o0) 39394 | << "((t" << expr_gen.to_str(o1) 39395 | << "t)" << expr_gen.to_str(o2) 39396 | << "t)" 39397 | } 39398 | }; 39399 | 39400 | struct synthesize_vovocov_expression2 39401 | { 39402 | typedef typename vovocov_t::type2 node_type; 39403 | typedef typename vovocov_t::sf4_type sf4_type; 39404 | typedef typename node_type::T0 T0; 39405 | typedef typename node_type::T1 T1; 39406 | typedef typename node_type::T2 T2; 39407 | typedef typename node_type::T3 T3; 39408 | 39409 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39410 | const details::operator_type& operation, 39411 | expression_node_ptr (&branch)[2]) 39412 | { 39413 | // v0 o0 ((v1 o1 c) o2 v2) 39414 | typedef typename synthesize_vocov_expression0::node_type lcl_vocov_t; 39415 | 39416 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[1]); 39417 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39418 | const Type& v1 = vocov->t0(); 39419 | const Type c = vocov->t1(); 39420 | const Type& v2 = vocov->t2(); 39421 | const details::operator_type o0 = operation; 39422 | const details::operator_type o1 = expr_gen.get_operator(vocov->f0()); 39423 | const details::operator_type o2 = expr_gen.get_operator(vocov->f1()); 39424 | 39425 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39426 | binary_functor_t f1 = vocov->f0(); 39427 | binary_functor_t f2 = vocov->f1(); 39428 | 39429 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39430 | 39431 | expression_node_ptr result = error_node(); 39432 | 39433 | const bool synthesis_result = 39434 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39435 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, c, v2, result); 39436 | 39437 | if (synthesis_result) 39438 | return result; 39439 | else if (!expr_gen.valid_operator(o0,f0)) 39440 | return error_node(); 39441 | 39442 | exprtk_debug(("v0 o0 ((v1 o1 c) o2 v2)\n")); 39443 | 39444 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, v2, f0, f1, f2); 39445 | } 39446 | 39447 | static inline std::string id(expression_generator<Type>& expr_gen, 39448 | const details::operator_type o0, 39449 | const details::operator_type o1, 39450 | const details::operator_type o2) 39451 | { 39452 | return details::build_string() 39453 | << "t" << expr_gen.to_str(o0) 39454 | << "((t" << expr_gen.to_str(o1) 39455 | << "t)" << expr_gen.to_str(o2) 39456 | << "t)" 39457 | } 39458 | }; 39459 | 39460 | struct synthesize_vocovov_expression2 39461 | { 39462 | typedef typename vocovov_t::type2 node_type; 39463 | typedef typename vocovov_t::sf4_type sf4_type; 39464 | typedef typename node_type::T0 T0; 39465 | typedef typename node_type::T1 T1; 39466 | typedef typename node_type::T2 T2; 39467 | typedef typename node_type::T3 T3; 39468 | 39469 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39470 | const details::operator_type& operation, 39471 | expression_node_ptr (&branch)[2]) 39472 | { 39473 | // v0 o0 ((c o1 v1) o2 v2) 39474 | typedef typename synthesize_covov_expression0::node_type lcl_covov_t; 39475 | 39476 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[1]); 39477 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39478 | const Type c = covov->t0(); 39479 | const Type& v1 = covov->t1(); 39480 | const Type& v2 = covov->t2(); 39481 | const details::operator_type o0 = operation; 39482 | const details::operator_type o1 = expr_gen.get_operator(covov->f0()); 39483 | const details::operator_type o2 = expr_gen.get_operator(covov->f1()); 39484 | 39485 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39486 | binary_functor_t f1 = covov->f0(); 39487 | binary_functor_t f2 = covov->f1(); 39488 | 39489 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39490 | 39491 | expression_node_ptr result = error_node(); 39492 | 39493 | const bool synthesis_result = 39494 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39495 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c, v1, v2, result); 39496 | 39497 | if (synthesis_result) 39498 | return result; 39499 | else if (!expr_gen.valid_operator(o0,f0)) 39500 | return error_node(); 39501 | 39502 | exprtk_debug(("v0 o0 ((c o1 v1) o2 v2)\n")); 39503 | 39504 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, v2, f0, f1, f2); 39505 | } 39506 | 39507 | static inline std::string id(expression_generator<Type>& expr_gen, 39508 | const details::operator_type o0, 39509 | const details::operator_type o1, 39510 | const details::operator_type o2) 39511 | { 39512 | return details::build_string() 39513 | << "t" << expr_gen.to_str(o0) 39514 | << "((t" << expr_gen.to_str(o1) 39515 | << "t)" << expr_gen.to_str(o2) 39516 | << "t)" 39517 | } 39518 | }; 39519 | 39520 | struct synthesize_covovov_expression2 39521 | { 39522 | typedef typename covovov_t::type2 node_type; 39523 | typedef typename covovov_t::sf4_type sf4_type; 39524 | typedef typename node_type::T0 T0; 39525 | typedef typename node_type::T1 T1; 39526 | typedef typename node_type::T2 T2; 39527 | typedef typename node_type::T3 T3; 39528 | 39529 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39530 | const details::operator_type& operation, 39531 | expression_node_ptr (&branch)[2]) 39532 | { 39533 | // c o0 ((v1 o1 v2) o2 v3) 39534 | typedef typename synthesize_vovov_expression0::node_type lcl_vovov_t; 39535 | 39536 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[1]); 39537 | const Type c = static_cast<details::literal_node<Type>*>(branch[0])->value(); 39538 | const Type& v0 = vovov->t0(); 39539 | const Type& v1 = vovov->t1(); 39540 | const Type& v2 = vovov->t2(); 39541 | const details::operator_type o0 = operation; 39542 | const details::operator_type o1 = expr_gen.get_operator(vovov->f0()); 39543 | const details::operator_type o2 = expr_gen.get_operator(vovov->f1()); 39544 | 39545 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39546 | binary_functor_t f1 = vovov->f0(); 39547 | binary_functor_t f2 = vovov->f1(); 39548 | 39549 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39550 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39551 | 39552 | expression_node_ptr result = error_node(); 39553 | 39554 | const bool synthesis_result = 39555 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39556 | (expr_gen, id(expr_gen, o0, o1, o2), c, v0, v1, v2, result); 39557 | 39558 | if (synthesis_result) 39559 | return result; 39560 | else if (!expr_gen.valid_operator(o0,f0)) 39561 | return error_node(); 39562 | 39563 | exprtk_debug(("c o0 ((v1 o1 v2) o2 v3)\n")); 39564 | 39565 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, v2, f0, f1, f2); 39566 | } 39567 | 39568 | static inline std::string id(expression_generator<Type>& expr_gen, 39569 | const details::operator_type o0, 39570 | const details::operator_type o1, 39571 | const details::operator_type o2) 39572 | { 39573 | return details::build_string() 39574 | << "t" << expr_gen.to_str(o0) 39575 | << "((t" << expr_gen.to_str(o1) 39576 | << "t)" << expr_gen.to_str(o2) 39577 | << "t)" 39578 | } 39579 | }; 39580 | 39581 | struct synthesize_covocov_expression2 39582 | { 39583 | typedef typename covocov_t::type2 node_type; 39584 | typedef typename covocov_t::sf4_type sf4_type; 39585 | typedef typename node_type::T0 T0; 39586 | typedef typename node_type::T1 T1; 39587 | typedef typename node_type::T2 T2; 39588 | typedef typename node_type::T3 T3; 39589 | 39590 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39591 | const details::operator_type& operation, 39592 | expression_node_ptr (&branch)[2]) 39593 | { 39594 | // c0 o0 ((v0 o1 c1) o2 v1) 39595 | typedef typename synthesize_vocov_expression0::node_type lcl_vocov_t; 39596 | 39597 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[1]); 39598 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 39599 | const Type& v0 = vocov->t0(); 39600 | const Type c1 = vocov->t1(); 39601 | const Type& v1 = vocov->t2(); 39602 | const details::operator_type o0 = operation; 39603 | const details::operator_type o1 = expr_gen.get_operator(vocov->f0()); 39604 | const details::operator_type o2 = expr_gen.get_operator(vocov->f1()); 39605 | 39606 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39607 | binary_functor_t f1 = vocov->f0(); 39608 | binary_functor_t f2 = vocov->f1(); 39609 | 39610 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39611 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39612 | 39613 | expression_node_ptr result = error_node(); 39614 | 39615 | const bool synthesis_result = 39616 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39617 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, c1, v1, result); 39618 | 39619 | if (synthesis_result) 39620 | return result; 39621 | else if (!expr_gen.valid_operator(o0,f0)) 39622 | return error_node(); 39623 | 39624 | exprtk_debug(("c0 o0 ((v0 o1 c1) o2 v1)\n")); 39625 | 39626 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, c1, v1, f0, f1, f2); 39627 | } 39628 | 39629 | static inline std::string id(expression_generator<Type>& expr_gen, 39630 | const details::operator_type o0, 39631 | const details::operator_type o1, 39632 | const details::operator_type o2) 39633 | { 39634 | return details::build_string() 39635 | << "t" << expr_gen.to_str(o0) 39636 | << "((t" << expr_gen.to_str(o1) 39637 | << "t)" << expr_gen.to_str(o2) 39638 | << "t)" 39639 | } 39640 | }; 39641 | 39642 | struct synthesize_vocovoc_expression2 39643 | { 39644 | typedef typename vocovoc_t::type2 node_type; 39645 | typedef typename vocovoc_t::sf4_type sf4_type; 39646 | typedef typename node_type::T0 T0; 39647 | typedef typename node_type::T1 T1; 39648 | typedef typename node_type::T2 T2; 39649 | typedef typename node_type::T3 T3; 39650 | 39651 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39652 | const details::operator_type& operation, 39653 | expression_node_ptr (&branch)[2]) 39654 | { 39655 | // v0 o0 ((c0 o1 v1) o2 c1) 39656 | typedef typename synthesize_covoc_expression0::node_type lcl_covoc_t; 39657 | 39658 | const lcl_covoc_t* covoc = static_cast<const lcl_covoc_t*>(branch[1]); 39659 | const Type& v0 = static_cast<details::variable_node<Type>*>(branch[0])->ref(); 39660 | const Type c0 = covoc->t0(); 39661 | const Type& v1 = covoc->t1(); 39662 | const Type c1 = covoc->t2(); 39663 | const details::operator_type o0 = operation; 39664 | const details::operator_type o1 = expr_gen.get_operator(covoc->f0()); 39665 | const details::operator_type o2 = expr_gen.get_operator(covoc->f1()); 39666 | 39667 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39668 | binary_functor_t f1 = covoc->f0(); 39669 | binary_functor_t f2 = covoc->f1(); 39670 | 39671 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39672 | 39673 | expression_node_ptr result = error_node(); 39674 | 39675 | const bool synthesis_result = 39676 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39677 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, v1, c1, result); 39678 | 39679 | if (synthesis_result) 39680 | return result; 39681 | else if (!expr_gen.valid_operator(o0,f0)) 39682 | return error_node(); 39683 | 39684 | exprtk_debug(("v0 o0 ((c0 o1 v1) o2 c1)\n")); 39685 | 39686 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, v1, c1, f0, f1, f2); 39687 | } 39688 | 39689 | static inline std::string id(expression_generator<Type>& expr_gen, 39690 | const details::operator_type o0, 39691 | const details::operator_type o1, 39692 | const details::operator_type o2) 39693 | { 39694 | return details::build_string() 39695 | << "t" << expr_gen.to_str(o0) 39696 | << "((t" << expr_gen.to_str(o1) 39697 | << "t)" << expr_gen.to_str(o2) 39698 | << "t)" 39699 | } 39700 | }; 39701 | 39702 | struct synthesize_covovoc_expression2 39703 | { 39704 | typedef typename covovoc_t::type2 node_type; 39705 | typedef typename covovoc_t::sf4_type sf4_type; 39706 | typedef typename node_type::T0 T0; 39707 | typedef typename node_type::T1 T1; 39708 | typedef typename node_type::T2 T2; 39709 | typedef typename node_type::T3 T3; 39710 | 39711 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39712 | const details::operator_type& operation, 39713 | expression_node_ptr (&branch)[2]) 39714 | { 39715 | // c0 o0 ((v0 o1 v1) o2 c1) 39716 | typedef typename synthesize_vovoc_expression0::node_type lcl_vovoc_t; 39717 | 39718 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[1]); 39719 | const Type c0 = static_cast<details::literal_node<Type>*>(branch[0])->value(); 39720 | const Type& v0 = vovoc->t0(); 39721 | const Type& v1 = vovoc->t1(); 39722 | const Type c1 = vovoc->t2(); 39723 | const details::operator_type o0 = operation; 39724 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f0()); 39725 | const details::operator_type o2 = expr_gen.get_operator(vovoc->f1()); 39726 | 39727 | binary_functor_t f0 = reinterpret_cast<binary_functor_t>(0); 39728 | binary_functor_t f1 = vovoc->f0(); 39729 | binary_functor_t f2 = vovoc->f1(); 39730 | 39731 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39732 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39733 | 39734 | expression_node_ptr result = error_node(); 39735 | 39736 | const bool synthesis_result = 39737 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39738 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, v1, c1, result); 39739 | 39740 | if (synthesis_result) 39741 | return result; 39742 | else if (!expr_gen.valid_operator(o0,f0)) 39743 | return error_node(); 39744 | 39745 | exprtk_debug(("c0 o0 ((v0 o1 v1) o2 c1)\n")); 39746 | 39747 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, v1, c1, f0, f1, f2); 39748 | } 39749 | 39750 | static inline std::string id(expression_generator<Type>& expr_gen, 39751 | const details::operator_type o0, 39752 | const details::operator_type o1, 39753 | const details::operator_type o2) 39754 | { 39755 | return details::build_string() 39756 | << "t" << expr_gen.to_str(o0) 39757 | << "((t" << expr_gen.to_str(o1) 39758 | << "t)" << expr_gen.to_str(o2) 39759 | << "t)" 39760 | } 39761 | }; 39762 | 39763 | struct synthesize_vococov_expression2 39764 | { 39765 | typedef typename vococov_t::type2 node_type; 39766 | static inline expression_node_ptr process(expression_generator<Type>&, 39767 | const details::operator_type&, 39768 | expression_node_ptr (&)[2]) 39769 | { 39770 | // v0 o0 ((c0 o1 c1) o2 v1) - Not possible 39771 | exprtk_debug(("v0 o0 ((c0 o1 c1) o2 v1) - Not possible\n")); 39772 | return error_node(); 39773 | } 39774 | 39775 | static inline std::string id(expression_generator<Type>&, 39776 | const details::operator_type, 39777 | const details::operator_type, 39778 | const details::operator_type) 39779 | { 39780 | return "INVALID" 39781 | } 39782 | }; 39783 | 39784 | struct synthesize_vovovov_expression3 39785 | { 39786 | typedef typename vovovov_t::type3 node_type; 39787 | typedef typename vovovov_t::sf4_type sf4_type; 39788 | typedef typename node_type::T0 T0; 39789 | typedef typename node_type::T1 T1; 39790 | typedef typename node_type::T2 T2; 39791 | typedef typename node_type::T3 T3; 39792 | 39793 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39794 | const details::operator_type& operation, 39795 | expression_node_ptr (&branch)[2]) 39796 | { 39797 | // ((v0 o0 v1) o1 v2) o2 v3 39798 | typedef typename synthesize_vovov_expression0::node_type lcl_vovov_t; 39799 | 39800 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[0]); 39801 | const Type& v0 = vovov->t0(); 39802 | const Type& v1 = vovov->t1(); 39803 | const Type& v2 = vovov->t2(); 39804 | const Type& v3 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 39805 | const details::operator_type o0 = expr_gen.get_operator(vovov->f0()); 39806 | const details::operator_type o1 = expr_gen.get_operator(vovov->f1()); 39807 | const details::operator_type o2 = operation; 39808 | 39809 | binary_functor_t f0 = vovov->f0(); 39810 | binary_functor_t f1 = vovov->f1(); 39811 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 39812 | 39813 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39814 | 39815 | expression_node_ptr result = error_node(); 39816 | 39817 | const bool synthesis_result = 39818 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39819 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, v3, result); 39820 | 39821 | if (synthesis_result) 39822 | return result; 39823 | else if (!expr_gen.valid_operator(o2,f2)) 39824 | return error_node(); 39825 | 39826 | exprtk_debug(("((v0 o0 v1) o1 v2) o2 v3\n")); 39827 | 39828 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, v3, f0, f1, f2); 39829 | } 39830 | 39831 | static inline std::string id(expression_generator<Type>& expr_gen, 39832 | const details::operator_type o0, 39833 | const details::operator_type o1, 39834 | const details::operator_type o2) 39835 | { 39836 | return details::build_string() 39837 | << "((t" << expr_gen.to_str(o0) 39838 | << "t)" << expr_gen.to_str(o1) 39839 | << "t)" << expr_gen.to_str(o2) 39840 | << "t" 39841 | } 39842 | }; 39843 | 39844 | struct synthesize_vovovoc_expression3 39845 | { 39846 | typedef typename vovovoc_t::type3 node_type; 39847 | typedef typename vovovoc_t::sf4_type sf4_type; 39848 | typedef typename node_type::T0 T0; 39849 | typedef typename node_type::T1 T1; 39850 | typedef typename node_type::T2 T2; 39851 | typedef typename node_type::T3 T3; 39852 | 39853 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39854 | const details::operator_type& operation, 39855 | expression_node_ptr (&branch)[2]) 39856 | { 39857 | // ((v0 o0 v1) o1 v2) o2 c 39858 | typedef typename synthesize_vovov_expression0::node_type lcl_vovov_t; 39859 | 39860 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[0]); 39861 | const Type& v0 = vovov->t0(); 39862 | const Type& v1 = vovov->t1(); 39863 | const Type& v2 = vovov->t2(); 39864 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 39865 | const details::operator_type o0 = expr_gen.get_operator(vovov->f0()); 39866 | const details::operator_type o1 = expr_gen.get_operator(vovov->f1()); 39867 | const details::operator_type o2 = operation; 39868 | 39869 | binary_functor_t f0 = vovov->f0(); 39870 | binary_functor_t f1 = vovov->f1(); 39871 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 39872 | 39873 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39874 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 39875 | 39876 | expression_node_ptr result = error_node(); 39877 | 39878 | const bool synthesis_result = 39879 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39880 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, c, result); 39881 | 39882 | if (synthesis_result) 39883 | return result; 39884 | else if (!expr_gen.valid_operator(o2,f2)) 39885 | return error_node(); 39886 | 39887 | exprtk_debug(("((v0 o0 v1) o1 v2) o2 c\n")); 39888 | 39889 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, c, f0, f1, f2); 39890 | } 39891 | 39892 | static inline std::string id(expression_generator<Type>& expr_gen, 39893 | const details::operator_type o0, 39894 | const details::operator_type o1, 39895 | const details::operator_type o2) 39896 | { 39897 | return details::build_string() 39898 | << "((t" << expr_gen.to_str(o0) 39899 | << "t)" << expr_gen.to_str(o1) 39900 | << "t)" << expr_gen.to_str(o2) 39901 | << "t" 39902 | } 39903 | }; 39904 | 39905 | struct synthesize_vovocov_expression3 39906 | { 39907 | typedef typename vovocov_t::type3 node_type; 39908 | typedef typename vovocov_t::sf4_type sf4_type; 39909 | typedef typename node_type::T0 T0; 39910 | typedef typename node_type::T1 T1; 39911 | typedef typename node_type::T2 T2; 39912 | typedef typename node_type::T3 T3; 39913 | 39914 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39915 | const details::operator_type& operation, 39916 | expression_node_ptr (&branch)[2]) 39917 | { 39918 | // ((v0 o0 v1) o1 c) o2 v2 39919 | typedef typename synthesize_vovoc_expression0::node_type lcl_vovoc_t; 39920 | 39921 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[0]); 39922 | const Type& v0 = vovoc->t0(); 39923 | const Type& v1 = vovoc->t1(); 39924 | const Type c = vovoc->t2(); 39925 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 39926 | const details::operator_type o0 = expr_gen.get_operator(vovoc->f0()); 39927 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f1()); 39928 | const details::operator_type o2 = operation; 39929 | 39930 | binary_functor_t f0 = vovoc->f0(); 39931 | binary_functor_t f1 = vovoc->f1(); 39932 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 39933 | 39934 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39935 | 39936 | expression_node_ptr result = error_node(); 39937 | 39938 | const bool synthesis_result = 39939 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 39940 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, c, v2, result); 39941 | 39942 | if (synthesis_result) 39943 | return result; 39944 | else if (!expr_gen.valid_operator(o2,f2)) 39945 | return error_node(); 39946 | 39947 | exprtk_debug(("((v0 o0 v1) o1 c) o2 v2\n")); 39948 | 39949 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, v2, f0, f1, f2); 39950 | } 39951 | 39952 | static inline std::string id(expression_generator<Type>& expr_gen, 39953 | const details::operator_type o0, 39954 | const details::operator_type o1, 39955 | const details::operator_type o2) 39956 | { 39957 | return details::build_string() 39958 | << "((t" << expr_gen.to_str(o0) 39959 | << "t)" << expr_gen.to_str(o1) 39960 | << "t)" << expr_gen.to_str(o2) 39961 | << "t" 39962 | } 39963 | }; 39964 | 39965 | struct synthesize_vocovov_expression3 39966 | { 39967 | typedef typename vocovov_t::type3 node_type; 39968 | typedef typename vocovov_t::sf4_type sf4_type; 39969 | typedef typename node_type::T0 T0; 39970 | typedef typename node_type::T1 T1; 39971 | typedef typename node_type::T2 T2; 39972 | typedef typename node_type::T3 T3; 39973 | 39974 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 39975 | const details::operator_type& operation, 39976 | expression_node_ptr (&branch)[2]) 39977 | { 39978 | // ((v0 o0 c) o1 v1) o2 v2 39979 | typedef typename synthesize_vocov_expression0::node_type lcl_vocov_t; 39980 | 39981 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[0]); 39982 | const Type& v0 = vocov->t0(); 39983 | const Type c = vocov->t1(); 39984 | const Type& v1 = vocov->t2(); 39985 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 39986 | const details::operator_type o0 = expr_gen.get_operator(vocov->f0()); 39987 | const details::operator_type o1 = expr_gen.get_operator(vocov->f1()); 39988 | const details::operator_type o2 = operation; 39989 | 39990 | binary_functor_t f0 = vocov->f0(); 39991 | binary_functor_t f1 = vocov->f1(); 39992 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 39993 | 39994 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 39995 | 39996 | expression_node_ptr result = error_node(); 39997 | 39998 | const bool synthesis_result = 39999 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40000 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c, v1, v2, result); 40001 | 40002 | if (synthesis_result) 40003 | return result; 40004 | else if (!expr_gen.valid_operator(o2,f2)) 40005 | return error_node(); 40006 | 40007 | exprtk_debug(("((v0 o0 c) o1 v1) o2 v2\n")); 40008 | 40009 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, v2, f0, f1, f2); 40010 | } 40011 | 40012 | static inline std::string id(expression_generator<Type>& expr_gen, 40013 | const details::operator_type o0, 40014 | const details::operator_type o1, 40015 | const details::operator_type o2) 40016 | { 40017 | return details::build_string() 40018 | << "((t" << expr_gen.to_str(o0) 40019 | << "t)" << expr_gen.to_str(o1) 40020 | << "t)" << expr_gen.to_str(o2) 40021 | << "t" 40022 | } 40023 | }; 40024 | 40025 | struct synthesize_covovov_expression3 40026 | { 40027 | typedef typename covovov_t::type3 node_type; 40028 | typedef typename covovov_t::sf4_type sf4_type; 40029 | typedef typename node_type::T0 T0; 40030 | typedef typename node_type::T1 T1; 40031 | typedef typename node_type::T2 T2; 40032 | typedef typename node_type::T3 T3; 40033 | 40034 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40035 | const details::operator_type& operation, 40036 | expression_node_ptr (&branch)[2]) 40037 | { 40038 | // ((c o0 v0) o1 v1) o2 v2 40039 | typedef typename synthesize_covov_expression0::node_type lcl_covov_t; 40040 | 40041 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[0]); 40042 | const Type c = covov->t0(); 40043 | const Type& v0 = covov->t1(); 40044 | const Type& v1 = covov->t2(); 40045 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40046 | const details::operator_type o0 = expr_gen.get_operator(covov->f0()); 40047 | const details::operator_type o1 = expr_gen.get_operator(covov->f1()); 40048 | const details::operator_type o2 = operation; 40049 | 40050 | binary_functor_t f0 = covov->f0(); 40051 | binary_functor_t f1 = covov->f1(); 40052 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40053 | 40054 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40055 | 40056 | expression_node_ptr result = error_node(); 40057 | 40058 | const bool synthesis_result = 40059 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40060 | (expr_gen, id(expr_gen, o0, o1, o2), c, v0, v1, v2, result); 40061 | 40062 | if (synthesis_result) 40063 | return result; 40064 | else if (!expr_gen.valid_operator(o2,f2)) 40065 | return error_node(); 40066 | 40067 | exprtk_debug(("((c o0 v0) o1 v1) o2 v2\n")); 40068 | 40069 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, v2, f0, f1, f2); 40070 | } 40071 | 40072 | static inline std::string id(expression_generator<Type>& expr_gen, 40073 | const details::operator_type o0, 40074 | const details::operator_type o1, 40075 | const details::operator_type o2) 40076 | { 40077 | return details::build_string() 40078 | << "((t" << expr_gen.to_str(o0) 40079 | << "t)" << expr_gen.to_str(o1) 40080 | << "t)" << expr_gen.to_str(o2) 40081 | << "t" 40082 | } 40083 | }; 40084 | 40085 | struct synthesize_covocov_expression3 40086 | { 40087 | typedef typename covocov_t::type3 node_type; 40088 | typedef typename covocov_t::sf4_type sf4_type; 40089 | typedef typename node_type::T0 T0; 40090 | typedef typename node_type::T1 T1; 40091 | typedef typename node_type::T2 T2; 40092 | typedef typename node_type::T3 T3; 40093 | 40094 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40095 | const details::operator_type& operation, 40096 | expression_node_ptr (&branch)[2]) 40097 | { 40098 | // ((c0 o0 v0) o1 c1) o2 v1 40099 | typedef typename synthesize_covoc_expression0::node_type lcl_covoc_t; 40100 | 40101 | const lcl_covoc_t* covoc = static_cast<const lcl_covoc_t*>(branch[0]); 40102 | const Type c0 = covoc->t0(); 40103 | const Type& v0 = covoc->t1(); 40104 | const Type c1 = covoc->t2(); 40105 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40106 | const details::operator_type o0 = expr_gen.get_operator(covoc->f0()); 40107 | const details::operator_type o1 = expr_gen.get_operator(covoc->f1()); 40108 | const details::operator_type o2 = operation; 40109 | 40110 | binary_functor_t f0 = covoc->f0(); 40111 | binary_functor_t f1 = covoc->f1(); 40112 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40113 | 40114 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40115 | 40116 | expression_node_ptr result = error_node(); 40117 | 40118 | const bool synthesis_result = 40119 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40120 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, c1, v1, result); 40121 | 40122 | if (synthesis_result) 40123 | return result; 40124 | else if (!expr_gen.valid_operator(o2,f2)) 40125 | return error_node(); 40126 | 40127 | exprtk_debug(("((c0 o0 v0) o1 c1) o2 v1\n")); 40128 | 40129 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, c1, v1, f0, f1, f2); 40130 | } 40131 | 40132 | static inline std::string id(expression_generator<Type>& expr_gen, 40133 | const details::operator_type o0, 40134 | const details::operator_type o1, 40135 | const details::operator_type o2) 40136 | { 40137 | return details::build_string() 40138 | << "((t" << expr_gen.to_str(o0) 40139 | << "t)" << expr_gen.to_str(o1) 40140 | << "t)" << expr_gen.to_str(o2) 40141 | << "t" 40142 | } 40143 | }; 40144 | 40145 | struct synthesize_vocovoc_expression3 40146 | { 40147 | typedef typename vocovoc_t::type3 node_type; 40148 | typedef typename vocovoc_t::sf4_type sf4_type; 40149 | typedef typename node_type::T0 T0; 40150 | typedef typename node_type::T1 T1; 40151 | typedef typename node_type::T2 T2; 40152 | typedef typename node_type::T3 T3; 40153 | 40154 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40155 | const details::operator_type& operation, 40156 | expression_node_ptr (&branch)[2]) 40157 | { 40158 | // ((v0 o0 c0) o1 v1) o2 c1 40159 | typedef typename synthesize_vocov_expression0::node_type lcl_vocov_t; 40160 | 40161 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[0]); 40162 | const Type& v0 = vocov->t0(); 40163 | const Type c0 = vocov->t1(); 40164 | const Type& v1 = vocov->t2(); 40165 | const Type c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 40166 | const details::operator_type o0 = expr_gen.get_operator(vocov->f0()); 40167 | const details::operator_type o1 = expr_gen.get_operator(vocov->f1()); 40168 | const details::operator_type o2 = operation; 40169 | 40170 | binary_functor_t f0 = vocov->f0(); 40171 | binary_functor_t f1 = vocov->f1(); 40172 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40173 | 40174 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40175 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 40176 | 40177 | expression_node_ptr result = error_node(); 40178 | 40179 | const bool synthesis_result = 40180 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40181 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, v1, c1, result); 40182 | 40183 | if (synthesis_result) 40184 | return result; 40185 | else if (!expr_gen.valid_operator(o2,f2)) 40186 | return error_node(); 40187 | 40188 | exprtk_debug(("((v0 o0 c0) o1 v1) o2 c1\n")); 40189 | 40190 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, v1, c1, f0, f1, f2); 40191 | } 40192 | 40193 | static inline std::string id(expression_generator<Type>& expr_gen, 40194 | const details::operator_type o0, 40195 | const details::operator_type o1, 40196 | const details::operator_type o2) 40197 | { 40198 | return details::build_string() 40199 | << "((t" << expr_gen.to_str(o0) 40200 | << "t)" << expr_gen.to_str(o1) 40201 | << "t)" << expr_gen.to_str(o2) 40202 | << "t" 40203 | } 40204 | }; 40205 | 40206 | struct synthesize_covovoc_expression3 40207 | { 40208 | typedef typename covovoc_t::type3 node_type; 40209 | typedef typename covovoc_t::sf4_type sf4_type; 40210 | typedef typename node_type::T0 T0; 40211 | typedef typename node_type::T1 T1; 40212 | typedef typename node_type::T2 T2; 40213 | typedef typename node_type::T3 T3; 40214 | 40215 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40216 | const details::operator_type& operation, 40217 | expression_node_ptr (&branch)[2]) 40218 | { 40219 | // ((c0 o0 v0) o1 v1) o2 c1 40220 | typedef typename synthesize_covov_expression0::node_type lcl_covov_t; 40221 | 40222 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[0]); 40223 | const Type c0 = covov->t0(); 40224 | const Type& v0 = covov->t1(); 40225 | const Type& v1 = covov->t2(); 40226 | const Type c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 40227 | const details::operator_type o0 = expr_gen.get_operator(covov->f0()); 40228 | const details::operator_type o1 = expr_gen.get_operator(covov->f1()); 40229 | const details::operator_type o2 = operation; 40230 | 40231 | binary_functor_t f0 = covov->f0(); 40232 | binary_functor_t f1 = covov->f1(); 40233 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40234 | 40235 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40236 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 40237 | 40238 | expression_node_ptr result = error_node(); 40239 | 40240 | const bool synthesis_result = 40241 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40242 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, v1, c1, result); 40243 | 40244 | if (synthesis_result) 40245 | return result; 40246 | else if (!expr_gen.valid_operator(o2,f2)) 40247 | return error_node(); 40248 | 40249 | exprtk_debug(("((c0 o0 v0) o1 v1) o2 c1\n")); 40250 | 40251 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, v1, c1, f0, f1, f2); 40252 | } 40253 | 40254 | static inline std::string id(expression_generator<Type>& expr_gen, 40255 | const details::operator_type o0, 40256 | const details::operator_type o1, 40257 | const details::operator_type o2) 40258 | { 40259 | return details::build_string() 40260 | << "((t" << expr_gen.to_str(o0) 40261 | << "t)" << expr_gen.to_str(o1) 40262 | << "t)" << expr_gen.to_str(o2) 40263 | << "t" 40264 | } 40265 | }; 40266 | 40267 | struct synthesize_vococov_expression3 40268 | { 40269 | typedef typename vococov_t::type3 node_type; 40270 | typedef typename vococov_t::sf4_type sf4_type; 40271 | typedef typename node_type::T0 T0; 40272 | typedef typename node_type::T1 T1; 40273 | typedef typename node_type::T2 T2; 40274 | typedef typename node_type::T3 T3; 40275 | 40276 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40277 | const details::operator_type& operation, 40278 | expression_node_ptr (&branch)[2]) 40279 | { 40280 | // ((v0 o0 c0) o1 c1) o2 v1 40281 | typedef typename synthesize_vococ_expression0::node_type lcl_vococ_t; 40282 | 40283 | const lcl_vococ_t* vococ = static_cast<const lcl_vococ_t*>(branch[0]); 40284 | const Type& v0 = vococ->t0(); 40285 | const Type c0 = vococ->t1(); 40286 | const Type c1 = vococ->t2(); 40287 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40288 | const details::operator_type o0 = expr_gen.get_operator(vococ->f0()); 40289 | const details::operator_type o1 = expr_gen.get_operator(vococ->f1()); 40290 | const details::operator_type o2 = operation; 40291 | 40292 | binary_functor_t f0 = vococ->f0(); 40293 | binary_functor_t f1 = vococ->f1(); 40294 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40295 | 40296 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40297 | 40298 | expression_node_ptr result = error_node(); 40299 | 40300 | const bool synthesis_result = 40301 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40302 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, c1, v1, result); 40303 | 40304 | if (synthesis_result) 40305 | return result; 40306 | else if (!expr_gen.valid_operator(o2,f2)) 40307 | return error_node(); 40308 | 40309 | exprtk_debug(("((v0 o0 c0) o1 c1) o2 v1\n")); 40310 | 40311 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, c1, v1, f0, f1, f2); 40312 | } 40313 | 40314 | static inline std::string id(expression_generator<Type>& expr_gen, 40315 | const details::operator_type o0, 40316 | const details::operator_type o1, 40317 | const details::operator_type o2) 40318 | { 40319 | return details::build_string() 40320 | << "((t" << expr_gen.to_str(o0) 40321 | << "t)" << expr_gen.to_str(o1) 40322 | << "t)" << expr_gen.to_str(o2) 40323 | << "t" 40324 | } 40325 | }; 40326 | 40327 | struct synthesize_vovovov_expression4 40328 | { 40329 | typedef typename vovovov_t::type4 node_type; 40330 | typedef typename vovovov_t::sf4_type sf4_type; 40331 | typedef typename node_type::T0 T0; 40332 | typedef typename node_type::T1 T1; 40333 | typedef typename node_type::T2 T2; 40334 | typedef typename node_type::T3 T3; 40335 | 40336 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40337 | const details::operator_type& operation, 40338 | expression_node_ptr (&branch)[2]) 40339 | { 40340 | // (v0 o0 (v1 o1 v2)) o2 v3 40341 | typedef typename synthesize_vovov_expression1::node_type lcl_vovov_t; 40342 | 40343 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[0]); 40344 | const Type& v0 = vovov->t0(); 40345 | const Type& v1 = vovov->t1(); 40346 | const Type& v2 = vovov->t2(); 40347 | const Type& v3 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40348 | const details::operator_type o0 = expr_gen.get_operator(vovov->f0()); 40349 | const details::operator_type o1 = expr_gen.get_operator(vovov->f1()); 40350 | const details::operator_type o2 = operation; 40351 | 40352 | binary_functor_t f0 = vovov->f0(); 40353 | binary_functor_t f1 = vovov->f1(); 40354 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40355 | 40356 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40357 | 40358 | expression_node_ptr result = error_node(); 40359 | 40360 | const bool synthesis_result = 40361 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40362 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, v3, result); 40363 | 40364 | if (synthesis_result) 40365 | return result; 40366 | else if (!expr_gen.valid_operator(o2,f2)) 40367 | return error_node(); 40368 | 40369 | exprtk_debug(("(v0 o0 (v1 o1 v2)) o2 v3\n")); 40370 | 40371 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, v3, f0, f1, f2); 40372 | } 40373 | 40374 | static inline std::string id(expression_generator<Type>& expr_gen, 40375 | const details::operator_type o0, 40376 | const details::operator_type o1, 40377 | const details::operator_type o2) 40378 | { 40379 | return details::build_string() 40380 | << "(t" << expr_gen.to_str(o0) 40381 | << "(t" << expr_gen.to_str(o1) 40382 | << "t)" << expr_gen.to_str(o2) 40383 | << "t" 40384 | } 40385 | }; 40386 | 40387 | struct synthesize_vovovoc_expression4 40388 | { 40389 | typedef typename vovovoc_t::type4 node_type; 40390 | typedef typename vovovoc_t::sf4_type sf4_type; 40391 | typedef typename node_type::T0 T0; 40392 | typedef typename node_type::T1 T1; 40393 | typedef typename node_type::T2 T2; 40394 | typedef typename node_type::T3 T3; 40395 | 40396 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40397 | const details::operator_type& operation, 40398 | expression_node_ptr (&branch)[2]) 40399 | { 40400 | // ((v0 o0 (v1 o1 v2)) o2 c) 40401 | typedef typename synthesize_vovov_expression1::node_type lcl_vovov_t; 40402 | 40403 | const lcl_vovov_t* vovov = static_cast<const lcl_vovov_t*>(branch[0]); 40404 | const Type& v0 = vovov->t0(); 40405 | const Type& v1 = vovov->t1(); 40406 | const Type& v2 = vovov->t2(); 40407 | const Type c = static_cast<details::literal_node<Type>*>(branch[1])->value(); 40408 | const details::operator_type o0 = expr_gen.get_operator(vovov->f0()); 40409 | const details::operator_type o1 = expr_gen.get_operator(vovov->f1()); 40410 | const details::operator_type o2 = operation; 40411 | 40412 | binary_functor_t f0 = vovov->f0(); 40413 | binary_functor_t f1 = vovov->f1(); 40414 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40415 | 40416 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40417 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 40418 | 40419 | expression_node_ptr result = error_node(); 40420 | 40421 | const bool synthesis_result = 40422 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40423 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, v2, c, result); 40424 | 40425 | if (synthesis_result) 40426 | return result; 40427 | else if (!expr_gen.valid_operator(o2,f2)) 40428 | return error_node(); 40429 | 40430 | exprtk_debug(("((v0 o0 (v1 o1 v2)) o2 c)\n")); 40431 | 40432 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, v2, c, f0, f1, f2); 40433 | } 40434 | 40435 | static inline std::string id(expression_generator<Type>& expr_gen, 40436 | const details::operator_type o0, 40437 | const details::operator_type o1, 40438 | const details::operator_type o2) 40439 | { 40440 | return details::build_string() 40441 | << "(t" << expr_gen.to_str(o0) 40442 | << "(t" << expr_gen.to_str(o1) 40443 | << "t)" << expr_gen.to_str(o2) 40444 | << "t" 40445 | } 40446 | }; 40447 | 40448 | struct synthesize_vovocov_expression4 40449 | { 40450 | typedef typename vovocov_t::type4 node_type; 40451 | typedef typename vovocov_t::sf4_type sf4_type; 40452 | typedef typename node_type::T0 T0; 40453 | typedef typename node_type::T1 T1; 40454 | typedef typename node_type::T2 T2; 40455 | typedef typename node_type::T3 T3; 40456 | 40457 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40458 | const details::operator_type& operation, 40459 | expression_node_ptr (&branch)[2]) 40460 | { 40461 | // ((v0 o0 (v1 o1 c)) o2 v1) 40462 | typedef typename synthesize_vovoc_expression1::node_type lcl_vovoc_t; 40463 | 40464 | const lcl_vovoc_t* vovoc = static_cast<const lcl_vovoc_t*>(branch[0]); 40465 | const Type& v0 = vovoc->t0(); 40466 | const Type& v1 = vovoc->t1(); 40467 | const Type c = vovoc->t2(); 40468 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40469 | const details::operator_type o0 = expr_gen.get_operator(vovoc->f0()); 40470 | const details::operator_type o1 = expr_gen.get_operator(vovoc->f1()); 40471 | const details::operator_type o2 = operation; 40472 | 40473 | binary_functor_t f0 = vovoc->f0(); 40474 | binary_functor_t f1 = vovoc->f1(); 40475 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40476 | 40477 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40478 | 40479 | expression_node_ptr result = error_node(); 40480 | 40481 | const bool synthesis_result = 40482 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40483 | (expr_gen, id(expr_gen, o0, o1, o2), v0, v1, c, v2, result); 40484 | 40485 | if (synthesis_result) 40486 | return result; 40487 | else if (!expr_gen.valid_operator(o2,f2)) 40488 | return error_node(); 40489 | 40490 | exprtk_debug(("((v0 o0 (v1 o1 c)) o2 v1)\n")); 40491 | 40492 | return node_type::allocate(*(expr_gen.node_allocator_), v0, v1, c, v2, f0, f1, f2); 40493 | } 40494 | 40495 | static inline std::string id(expression_generator<Type>& expr_gen, 40496 | const details::operator_type o0, 40497 | const details::operator_type o1, 40498 | const details::operator_type o2) 40499 | { 40500 | return details::build_string() 40501 | << "(t" << expr_gen.to_str(o0) 40502 | << "(t" << expr_gen.to_str(o1) 40503 | << "t)" << expr_gen.to_str(o2) 40504 | << "t" 40505 | } 40506 | }; 40507 | 40508 | struct synthesize_vocovov_expression4 40509 | { 40510 | typedef typename vocovov_t::type4 node_type; 40511 | typedef typename vocovov_t::sf4_type sf4_type; 40512 | typedef typename node_type::T0 T0; 40513 | typedef typename node_type::T1 T1; 40514 | typedef typename node_type::T2 T2; 40515 | typedef typename node_type::T3 T3; 40516 | 40517 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40518 | const details::operator_type& operation, 40519 | expression_node_ptr (&branch)[2]) 40520 | { 40521 | // ((v0 o0 (c o1 v1)) o2 v2) 40522 | typedef typename synthesize_vocov_expression1::node_type lcl_vocov_t; 40523 | 40524 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[0]); 40525 | const Type& v0 = vocov->t0(); 40526 | const Type c = vocov->t1(); 40527 | const Type& v1 = vocov->t2(); 40528 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40529 | const details::operator_type o0 = expr_gen.get_operator(vocov->f0()); 40530 | const details::operator_type o1 = expr_gen.get_operator(vocov->f1()); 40531 | const details::operator_type o2 = operation; 40532 | 40533 | binary_functor_t f0 = vocov->f0(); 40534 | binary_functor_t f1 = vocov->f1(); 40535 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40536 | 40537 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40538 | expression_node_ptr result = error_node(); 40539 | 40540 | const bool synthesis_result = 40541 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40542 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c, v1, v2, result); 40543 | 40544 | if (synthesis_result) 40545 | return result; 40546 | else if (!expr_gen.valid_operator(o2,f2)) 40547 | return error_node(); 40548 | 40549 | exprtk_debug(("((v0 o0 (c o1 v1)) o2 v2)\n")); 40550 | 40551 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c, v1, v2, f0, f1, f2); 40552 | } 40553 | 40554 | static inline std::string id(expression_generator<Type>& expr_gen, 40555 | const details::operator_type o0, 40556 | const details::operator_type o1, 40557 | const details::operator_type o2) 40558 | { 40559 | return details::build_string() 40560 | << "(t" << expr_gen.to_str(o0) 40561 | << "(t" << expr_gen.to_str(o1) 40562 | << "t)" << expr_gen.to_str(o2) 40563 | << "t" 40564 | } 40565 | }; 40566 | 40567 | struct synthesize_covovov_expression4 40568 | { 40569 | typedef typename covovov_t::type4 node_type; 40570 | typedef typename covovov_t::sf4_type sf4_type; 40571 | typedef typename node_type::T0 T0; 40572 | typedef typename node_type::T1 T1; 40573 | typedef typename node_type::T2 T2; 40574 | typedef typename node_type::T3 T3; 40575 | 40576 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40577 | const details::operator_type& operation, 40578 | expression_node_ptr (&branch)[2]) 40579 | { 40580 | // ((c o0 (v0 o1 v1)) o2 v2) 40581 | typedef typename synthesize_covov_expression1::node_type lcl_covov_t; 40582 | 40583 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[0]); 40584 | const Type c = covov->t0(); 40585 | const Type& v0 = covov->t1(); 40586 | const Type& v1 = covov->t2(); 40587 | const Type& v2 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40588 | const details::operator_type o0 = expr_gen.get_operator(covov->f0()); 40589 | const details::operator_type o1 = expr_gen.get_operator(covov->f1()); 40590 | const details::operator_type o2 = operation; 40591 | 40592 | binary_functor_t f0 = covov->f0(); 40593 | binary_functor_t f1 = covov->f1(); 40594 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40595 | 40596 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40597 | 40598 | expression_node_ptr result = error_node(); 40599 | 40600 | const bool synthesis_result = 40601 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40602 | (expr_gen, id(expr_gen, o0, o1, o2), c, v0, v1, v2, result); 40603 | 40604 | if (synthesis_result) 40605 | return result; 40606 | else if (!expr_gen.valid_operator(o2,f2)) 40607 | return error_node(); 40608 | 40609 | exprtk_debug(("((c o0 (v0 o1 v1)) o2 v2)\n")); 40610 | 40611 | return node_type::allocate(*(expr_gen.node_allocator_), c, v0, v1, v2, f0, f1, f2); 40612 | } 40613 | 40614 | static inline std::string id(expression_generator<Type>& expr_gen, 40615 | const details::operator_type o0, 40616 | const details::operator_type o1, 40617 | const details::operator_type o2) 40618 | { 40619 | return details::build_string() 40620 | << "(t" << expr_gen.to_str(o0) 40621 | << "(t" << expr_gen.to_str(o1) 40622 | << "t)" << expr_gen.to_str(o2) 40623 | << "t" 40624 | } 40625 | }; 40626 | 40627 | struct synthesize_covocov_expression4 40628 | { 40629 | typedef typename covocov_t::type4 node_type; 40630 | typedef typename covocov_t::sf4_type sf4_type; 40631 | typedef typename node_type::T0 T0; 40632 | typedef typename node_type::T1 T1; 40633 | typedef typename node_type::T2 T2; 40634 | typedef typename node_type::T3 T3; 40635 | 40636 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40637 | const details::operator_type& operation, 40638 | expression_node_ptr (&branch)[2]) 40639 | { 40640 | // ((c0 o0 (v0 o1 c1)) o2 v1) 40641 | typedef typename synthesize_covoc_expression1::node_type lcl_covoc_t; 40642 | 40643 | const lcl_covoc_t* covoc = static_cast<const lcl_covoc_t*>(branch[0]); 40644 | const Type c0 = covoc->t0(); 40645 | const Type& v0 = covoc->t1(); 40646 | const Type c1 = covoc->t2(); 40647 | const Type& v1 = static_cast<details::variable_node<Type>*>(branch[1])->ref(); 40648 | const details::operator_type o0 = expr_gen.get_operator(covoc->f0()); 40649 | const details::operator_type o1 = expr_gen.get_operator(covoc->f1()); 40650 | const details::operator_type o2 = operation; 40651 | 40652 | binary_functor_t f0 = covoc->f0(); 40653 | binary_functor_t f1 = covoc->f1(); 40654 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40655 | 40656 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40657 | 40658 | expression_node_ptr result = error_node(); 40659 | 40660 | const bool synthesis_result = 40661 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40662 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, c1, v1, result); 40663 | 40664 | if (synthesis_result) 40665 | return result; 40666 | else if (!expr_gen.valid_operator(o2,f2)) 40667 | return error_node(); 40668 | 40669 | exprtk_debug(("((c0 o0 (v0 o1 c1)) o2 v1)\n")); 40670 | 40671 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, c1, v1, f0, f1, f2); 40672 | } 40673 | 40674 | static inline std::string id(expression_generator<Type>& expr_gen, 40675 | const details::operator_type o0, 40676 | const details::operator_type o1, 40677 | const details::operator_type o2) 40678 | { 40679 | return details::build_string() 40680 | << "(t" << expr_gen.to_str(o0) 40681 | << "(t" << expr_gen.to_str(o1) 40682 | << "t)" << expr_gen.to_str(o2) 40683 | << "t" 40684 | } 40685 | }; 40686 | 40687 | struct synthesize_vocovoc_expression4 40688 | { 40689 | typedef typename vocovoc_t::type4 node_type; 40690 | typedef typename vocovoc_t::sf4_type sf4_type; 40691 | typedef typename node_type::T0 T0; 40692 | typedef typename node_type::T1 T1; 40693 | typedef typename node_type::T2 T2; 40694 | typedef typename node_type::T3 T3; 40695 | 40696 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40697 | const details::operator_type& operation, 40698 | expression_node_ptr (&branch)[2]) 40699 | { 40700 | // ((v0 o0 (c0 o1 v1)) o2 c1) 40701 | typedef typename synthesize_vocov_expression1::node_type lcl_vocov_t; 40702 | 40703 | const lcl_vocov_t* vocov = static_cast<const lcl_vocov_t*>(branch[0]); 40704 | const Type& v0 = vocov->t0(); 40705 | const Type c0 = vocov->t1(); 40706 | const Type& v1 = vocov->t2(); 40707 | const Type c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 40708 | const details::operator_type o0 = expr_gen.get_operator(vocov->f0()); 40709 | const details::operator_type o1 = expr_gen.get_operator(vocov->f1()); 40710 | const details::operator_type o2 = operation; 40711 | 40712 | binary_functor_t f0 = vocov->f0(); 40713 | binary_functor_t f1 = vocov->f1(); 40714 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40715 | 40716 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40717 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 40718 | 40719 | expression_node_ptr result = error_node(); 40720 | 40721 | const bool synthesis_result = 40722 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40723 | (expr_gen, id(expr_gen, o0, o1, o2), v0, c0, v1, c1, result); 40724 | 40725 | if (synthesis_result) 40726 | return result; 40727 | else if (!expr_gen.valid_operator(o2,f2)) 40728 | return error_node(); 40729 | 40730 | exprtk_debug(("((v0 o0 (c0 o1 v1)) o2 c1)\n")); 40731 | 40732 | return node_type::allocate(*(expr_gen.node_allocator_), v0, c0, v1, c1, f0, f1, f2); 40733 | } 40734 | 40735 | static inline std::string id(expression_generator<Type>& expr_gen, 40736 | const details::operator_type o0, 40737 | const details::operator_type o1, 40738 | const details::operator_type o2) 40739 | { 40740 | return details::build_string() 40741 | << "(t" << expr_gen.to_str(o0) 40742 | << "(t" << expr_gen.to_str(o1) 40743 | << "t)" << expr_gen.to_str(o2) 40744 | << "t" 40745 | } 40746 | }; 40747 | 40748 | struct synthesize_covovoc_expression4 40749 | { 40750 | typedef typename covovoc_t::type4 node_type; 40751 | typedef typename covovoc_t::sf4_type sf4_type; 40752 | typedef typename node_type::T0 T0; 40753 | typedef typename node_type::T1 T1; 40754 | typedef typename node_type::T2 T2; 40755 | typedef typename node_type::T3 T3; 40756 | 40757 | static inline expression_node_ptr process(expression_generator<Type>& expr_gen, 40758 | const details::operator_type& operation, 40759 | expression_node_ptr (&branch)[2]) 40760 | { 40761 | // ((c0 o0 (v0 o1 v1)) o2 c1) 40762 | typedef typename synthesize_covov_expression1::node_type lcl_covov_t; 40763 | 40764 | const lcl_covov_t* covov = static_cast<const lcl_covov_t*>(branch[0]); 40765 | const Type c0 = covov->t0(); 40766 | const Type& v0 = covov->t1(); 40767 | const Type& v1 = covov->t2(); 40768 | const Type c1 = static_cast<details::literal_node<Type>*>(branch[1])->value(); 40769 | const details::operator_type o0 = expr_gen.get_operator(covov->f0()); 40770 | const details::operator_type o1 = expr_gen.get_operator(covov->f1()); 40771 | const details::operator_type o2 = operation; 40772 | 40773 | binary_functor_t f0 = covov->f0(); 40774 | binary_functor_t f1 = covov->f1(); 40775 | binary_functor_t f2 = reinterpret_cast<binary_functor_t>(0); 40776 | 40777 | details::free_node(*(expr_gen.node_allocator_),branch[0]); 40778 | details::free_node(*(expr_gen.node_allocator_),branch[1]); 40779 | 40780 | expression_node_ptr result = error_node(); 40781 | 40782 | const bool synthesis_result = 40783 | synthesize_sf4ext_expression::template compile<T0, T1, T2, T3> 40784 | (expr_gen, id(expr_gen, o0, o1, o2), c0, v0, v1, c1, result); 40785 | 40786 | if (synthesis_result) 40787 | return result; 40788 | else if (!expr_gen.valid_operator(o2,f2)) 40789 | return error_node(); 40790 | 40791 | exprtk_debug(("((c0 o0 (v0 o1 v1)) o2 c1)\n")); 40792 | 40793 | return node_type::allocate(*(expr_gen.node_allocator_), c0, v0, v1, c1, f0, f1, f2); 40794 | } 40795 | 40796 | static inline std::string id(expression_generator<Type>& expr_gen, 40797 | const details::operator_type o0, 40798 | const details::operator_type o1, 40799 | const details::operator_type o2) 40800 | { 40801 | return details::build_string() 40802 | << "(t" << expr_gen.to_str(o0) 40803 | << "(t" << expr_gen.to_str(o1) 40804 | << "t)" << expr_gen.to_str(o2) 40805 | << "t" 40806 | } 40807 | }; 40808 | 40809 | struct synthesize_vococov_expression4 40810 | { 40811 | typedef typename vococov_t::type4 node_type; 40812 | static inline expression_node_ptr process(expression_generator<Type>&, 40813 | const details::operator_type&, 40814 | expression_node_ptr (&)[2]) 40815 | { 40816 | // ((v0 o0 (c0 o1 c1)) o2 v1) - Not possible 40817 | exprtk_debug(("((v0 o0 (c0 o1 c1)) o2 v1) - Not possible\n")); 40818 | return error_node(); 40819 | } 40820 | 40821 | static inline std::string id(expression_generator<Type>&, 40822 | const details::operator_type, 40823 | const details::operator_type, 40824 | const details::operator_type) 40825 | { 40826 | return "INVALID" 40827 | } 40828 | }; 40829 | #endif 40830 | 40831 | inline expression_node_ptr synthesize_uvouv_expression(const details::operator_type& operation, expression_node_ptr (&branch)[2]) 40832 | { 40833 | // Definition: uv o uv 40834 | details::operator_type o0 = static_cast<details::uv_base_node<Type>*>(branch[0])->operation(); 40835 | details::operator_type o1 = static_cast<details::uv_base_node<Type>*>(branch[1])->operation(); 40836 | const Type& v0 = static_cast<details::uv_base_node<Type>*>(branch[0])->v(); 40837 | const Type& v1 = static_cast<details::uv_base_node<Type>*>(branch[1])->v(); 40838 | unary_functor_t u0 = reinterpret_cast<unary_functor_t> (0); 40839 | unary_functor_t u1 = reinterpret_cast<unary_functor_t> (0); 40840 | binary_functor_t f = reinterpret_cast<binary_functor_t>(0); 40841 | 40842 | if (!valid_operator(o0,u0)) 40843 | return error_node(); 40844 | else if (!valid_operator(o1,u1)) 40845 | return error_node(); 40846 | else if (!valid_operator(operation,f)) 40847 | return error_node(); 40848 | 40849 | expression_node_ptr result = error_node(); 40850 | 40851 | if ( 40852 | (details::e_neg == o0) && 40853 | (details::e_neg == o1) 40854 | ) 40855 | { 40856 | switch (operation) 40857 | { 40858 | // (-v0 + -v1) --> -(v0 + v1) 40859 | case details::e_add : result = (*this)(details::e_neg, 40860 | node_allocator_-> 40861 | allocate_rr<typename details:: 40862 | vov_node<Type,details::add_op<Type> > >(v0, v1)); 40863 | exprtk_debug(("(-v0 + -v1) --> -(v0 + v1)\n")); 40864 | break; 40865 | 40866 | // (-v0 - -v1) --> (v1 - v0) 40867 | case details::e_sub : result = node_allocator_-> 40868 | allocate_rr<typename details:: 40869 | vov_node<Type,details::sub_op<Type> > >(v1, v0); 40870 | exprtk_debug(("(-v0 - -v1) --> (v1 - v0)\n")); 40871 | break; 40872 | 40873 | // (-v0 * -v1) --> (v0 * v1) 40874 | case details::e_mul : result = node_allocator_-> 40875 | allocate_rr<typename details:: 40876 | vov_node<Type,details::mul_op<Type> > >(v0, v1); 40877 | exprtk_debug(("(-v0 * -v1) --> (v0 * v1)\n")); 40878 | break; 40879 | 40880 | // (-v0 / -v1) --> (v0 / v1) 40881 | case details::e_div : result = node_allocator_-> 40882 | allocate_rr<typename details:: 40883 | vov_node<Type,details::div_op<Type> > >(v0, v1); 40884 | exprtk_debug(("(-v0 / -v1) --> (v0 / v1)\n")); 40885 | break; 40886 | 40887 | default : break; 40888 | } 40889 | } 40890 | 40891 | if (0 == result) 40892 | { 40893 | result = node_allocator_-> 40894 | allocate_rrrrr<typename details::uvouv_node<Type> >(v0, v1, u0, u1, f); 40895 | } 40896 | 40897 | details::free_all_nodes(*node_allocator_,branch); 40898 | return result; 40899 | } 40900 | 40901 | #undef basic_opr_switch_statements 40902 | #undef extended_opr_switch_statements 40903 | #undef unary_opr_switch_statements 40904 | 40905 | #ifndef exprtk_disable_string_capabilities 40906 | 40907 | #define string_opr_switch_statements \ 40908 | case_stmt(details::e_lt , details::lt_op ) \ 40909 | case_stmt(details::e_lte , details::lte_op ) \ 40910 | case_stmt(details::e_gt , details::gt_op ) \ 40911 | case_stmt(details::e_gte , details::gte_op ) \ 40912 | case_stmt(details::e_eq , details::eq_op ) \ 40913 | case_stmt(details::e_ne , details::ne_op ) \ 40914 | case_stmt(details::e_in , details::in_op ) \ 40915 | case_stmt(details::e_like , details::like_op ) \ 40916 | case_stmt(details::e_ilike , details::ilike_op) \ 40917 | 40918 | template <typename T0, typename T1> 40919 | inline expression_node_ptr synthesize_str_xrox_expression_impl(const details::operator_type& opr, 40920 | T0 s0, T1 s1, 40921 | range_t rp0) 40922 | { 40923 | switch (opr) 40924 | { 40925 | #define case_stmt(op0, op1) \ 40926 | case op0 : return node_allocator_-> \ 40927 | allocate_ttt<typename details::str_xrox_node<Type,T0,T1,range_t,op1<Type> >,T0,T1> \ 40928 | (s0, s1, rp0); \ 40929 | 40930 | string_opr_switch_statements 40931 | #undef case_stmt 40932 | default : return error_node(); 40933 | } 40934 | } 40935 | 40936 | template <typename T0, typename T1> 40937 | inline expression_node_ptr synthesize_str_xoxr_expression_impl(const details::operator_type& opr, 40938 | T0 s0, T1 s1, 40939 | range_t rp1) 40940 | { 40941 | switch (opr) 40942 | { 40943 | #define case_stmt(op0, op1) \ 40944 | case op0 : return node_allocator_-> \ 40945 | allocate_ttt<typename details::str_xoxr_node<Type,T0,T1,range_t,op1<Type> >,T0,T1> \ 40946 | (s0, s1, rp1); \ 40947 | 40948 | string_opr_switch_statements 40949 | #undef case_stmt 40950 | default : return error_node(); 40951 | } 40952 | } 40953 | 40954 | template <typename T0, typename T1> 40955 | inline expression_node_ptr synthesize_str_xroxr_expression_impl(const details::operator_type& opr, 40956 | T0 s0, T1 s1, 40957 | range_t rp0, range_t rp1) 40958 | { 40959 | switch (opr) 40960 | { 40961 | #define case_stmt(op0, op1) \ 40962 | case op0 : return node_allocator_-> \ 40963 | allocate_tttt<typename details::str_xroxr_node<Type,T0,T1,range_t,op1<Type> >,T0,T1> \ 40964 | (s0, s1, rp0, rp1); \ 40965 | 40966 | string_opr_switch_statements 40967 | #undef case_stmt 40968 | default : return error_node(); 40969 | } 40970 | } 40971 | 40972 | template <typename T0, typename T1> 40973 | inline expression_node_ptr synthesize_sos_expression_impl(const details::operator_type& opr, T0 s0, T1 s1) 40974 | { 40975 | switch (opr) 40976 | { 40977 | #define case_stmt(op0, op1) \ 40978 | case op0 : return node_allocator_-> \ 40979 | allocate_tt<typename details::sos_node<Type,T0,T1,op1<Type> >,T0,T1>(s0, s1); \ 40980 | 40981 | string_opr_switch_statements 40982 | #undef case_stmt 40983 | default : return error_node(); 40984 | } 40985 | } 40986 | 40987 | inline expression_node_ptr synthesize_sos_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 40988 | { 40989 | std::string& s0 = static_cast<details::stringvar_node<Type>*>(branch[0])->ref(); 40990 | std::string& s1 = static_cast<details::stringvar_node<Type>*>(branch[1])->ref(); 40991 | 40992 | return synthesize_sos_expression_impl<std::string&,std::string&>(opr, s0, s1); 40993 | } 40994 | 40995 | inline expression_node_ptr synthesize_sros_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 40996 | { 40997 | std::string& s0 = static_cast<details::string_range_node<Type>*>(branch[0])->ref (); 40998 | std::string& s1 = static_cast<details::stringvar_node<Type>*> (branch[1])->ref (); 40999 | range_t rp0 = static_cast<details::string_range_node<Type>*>(branch[0])->range(); 41000 | 41001 | static_cast<details::string_range_node<Type>*>(branch[0])->range_ref().clear(); 41002 | 41003 | details::free_node(*node_allocator_,branch[0]); 41004 | 41005 | return synthesize_str_xrox_expression_impl<std::string&,std::string&>(opr, s0, s1, rp0); 41006 | } 41007 | 41008 | inline expression_node_ptr synthesize_sosr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41009 | { 41010 | std::string& s0 = static_cast<details::stringvar_node<Type>*> (branch[0])->ref (); 41011 | std::string& s1 = static_cast<details::string_range_node<Type>*>(branch[1])->ref (); 41012 | range_t rp1 = static_cast<details::string_range_node<Type>*>(branch[1])->range(); 41013 | 41014 | static_cast<details::string_range_node<Type>*>(branch[1])->range_ref().clear(); 41015 | 41016 | details::free_node(*node_allocator_,branch[1]); 41017 | 41018 | return synthesize_str_xoxr_expression_impl<std::string&,std::string&>(opr, s0, s1, rp1); 41019 | } 41020 | 41021 | inline expression_node_ptr synthesize_socsr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41022 | { 41023 | std::string& s0 = static_cast<details::stringvar_node<Type>*> (branch[0])->ref (); 41024 | std::string s1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->str (); 41025 | range_t rp1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->range(); 41026 | 41027 | static_cast<details::const_string_range_node<Type>*>(branch[1])->range_ref().clear(); 41028 | 41029 | details::free_node(*node_allocator_,branch[1]); 41030 | 41031 | return synthesize_str_xoxr_expression_impl<std::string&, const std::string>(opr, s0, s1, rp1); 41032 | } 41033 | 41034 | inline expression_node_ptr synthesize_srosr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41035 | { 41036 | std::string& s0 = static_cast<details::string_range_node<Type>*>(branch[0])->ref (); 41037 | std::string& s1 = static_cast<details::string_range_node<Type>*>(branch[1])->ref (); 41038 | range_t rp0 = static_cast<details::string_range_node<Type>*>(branch[0])->range(); 41039 | range_t rp1 = static_cast<details::string_range_node<Type>*>(branch[1])->range(); 41040 | 41041 | static_cast<details::string_range_node<Type>*>(branch[0])->range_ref().clear(); 41042 | static_cast<details::string_range_node<Type>*>(branch[1])->range_ref().clear(); 41043 | 41044 | details::free_node(*node_allocator_,branch[0]); 41045 | details::free_node(*node_allocator_,branch[1]); 41046 | 41047 | return synthesize_str_xroxr_expression_impl<std::string&,std::string&>(opr, s0, s1, rp0, rp1); 41048 | } 41049 | 41050 | inline expression_node_ptr synthesize_socs_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41051 | { 41052 | std::string& s0 = static_cast< details::stringvar_node<Type>*>(branch[0])->ref(); 41053 | std::string s1 = static_cast<details::string_literal_node<Type>*>(branch[1])->str(); 41054 | 41055 | details::free_node(*node_allocator_,branch[1]); 41056 | 41057 | return synthesize_sos_expression_impl<std::string&, const std::string>(opr, s0, s1); 41058 | } 41059 | 41060 | inline expression_node_ptr synthesize_csos_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41061 | { 41062 | std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str(); 41063 | std::string& s1 = static_cast<details::stringvar_node<Type>* >(branch[1])->ref(); 41064 | 41065 | details::free_node(*node_allocator_,branch[0]); 41066 | 41067 | return synthesize_sos_expression_impl<const std::string,std::string&>(opr, s0, s1); 41068 | } 41069 | 41070 | inline expression_node_ptr synthesize_csosr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41071 | { 41072 | std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str (); 41073 | std::string& s1 = static_cast<details::string_range_node<Type>* >(branch[1])->ref (); 41074 | range_t rp1 = static_cast<details::string_range_node<Type>* >(branch[1])->range(); 41075 | 41076 | static_cast<details::string_range_node<Type>*>(branch[1])->range_ref().clear(); 41077 | 41078 | details::free_node(*node_allocator_,branch[0]); 41079 | details::free_node(*node_allocator_,branch[1]); 41080 | 41081 | return synthesize_str_xoxr_expression_impl<const std::string,std::string&>(opr, s0, s1, rp1); 41082 | } 41083 | 41084 | inline expression_node_ptr synthesize_srocs_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41085 | { 41086 | std::string& s0 = static_cast<details::string_range_node<Type>* >(branch[0])->ref (); 41087 | std::string s1 = static_cast<details::string_literal_node<Type>*>(branch[1])->str (); 41088 | range_t rp0 = static_cast<details::string_range_node<Type>* >(branch[0])->range(); 41089 | 41090 | static_cast<details::string_range_node<Type>*>(branch[0])->range_ref().clear(); 41091 | 41092 | details::free_node(*node_allocator_,branch[0]); 41093 | details::free_node(*node_allocator_,branch[1]); 41094 | 41095 | return synthesize_str_xrox_expression_impl<std::string&, const std::string>(opr, s0, s1, rp0); 41096 | } 41097 | 41098 | inline expression_node_ptr synthesize_srocsr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41099 | { 41100 | std::string& s0 = static_cast<details::string_range_node<Type>* >(branch[0])->ref (); 41101 | std::string s1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->str (); 41102 | range_t rp0 = static_cast<details::string_range_node<Type>* >(branch[0])->range(); 41103 | range_t rp1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->range(); 41104 | 41105 | static_cast<details::string_range_node<Type>*> (branch[0])->range_ref().clear(); 41106 | static_cast<details::const_string_range_node<Type>*>(branch[1])->range_ref().clear(); 41107 | 41108 | details::free_node(*node_allocator_,branch[0]); 41109 | details::free_node(*node_allocator_,branch[1]); 41110 | 41111 | return synthesize_str_xroxr_expression_impl<std::string&, const std::string>(opr, s0, s1, rp0, rp1); 41112 | } 41113 | 41114 | inline expression_node_ptr synthesize_csocs_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41115 | { 41116 | const std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str(); 41117 | const std::string s1 = static_cast<details::string_literal_node<Type>*>(branch[1])->str(); 41118 | 41119 | expression_node_ptr result = error_node(); 41120 | 41121 | if (details::e_add == opr) 41122 | result = node_allocator_->allocate_c<details::string_literal_node<Type> >(s0 + s1); 41123 | else if (details::e_in == opr) 41124 | result = node_allocator_->allocate_c<details::literal_node<Type> >(details::in_op <Type>::process(s0,s1)); 41125 | else if (details::e_like == opr) 41126 | result = node_allocator_->allocate_c<details::literal_node<Type> >(details::like_op <Type>::process(s0,s1)); 41127 | else if (details::e_ilike == opr) 41128 | result = node_allocator_->allocate_c<details::literal_node<Type> >(details::ilike_op<Type>::process(s0,s1)); 41129 | else 41130 | { 41131 | expression_node_ptr temp = synthesize_sos_expression_impl<const std::string, const std::string>(opr, s0, s1); 41132 | 41133 | const Type v = temp->value(); 41134 | 41135 | details::free_node(*node_allocator_,temp); 41136 | 41137 | result = node_allocator_->allocate<literal_node_t>(v); 41138 | } 41139 | 41140 | details::free_all_nodes(*node_allocator_,branch); 41141 | 41142 | return result; 41143 | } 41144 | 41145 | inline expression_node_ptr synthesize_csocsr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41146 | { 41147 | const std::string s0 = static_cast<details::string_literal_node<Type>* >(branch[0])->str (); 41148 | std::string s1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->str (); 41149 | range_t rp1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->range(); 41150 | 41151 | static_cast<details::const_string_range_node<Type>*>(branch[1])->range_ref().clear(); 41152 | 41153 | details::free_node(*node_allocator_,branch[0]); 41154 | details::free_node(*node_allocator_,branch[1]); 41155 | 41156 | return synthesize_str_xoxr_expression_impl<const std::string, const std::string>(opr, s0, s1, rp1); 41157 | } 41158 | 41159 | inline expression_node_ptr synthesize_csros_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41160 | { 41161 | std::string s0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->str (); 41162 | std::string& s1 = static_cast<details::stringvar_node<Type>* >(branch[1])->ref (); 41163 | range_t rp0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->range(); 41164 | 41165 | static_cast<details::const_string_range_node<Type>*>(branch[0])->range_ref().clear(); 41166 | 41167 | details::free_node(*node_allocator_,branch[0]); 41168 | 41169 | return synthesize_str_xrox_expression_impl<const std::string,std::string&>(opr, s0, s1, rp0); 41170 | } 41171 | 41172 | inline expression_node_ptr synthesize_csrosr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41173 | { 41174 | const std::string s0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->str (); 41175 | std::string& s1 = static_cast<details::string_range_node<Type>* >(branch[1])->ref (); 41176 | const range_t rp0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->range(); 41177 | const range_t rp1 = static_cast<details::string_range_node<Type>* >(branch[1])->range(); 41178 | 41179 | static_cast<details::const_string_range_node<Type>*>(branch[0])->range_ref().clear(); 41180 | static_cast<details::string_range_node<Type>*> (branch[1])->range_ref().clear(); 41181 | 41182 | details::free_node(*node_allocator_,branch[0]); 41183 | details::free_node(*node_allocator_,branch[1]); 41184 | 41185 | return synthesize_str_xroxr_expression_impl<const std::string,std::string&>(opr, s0, s1, rp0, rp1); 41186 | } 41187 | 41188 | inline expression_node_ptr synthesize_csrocs_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41189 | { 41190 | const std::string s0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->str (); 41191 | const std::string s1 = static_cast<details::string_literal_node<Type>* >(branch[1])->str (); 41192 | const range_t rp0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->range(); 41193 | 41194 | static_cast<details::const_string_range_node<Type>*>(branch[0])->range_ref().clear(); 41195 | 41196 | details::free_all_nodes(*node_allocator_,branch); 41197 | 41198 | return synthesize_str_xrox_expression_impl<const std::string,std::string>(opr, s0, s1, rp0); 41199 | } 41200 | 41201 | inline expression_node_ptr synthesize_csrocsr_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41202 | { 41203 | const std::string s0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->str (); 41204 | const std::string s1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->str (); 41205 | const range_t rp0 = static_cast<details::const_string_range_node<Type>*>(branch[0])->range(); 41206 | const range_t rp1 = static_cast<details::const_string_range_node<Type>*>(branch[1])->range(); 41207 | 41208 | static_cast<details::const_string_range_node<Type>*>(branch[0])->range_ref().clear(); 41209 | static_cast<details::const_string_range_node<Type>*>(branch[1])->range_ref().clear(); 41210 | 41211 | details::free_all_nodes(*node_allocator_,branch); 41212 | 41213 | return synthesize_str_xroxr_expression_impl<const std::string, const std::string>(opr, s0, s1, rp0, rp1); 41214 | } 41215 | 41216 | inline expression_node_ptr synthesize_strogen_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41217 | { 41218 | switch (opr) 41219 | { 41220 | #define case_stmt(op0, op1) \ 41221 | case op0 : return node_allocator_-> \ 41222 | allocate_ttt<typename details::str_sogens_node<Type,op1<Type> > > \ 41223 | (opr, branch[0], branch[1]); \ 41224 | 41225 | string_opr_switch_statements 41226 | #undef case_stmt 41227 | default : return error_node(); 41228 | } 41229 | } 41230 | 41231 | #undef string_opr_switch_statements 41232 | #endif 41233 | 41234 | #ifndef exprtk_disable_string_capabilities 41235 | inline expression_node_ptr synthesize_string_expression(const details::operator_type& opr, expression_node_ptr (&branch)[2]) 41236 | { 41237 | if ((0 == branch[0]) || (0 == branch[1])) 41238 | { 41239 | details::free_all_nodes(*node_allocator_,branch); 41240 | 41241 | return error_node(); 41242 | } 41243 | 41244 | const bool b0_is_s = details::is_string_node (branch[0]); 41245 | const bool b0_is_cs = details::is_const_string_node (branch[0]); 41246 | const bool b0_is_sr = details::is_string_range_node (branch[0]); 41247 | const bool b0_is_csr = details::is_const_string_range_node(branch[0]); 41248 | 41249 | const bool b1_is_s = details::is_string_node (branch[1]); 41250 | const bool b1_is_cs = details::is_const_string_node (branch[1]); 41251 | const bool b1_is_sr = details::is_string_range_node (branch[1]); 41252 | const bool b1_is_csr = details::is_const_string_range_node(branch[1]); 41253 | 41254 | const bool b0_is_gen = details::is_string_assignment_node (branch[0]) || 41255 | details::is_genricstring_range_node(branch[0]) || 41256 | details::is_string_concat_node (branch[0]) || 41257 | details::is_string_function_node (branch[0]) || 41258 | details::is_string_condition_node (branch[0]) || 41259 | details::is_string_ccondition_node (branch[0]) || 41260 | details::is_string_vararg_node (branch[0]) ; 41261 | 41262 | const bool b1_is_gen = details::is_string_assignment_node (branch[1]) || 41263 | details::is_genricstring_range_node(branch[1]) || 41264 | details::is_string_concat_node (branch[1]) || 41265 | details::is_string_function_node (branch[1]) || 41266 | details::is_string_condition_node (branch[1]) || 41267 | details::is_string_ccondition_node (branch[1]) || 41268 | details::is_string_vararg_node (branch[1]) ; 41269 | 41270 | if (details::e_add == opr) 41271 | { 41272 | if (!b0_is_cs || !b1_is_cs) 41273 | { 41274 | return synthesize_expression<string_concat_node_t,2>(opr,branch); 41275 | } 41276 | } 41277 | 41278 | if (b0_is_gen || b1_is_gen) 41279 | { 41280 | return synthesize_strogen_expression(opr,branch); 41281 | } 41282 | else if (b0_is_s) 41283 | { 41284 | if (b1_is_s ) return synthesize_sos_expression (opr,branch); 41285 | else if (b1_is_cs ) return synthesize_socs_expression (opr,branch); 41286 | else if (b1_is_sr ) return synthesize_sosr_expression (opr,branch); 41287 | else if (b1_is_csr) return synthesize_socsr_expression (opr,branch); 41288 | } 41289 | else if (b0_is_cs) 41290 | { 41291 | if (b1_is_s ) return synthesize_csos_expression (opr,branch); 41292 | else if (b1_is_cs ) return synthesize_csocs_expression (opr,branch); 41293 | else if (b1_is_sr ) return synthesize_csosr_expression (opr,branch); 41294 | else if (b1_is_csr) return synthesize_csocsr_expression(opr,branch); 41295 | } 41296 | else if (b0_is_sr) 41297 | { 41298 | if (b1_is_s ) return synthesize_sros_expression (opr,branch); 41299 | else if (b1_is_sr ) return synthesize_srosr_expression (opr,branch); 41300 | else if (b1_is_cs ) return synthesize_srocs_expression (opr,branch); 41301 | else if (b1_is_csr) return synthesize_srocsr_expression(opr,branch); 41302 | } 41303 | else if (b0_is_csr) 41304 | { 41305 | if (b1_is_s ) return synthesize_csros_expression (opr,branch); 41306 | else if (b1_is_sr ) return synthesize_csrosr_expression (opr,branch); 41307 | else if (b1_is_cs ) return synthesize_csrocs_expression (opr,branch); 41308 | else if (b1_is_csr) return synthesize_csrocsr_expression(opr,branch); 41309 | } 41310 | 41311 | return error_node(); 41312 | } 41313 | #else 41314 | inline expression_node_ptr synthesize_string_expression(const details::operator_type&, expression_node_ptr (&branch)[2]) 41315 | { 41316 | details::free_all_nodes(*node_allocator_,branch); 41317 | return error_node(); 41318 | } 41319 | #endif 41320 | 41321 | #ifndef exprtk_disable_string_capabilities 41322 | inline expression_node_ptr synthesize_string_expression(const details::operator_type& opr, expression_node_ptr (&branch)[3]) 41323 | { 41324 | if (details::e_inrange != opr) 41325 | return error_node(); 41326 | else if ((0 == branch[0]) || (0 == branch[1]) || (0 == branch[2])) 41327 | { 41328 | details::free_all_nodes(*node_allocator_,branch); 41329 | 41330 | return error_node(); 41331 | } 41332 | else if ( 41333 | details::is_const_string_node(branch[0]) && 41334 | details::is_const_string_node(branch[1]) && 41335 | details::is_const_string_node(branch[2]) 41336 | ) 41337 | { 41338 | const std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str(); 41339 | const std::string s1 = static_cast<details::string_literal_node<Type>*>(branch[1])->str(); 41340 | const std::string s2 = static_cast<details::string_literal_node<Type>*>(branch[2])->str(); 41341 | 41342 | const Type v = (((s0 <= s1) && (s1 <= s2)) ? Type(1) : Type(0)); 41343 | 41344 | details::free_all_nodes(*node_allocator_,branch); 41345 | 41346 | return node_allocator_->allocate_c<details::literal_node<Type> >(v); 41347 | } 41348 | else if ( 41349 | details::is_string_node(branch[0]) && 41350 | details::is_string_node(branch[1]) && 41351 | details::is_string_node(branch[2]) 41352 | ) 41353 | { 41354 | std::string& s0 = static_cast<details::stringvar_node<Type>*>(branch[0])->ref(); 41355 | std::string& s1 = static_cast<details::stringvar_node<Type>*>(branch[1])->ref(); 41356 | std::string& s2 = static_cast<details::stringvar_node<Type>*>(branch[2])->ref(); 41357 | 41358 | typedef typename details::sosos_node<Type, std::string&, std::string&, std::string&, details::inrange_op<Type> > inrange_t; 41359 | 41360 | return node_allocator_->allocate_type<inrange_t, std::string&, std::string&, std::string&>(s0, s1, s2); 41361 | } 41362 | else if ( 41363 | details::is_const_string_node(branch[0]) && 41364 | details::is_string_node(branch[1]) && 41365 | details::is_const_string_node(branch[2]) 41366 | ) 41367 | { 41368 | std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str(); 41369 | std::string& s1 = static_cast<details::stringvar_node<Type>* >(branch[1])->ref(); 41370 | std::string s2 = static_cast<details::string_literal_node<Type>*>(branch[2])->str(); 41371 | 41372 | typedef typename details::sosos_node<Type, std::string, std::string&, std::string, details::inrange_op<Type> > inrange_t; 41373 | 41374 | details::free_node(*node_allocator_,branch[0]); 41375 | details::free_node(*node_allocator_,branch[2]); 41376 | 41377 | return node_allocator_->allocate_type<inrange_t, std::string, std::string&, std::string>(s0, s1, s2); 41378 | } 41379 | else if ( 41380 | details::is_string_node(branch[0]) && 41381 | details::is_const_string_node(branch[1]) && 41382 | details::is_string_node(branch[2]) 41383 | ) 41384 | { 41385 | std::string& s0 = static_cast<details::stringvar_node<Type>* >(branch[0])->ref(); 41386 | std::string s1 = static_cast<details::string_literal_node<Type>*>(branch[1])->str(); 41387 | std::string& s2 = static_cast<details::stringvar_node<Type>* >(branch[2])->ref(); 41388 | 41389 | typedef typename details::sosos_node<Type, std::string&, std::string, std::string&, details::inrange_op<Type> > inrange_t; 41390 | 41391 | details::free_node(*node_allocator_,branch[1]); 41392 | 41393 | return node_allocator_->allocate_type<inrange_t, std::string&, std::string, std::string&>(s0, s1, s2); 41394 | } 41395 | else if ( 41396 | details::is_string_node(branch[0]) && 41397 | details::is_string_node(branch[1]) && 41398 | details::is_const_string_node(branch[2]) 41399 | ) 41400 | { 41401 | std::string& s0 = static_cast<details::stringvar_node<Type>* >(branch[0])->ref(); 41402 | std::string& s1 = static_cast<details::stringvar_node<Type>* >(branch[1])->ref(); 41403 | std::string s2 = static_cast<details::string_literal_node<Type>*>(branch[2])->str(); 41404 | 41405 | typedef typename details::sosos_node<Type, std::string&, std::string&, std::string, details::inrange_op<Type> > inrange_t; 41406 | 41407 | details::free_node(*node_allocator_,branch[2]); 41408 | 41409 | return node_allocator_->allocate_type<inrange_t, std::string&, std::string&, std::string>(s0, s1, s2); 41410 | } 41411 | else if ( 41412 | details::is_const_string_node(branch[0]) && 41413 | details:: is_string_node(branch[1]) && 41414 | details:: is_string_node(branch[2]) 41415 | ) 41416 | { 41417 | std::string s0 = static_cast<details::string_literal_node<Type>*>(branch[0])->str(); 41418 | std::string& s1 = static_cast<details::stringvar_node<Type>* >(branch[1])->ref(); 41419 | std::string& s2 = static_cast<details::stringvar_node<Type>* >(branch[2])->ref(); 41420 | 41421 | typedef typename details::sosos_node<Type, std::string, std::string&, std::string&, details::inrange_op<Type> > inrange_t; 41422 | 41423 | details::free_node(*node_allocator_,branch[0]); 41424 | 41425 | return node_allocator_->allocate_type<inrange_t, std::string, std::string&, std::string&>(s0, s1, s2); 41426 | } 41427 | else 41428 | return error_node(); 41429 | } 41430 | #else 41431 | inline expression_node_ptr synthesize_string_expression(const details::operator_type&, expression_node_ptr (&branch)[3]) 41432 | { 41433 | details::free_all_nodes(*node_allocator_,branch); 41434 | return error_node(); 41435 | } 41436 | #endif 41437 | 41438 | inline expression_node_ptr synthesize_null_expression(const details::operator_type& operation, expression_node_ptr (&branch)[2]) 41439 | { 41440 | /* 41441 | Note: The following are the type promotion rules 41442 | that relate to operations that include 'null': 41443 | 0. null ==/!= null --> true false 41444 | 1. null operation null --> null 41445 | 2. x ==/!= null --> true/false 41446 | 3. null ==/!= x --> true/false 41447 | 4. x operation null --> x 41448 | 5. null operation x --> x 41449 | */ 41450 | 41451 | typedef typename details::null_eq_node<T> nulleq_node_t; 41452 | 41453 | const bool b0_null = details::is_null_node(branch[0]); 41454 | const bool b1_null = details::is_null_node(branch[1]); 41455 | 41456 | if (b0_null && b1_null) 41457 | { 41458 | expression_node_ptr result = error_node(); 41459 | 41460 | if (details::e_eq == operation) 41461 | result = node_allocator_->allocate_c<literal_node_t>(T(1)); 41462 | else if (details::e_ne == operation) 41463 | result = node_allocator_->allocate_c<literal_node_t>(T(0)); 41464 | 41465 | if (result) 41466 | { 41467 | details::free_node(*node_allocator_,branch[0]); 41468 | details::free_node(*node_allocator_,branch[1]); 41469 | 41470 | return result; 41471 | } 41472 | 41473 | details::free_node(*node_allocator_,branch[1]); 41474 | 41475 | return branch[0]; 41476 | } 41477 | else if (details::e_eq == operation) 41478 | { 41479 | expression_node_ptr result = node_allocator_-> 41480 | allocate_rc<nulleq_node_t>(branch[b0_null ? 0 : 1],true); 41481 | 41482 | details::free_node(*node_allocator_,branch[b0_null ? 1 : 0]); 41483 | 41484 | return result; 41485 | } 41486 | else if (details::e_ne == operation) 41487 | { 41488 | expression_node_ptr result = node_allocator_-> 41489 | allocate_rc<nulleq_node_t>(branch[b0_null ? 0 : 1],false); 41490 | 41491 | details::free_node(*node_allocator_,branch[b0_null ? 1 : 0]); 41492 | 41493 | return result; 41494 | } 41495 | else if (b0_null) 41496 | { 41497 | details::free_node(*node_allocator_,branch[0]); 41498 | branch[0] = branch[1]; 41499 | branch[1] = error_node(); 41500 | } 41501 | else if (b1_null) 41502 | { 41503 | details::free_node(*node_allocator_,branch[1]); 41504 | branch[1] = error_node(); 41505 | } 41506 | 41507 | if ( 41508 | (details::e_add == operation) || (details::e_sub == operation) || 41509 | (details::e_mul == operation) || (details::e_div == operation) || 41510 | (details::e_mod == operation) || (details::e_pow == operation) 41511 | ) 41512 | { 41513 | return branch[0]; 41514 | } 41515 | 41516 | details::free_node(*node_allocator_, branch[0]); 41517 | 41518 | if ( 41519 | (details::e_lt == operation) || (details::e_lte == operation) || 41520 | (details::e_gt == operation) || (details::e_gte == operation) || 41521 | (details::e_and == operation) || (details::e_nand == operation) || 41522 | (details::e_or == operation) || (details::e_nor == operation) || 41523 | (details::e_xor == operation) || (details::e_xnor == operation) || 41524 | (details::e_in == operation) || (details::e_like == operation) || 41525 | (details::e_ilike == operation) 41526 | ) 41527 | { 41528 | return node_allocator_->allocate_c<literal_node_t>(T(0)); 41529 | } 41530 | 41531 | return node_allocator_->allocate<details::null_node<Type> >(); 41532 | } 41533 | 41534 | template <typename NodeType, std::size_t N> 41535 | inline expression_node_ptr synthesize_expression(const details::operator_type& operation, expression_node_ptr (&branch)[N]) 41536 | { 41537 | if ( 41538 | (details::e_in == operation) || 41539 | (details::e_like == operation) || 41540 | (details::e_ilike == operation) 41541 | ) 41542 | { 41543 | free_all_nodes(*node_allocator_,branch); 41544 | 41545 | return error_node(); 41546 | } 41547 | else if (!details::all_nodes_valid<N>(branch)) 41548 | { 41549 | free_all_nodes(*node_allocator_,branch); 41550 | 41551 | return error_node(); 41552 | } 41553 | else if ((details::e_default != operation)) 41554 | { 41555 | // Attempt simple constant folding optimisation. 41556 | expression_node_ptr expression_point = node_allocator_->allocate<NodeType>(operation,branch); 41557 | 41558 | if (is_constant_foldable<N>(branch)) 41559 | { 41560 | const Type v = expression_point->value(); 41561 | details::free_node(*node_allocator_,expression_point); 41562 | 41563 | return node_allocator_->allocate<literal_node_t>(v); 41564 | } 41565 | 41566 | if (expression_point && expression_point->valid()) 41567 | { 41568 | return expression_point; 41569 | } 41570 | 41571 | parser_->set_error(parser_error::make_error( 41572 | parser_error::e_parser, 41573 | token_t(), 41574 | "ERR281 - Failed to synthesize node: NodeType", 41575 | exprtk_error_location)); 41576 | 41577 | details::free_node(*node_allocator_, expression_point); 41578 | } 41579 | 41580 | return error_node(); 41581 | } 41582 | 41583 | template <typename NodeType, std::size_t N> 41584 | inline expression_node_ptr synthesize_expression(F* f, expression_node_ptr (&branch)[N]) 41585 | { 41586 | if (!details::all_nodes_valid<N>(branch)) 41587 | { 41588 | free_all_nodes(*node_allocator_,branch); 41589 | 41590 | return error_node(); 41591 | } 41592 | 41593 | typedef typename details::function_N_node<T,ifunction_t,N> function_N_node_t; 41594 | 41595 | // Attempt simple constant folding optimisation. 41596 | 41597 | expression_node_ptr expression_point = node_allocator_->allocate<NodeType>(f); 41598 | function_N_node_t* func_node_ptr = dynamic_cast<function_N_node_t*>(expression_point); 41599 | 41600 | if (0 == func_node_ptr) 41601 | { 41602 | free_all_nodes(*node_allocator_,branch); 41603 | 41604 | return error_node(); 41605 | } 41606 | else 41607 | func_node_ptr->init_branches(branch); 41608 | 41609 | if (is_constant_foldable<N>(branch) && !f->has_side_effects()) 41610 | { 41611 | Type v = expression_point->value(); 41612 | details::free_node(*node_allocator_,expression_point); 41613 | 41614 | return node_allocator_->allocate<literal_node_t>(v); 41615 | } 41616 | 41617 | parser_->state_.activate_side_effect("synthesize_expression(function<NT,N>)"); 41618 | 41619 | return expression_point; 41620 | } 41621 | 41622 | bool strength_reduction_enabled_; 41623 | details::node_allocator* node_allocator_; 41624 | synthesize_map_t synthesize_map_; 41625 | unary_op_map_t* unary_op_map_; 41626 | binary_op_map_t* binary_op_map_; 41627 | inv_binary_op_map_t* inv_binary_op_map_; 41628 | sf3_map_t* sf3_map_; 41629 | sf4_map_t* sf4_map_; 41630 | parser_t* parser_; 41631 | }; // class expression_generator 41632 | 41633 | inline void set_error(const parser_error::type& error_type) 41634 | { 41635 | error_list_.push_back(error_type); 41636 | } 41637 | 41638 | inline void remove_last_error() 41639 | { 41640 | if (!error_list_.empty()) 41641 | { 41642 | error_list_.pop_back(); 41643 | } 41644 | } 41645 | 41646 | inline void set_synthesis_error(const std::string& synthesis_error_message) 41647 | { 41648 | if (synthesis_error_.empty()) 41649 | { 41650 | synthesis_error_ = synthesis_error_message; 41651 | } 41652 | } 41653 | 41654 | inline void register_local_vars(expression<T>& e) 41655 | { 41656 | for (std::size_t i = 0; i < sem_.size(); ++i) 41657 | { 41658 | scope_element& se = sem_.get_element(i); 41659 | 41660 | exprtk_debug(("register_local_vars() - se[%s]\n", se.name.c_str())); 41661 | 41662 | if ( 41663 | (scope_element::e_variable == se.type) || 41664 | (scope_element::e_literal == se.type) || 41665 | (scope_element::e_vecelem == se.type) 41666 | ) 41667 | { 41668 | if (se.var_node) 41669 | { 41670 | e.register_local_var(se.var_node); 41671 | } 41672 | 41673 | if (se.data) 41674 | { 41675 | e.register_local_data(se.data, 1, 0); 41676 | } 41677 | } 41678 | else if (scope_element::e_vector == se.type) 41679 | { 41680 | if (se.vec_node) 41681 | { 41682 | e.register_local_var(se.vec_node); 41683 | } 41684 | 41685 | if (se.data) 41686 | { 41687 | e.register_local_data(se.data, se.size, 1); 41688 | } 41689 | } 41690 | #ifndef exprtk_disable_string_capabilities 41691 | else if (scope_element::e_string == se.type) 41692 | { 41693 | if (se.str_node) 41694 | { 41695 | e.register_local_var(se.str_node); 41696 | } 41697 | 41698 | if (se.data) 41699 | { 41700 | e.register_local_data(se.data, se.size, 2); 41701 | } 41702 | } 41703 | #endif 41704 | 41705 | se.var_node = 0; 41706 | se.vec_node = 0; 41707 | #ifndef exprtk_disable_string_capabilities 41708 | se.str_node = 0; 41709 | #endif 41710 | se.data = 0; 41711 | se.ref_count = 0; 41712 | se.active = false; 41713 | } 41714 | } 41715 | 41716 | inline void register_return_results(expression<T>& e) 41717 | { 41718 | e.register_return_results(results_context_); 41719 | results_context_ = 0; 41720 | } 41721 | 41722 | inline void load_unary_operations_map(unary_op_map_t& m) 41723 | { 41724 | #define register_unary_op(Op, UnaryFunctor) \ 41725 | m.insert(std::make_pair(Op,UnaryFunctor<T>::process)); \ 41726 | 41727 | register_unary_op(details::e_abs , details::abs_op ) 41728 | register_unary_op(details::e_acos , details::acos_op ) 41729 | register_unary_op(details::e_acosh , details::acosh_op) 41730 | register_unary_op(details::e_asin , details::asin_op ) 41731 | register_unary_op(details::e_asinh , details::asinh_op) 41732 | register_unary_op(details::e_atanh , details::atanh_op) 41733 | register_unary_op(details::e_ceil , details::ceil_op ) 41734 | register_unary_op(details::e_cos , details::cos_op ) 41735 | register_unary_op(details::e_cosh , details::cosh_op ) 41736 | register_unary_op(details::e_exp , details::exp_op ) 41737 | register_unary_op(details::e_expm1 , details::expm1_op) 41738 | register_unary_op(details::e_floor , details::floor_op) 41739 | register_unary_op(details::e_log , details::log_op ) 41740 | register_unary_op(details::e_log10 , details::log10_op) 41741 | register_unary_op(details::e_log2 , details::log2_op ) 41742 | register_unary_op(details::e_log1p , details::log1p_op) 41743 | register_unary_op(details::e_neg , details::neg_op ) 41744 | register_unary_op(details::e_pos , details::pos_op ) 41745 | register_unary_op(details::e_round , details::round_op) 41746 | register_unary_op(details::e_sin , details::sin_op ) 41747 | register_unary_op(details::e_sinc , details::sinc_op ) 41748 | register_unary_op(details::e_sinh , details::sinh_op ) 41749 | register_unary_op(details::e_sqrt , details::sqrt_op ) 41750 | register_unary_op(details::e_tan , details::tan_op ) 41751 | register_unary_op(details::e_tanh , details::tanh_op ) 41752 | register_unary_op(details::e_cot , details::cot_op ) 41753 | register_unary_op(details::e_sec , details::sec_op ) 41754 | register_unary_op(details::e_csc , details::csc_op ) 41755 | register_unary_op(details::e_r2d , details::r2d_op ) 41756 | register_unary_op(details::e_d2r , details::d2r_op ) 41757 | register_unary_op(details::e_d2g , details::d2g_op ) 41758 | register_unary_op(details::e_g2d , details::g2d_op ) 41759 | register_unary_op(details::e_notl , details::notl_op ) 41760 | register_unary_op(details::e_sgn , details::sgn_op ) 41761 | register_unary_op(details::e_erf , details::erf_op ) 41762 | register_unary_op(details::e_erfc , details::erfc_op ) 41763 | register_unary_op(details::e_ncdf , details::ncdf_op ) 41764 | register_unary_op(details::e_frac , details::frac_op ) 41765 | register_unary_op(details::e_trunc , details::trunc_op) 41766 | #undef register_unary_op 41767 | } 41768 | 41769 | inline void load_binary_operations_map(binary_op_map_t& m) 41770 | { 41771 | typedef typename binary_op_map_t::value_type value_type; 41772 | 41773 | #define register_binary_op(Op, BinaryFunctor) \ 41774 | m.insert(value_type(Op,BinaryFunctor<T>::process)); \ 41775 | 41776 | register_binary_op(details::e_add , details::add_op ) 41777 | register_binary_op(details::e_sub , details::sub_op ) 41778 | register_binary_op(details::e_mul , details::mul_op ) 41779 | register_binary_op(details::e_div , details::div_op ) 41780 | register_binary_op(details::e_mod , details::mod_op ) 41781 | register_binary_op(details::e_pow , details::pow_op ) 41782 | register_binary_op(details::e_lt , details::lt_op ) 41783 | register_binary_op(details::e_lte , details::lte_op ) 41784 | register_binary_op(details::e_gt , details::gt_op ) 41785 | register_binary_op(details::e_gte , details::gte_op ) 41786 | register_binary_op(details::e_eq , details::eq_op ) 41787 | register_binary_op(details::e_ne , details::ne_op ) 41788 | register_binary_op(details::e_and , details::and_op ) 41789 | register_binary_op(details::e_nand , details::nand_op) 41790 | register_binary_op(details::e_or , details::or_op ) 41791 | register_binary_op(details::e_nor , details::nor_op ) 41792 | register_binary_op(details::e_xor , details::xor_op ) 41793 | register_binary_op(details::e_xnor , details::xnor_op) 41794 | #undef register_binary_op 41795 | } 41796 | 41797 | inline void load_inv_binary_operations_map(inv_binary_op_map_t& m) 41798 | { 41799 | typedef typename inv_binary_op_map_t::value_type value_type; 41800 | 41801 | #define register_binary_op(Op, BinaryFunctor) \ 41802 | m.insert(value_type(BinaryFunctor<T>::process,Op)); \ 41803 | 41804 | register_binary_op(details::e_add , details::add_op ) 41805 | register_binary_op(details::e_sub , details::sub_op ) 41806 | register_binary_op(details::e_mul , details::mul_op ) 41807 | register_binary_op(details::e_div , details::div_op ) 41808 | register_binary_op(details::e_mod , details::mod_op ) 41809 | register_binary_op(details::e_pow , details::pow_op ) 41810 | register_binary_op(details::e_lt , details::lt_op ) 41811 | register_binary_op(details::e_lte , details::lte_op ) 41812 | register_binary_op(details::e_gt , details::gt_op ) 41813 | register_binary_op(details::e_gte , details::gte_op ) 41814 | register_binary_op(details::e_eq , details::eq_op ) 41815 | register_binary_op(details::e_ne , details::ne_op ) 41816 | register_binary_op(details::e_and , details::and_op ) 41817 | register_binary_op(details::e_nand , details::nand_op) 41818 | register_binary_op(details::e_or , details::or_op ) 41819 | register_binary_op(details::e_nor , details::nor_op ) 41820 | register_binary_op(details::e_xor , details::xor_op ) 41821 | register_binary_op(details::e_xnor , details::xnor_op) 41822 | #undef register_binary_op 41823 | } 41824 | 41825 | inline void load_sf3_map(sf3_map_t& sf3_map) 41826 | { 41827 | typedef std::pair<trinary_functor_t,details::operator_type> pair_t; 41828 | 41829 | #define register_sf3(Op) \ 41830 | sf3_map[details::sf##Op##_op<T>::id()] = pair_t(details::sf##Op##_op<T>::process,details::e_sf##Op); \ 41831 | 41832 | register_sf3(00) register_sf3(01) register_sf3(02) register_sf3(03) 41833 | register_sf3(04) register_sf3(05) register_sf3(06) register_sf3(07) 41834 | register_sf3(08) register_sf3(09) register_sf3(10) register_sf3(11) 41835 | register_sf3(12) register_sf3(13) register_sf3(14) register_sf3(15) 41836 | register_sf3(16) register_sf3(17) register_sf3(18) register_sf3(19) 41837 | register_sf3(20) register_sf3(21) register_sf3(22) register_sf3(23) 41838 | register_sf3(24) register_sf3(25) register_sf3(26) register_sf3(27) 41839 | register_sf3(28) register_sf3(29) register_sf3(30) 41840 | #undef register_sf3 41841 | 41842 | #define register_sf3_extid(Id, Op) \ 41843 | sf3_map[Id] = pair_t(details::sf##Op##_op<T>::process,details::e_sf##Op); \ 41844 | 41845 | register_sf3_extid("(t-t)-t",23) // (t-t)-t --> t-(t+t) 41846 | #undef register_sf3_extid 41847 | } 41848 | 41849 | inline void load_sf4_map(sf4_map_t& sf4_map) 41850 | { 41851 | typedef std::pair<quaternary_functor_t,details::operator_type> pair_t; 41852 | 41853 | #define register_sf4(Op) \ 41854 | sf4_map[details::sf##Op##_op<T>::id()] = pair_t(details::sf##Op##_op<T>::process,details::e_sf##Op); \ 41855 | 41856 | register_sf4(48) register_sf4(49) register_sf4(50) register_sf4(51) 41857 | register_sf4(52) register_sf4(53) register_sf4(54) register_sf4(55) 41858 | register_sf4(56) register_sf4(57) register_sf4(58) register_sf4(59) 41859 | register_sf4(60) register_sf4(61) register_sf4(62) register_sf4(63) 41860 | register_sf4(64) register_sf4(65) register_sf4(66) register_sf4(67) 41861 | register_sf4(68) register_sf4(69) register_sf4(70) register_sf4(71) 41862 | register_sf4(72) register_sf4(73) register_sf4(74) register_sf4(75) 41863 | register_sf4(76) register_sf4(77) register_sf4(78) register_sf4(79) 41864 | register_sf4(80) register_sf4(81) register_sf4(82) register_sf4(83) 41865 | #undef register_sf4 41866 | 41867 | #define register_sf4ext(Op) \ 41868 | sf4_map[details::sfext##Op##_op<T>::id()] = pair_t(details::sfext##Op##_op<T>::process,details::e_sf4ext##Op); \ 41869 | 41870 | register_sf4ext(00) register_sf4ext(01) register_sf4ext(02) register_sf4ext(03) 41871 | register_sf4ext(04) register_sf4ext(05) register_sf4ext(06) register_sf4ext(07) 41872 | register_sf4ext(08) register_sf4ext(09) register_sf4ext(10) register_sf4ext(11) 41873 | register_sf4ext(12) register_sf4ext(13) register_sf4ext(14) register_sf4ext(15) 41874 | register_sf4ext(16) register_sf4ext(17) register_sf4ext(18) register_sf4ext(19) 41875 | register_sf4ext(20) register_sf4ext(21) register_sf4ext(22) register_sf4ext(23) 41876 | register_sf4ext(24) register_sf4ext(25) register_sf4ext(26) register_sf4ext(27) 41877 | register_sf4ext(28) register_sf4ext(29) register_sf4ext(30) register_sf4ext(31) 41878 | register_sf4ext(32) register_sf4ext(33) register_sf4ext(34) register_sf4ext(35) 41879 | register_sf4ext(36) register_sf4ext(36) register_sf4ext(38) register_sf4ext(39) 41880 | register_sf4ext(40) register_sf4ext(41) register_sf4ext(42) register_sf4ext(43) 41881 | register_sf4ext(44) register_sf4ext(45) register_sf4ext(46) register_sf4ext(47) 41882 | register_sf4ext(48) register_sf4ext(49) register_sf4ext(50) register_sf4ext(51) 41883 | register_sf4ext(52) register_sf4ext(53) register_sf4ext(54) register_sf4ext(55) 41884 | register_sf4ext(56) register_sf4ext(57) register_sf4ext(58) register_sf4ext(59) 41885 | register_sf4ext(60) register_sf4ext(61) 41886 | #undef register_sf4ext 41887 | } 41888 | 41889 | inline results_context_t& results_ctx() 41890 | { 41891 | if (0 == results_context_) 41892 | { 41893 | results_context_ = new results_context_t(); 41894 | } 41895 | 41896 | return (*results_context_); 41897 | } 41898 | 41899 | inline void return_cleanup() 41900 | { 41901 | #ifndef exprtk_disable_return_statement 41902 | if (results_context_) 41903 | { 41904 | delete results_context_; 41905 | results_context_ = 0; 41906 | } 41907 | 41908 | state_.return_stmt_present = false; 41909 | #endif 41910 | } 41911 | 41912 | inline bool valid_settings() 41913 | { 41914 | const std::size_t max_local_vector_size_bytes = sizeof(T) * settings_.max_local_vector_size(); 41915 | 41916 | if (max_local_vector_size_bytes > settings_.max_total_local_symbol_size_bytes()) 41917 | { 41918 | set_error(make_error( 41919 | parser_error::e_parser, 41920 | "ERR282 - Max local vector size of " + details::to_str(max_local_vector_size_bytes) + " bytes " 41921 | "is larger than max total local symbol size of " + details::to_str(settings_.max_total_local_symbol_size_bytes()) + " bytes", 41922 | exprtk_error_location)); 41923 | 41924 | return false; 41925 | } 41926 | 41927 | return true; 41928 | } 41929 | 41930 | private: 41931 | 41932 | parser(const parser<T>&) exprtk_delete; 41933 | parser<T>& operator=(const parser<T>&) exprtk_delete; 41934 | 41935 | settings_store settings_; 41936 | expression_generator<T> expression_generator_; 41937 | details::node_allocator node_allocator_; 41938 | symtab_store symtab_store_; 41939 | dependent_entity_collector dec_; 41940 | std::deque<parser_error::type> error_list_; 41941 | std::deque<bool> brkcnt_list_; 41942 | parser_state state_; 41943 | bool resolve_unknown_symbol_; 41944 | results_context_t* results_context_; 41945 | unknown_symbol_resolver* unknown_symbol_resolver_; 41946 | unknown_symbol_resolver default_usr_; 41947 | base_ops_map_t base_ops_map_; 41948 | unary_op_map_t unary_op_map_; 41949 | binary_op_map_t binary_op_map_; 41950 | inv_binary_op_map_t inv_binary_op_map_; 41951 | sf3_map_t sf3_map_; 41952 | sf4_map_t sf4_map_; 41953 | std::string synthesis_error_; 41954 | scope_element_manager sem_; 41955 | std::vector<state_t> current_state_stack_; 41956 | 41957 | immutable_memory_map_t immutable_memory_map_; 41958 | immutable_symtok_map_t immutable_symtok_map_; 41959 | 41960 | lexer::helper::helper_assembly helper_assembly_; 41961 | 41962 | lexer::helper::commutative_inserter commutative_inserter_; 41963 | lexer::helper::operator_joiner operator_joiner_2_; 41964 | lexer::helper::operator_joiner operator_joiner_3_; 41965 | lexer::helper::symbol_replacer symbol_replacer_; 41966 | lexer::helper::bracket_checker bracket_checker_; 41967 | lexer::helper::numeric_checker<T> numeric_checker_; 41968 | lexer::helper::sequence_validator sequence_validator_; 41969 | lexer::helper::sequence_validator_3tokens sequence_validator_3tkns_; 41970 | 41971 | loop_runtime_check_ptr loop_runtime_check_; 41972 | vector_access_runtime_check_ptr vector_access_runtime_check_; 41973 | compilation_check_ptr compilation_check_ptr_; 41974 | assert_check_ptr assert_check_; 41975 | std::set<std::string> assert_ids_; 41976 | 41977 | template <typename ParserType> 41978 | friend void details::disable_type_checking(ParserType& p); 41979 | }; // class parser 41980 | 41981 | namespace details 41982 | { 41983 | template <typename T> 41984 | struct collector_helper 41985 | { 41986 | typedef exprtk::symbol_table<T> symbol_table_t; 41987 | typedef exprtk::expression<T> expression_t; 41988 | typedef exprtk::parser<T> parser_t; 41989 | typedef typename parser_t::dependent_entity_collector::symbol_t symbol_t; 41990 | typedef typename parser_t::unknown_symbol_resolver usr_t; 41991 | 41992 | struct resolve_as_vector : public usr_t 41993 | { 41994 | typedef exprtk::parser<T> parser_t; 41995 | 41996 | using usr_t::process; 41997 | 41998 | resolve_as_vector() 41999 | : usr_t(usr_t::e_usrmode_extended) 42000 | {} 42001 | 42002 | virtual bool process(const std::string& unknown_symbol, 42003 | symbol_table_t& symbol_table, 42004 | std::string&) exprtk_override 42005 | { 42006 | static T v[1]; 42007 | symbol_table.add_vector(unknown_symbol,v); 42008 | return true; 42009 | } 42010 | }; 42011 | 42012 | static inline bool collection_pass(const std::string& expression_string, 42013 | std::set<std::string>& symbol_set, 42014 | const bool collect_variables, 42015 | const bool collect_functions, 42016 | const bool vector_pass, 42017 | symbol_table_t& ext_symbol_table) 42018 | { 42019 | symbol_table_t symbol_table; 42020 | expression_t expression; 42021 | parser_t parser; 42022 | 42023 | resolve_as_vector vect_resolver; 42024 | 42025 | expression.register_symbol_table(symbol_table ); 42026 | expression.register_symbol_table(ext_symbol_table); 42027 | 42028 | if (vector_pass) 42029 | parser.enable_unknown_symbol_resolver(&vect_resolver); 42030 | else 42031 | parser.enable_unknown_symbol_resolver(); 42032 | 42033 | if (collect_variables) 42034 | parser.dec().collect_variables() = true; 42035 | 42036 | if (collect_functions) 42037 | parser.dec().collect_functions() = true; 42038 | 42039 | bool pass_result = false; 42040 | 42041 | details::disable_type_checking(parser); 42042 | 42043 | if (parser.compile(expression_string, expression)) 42044 | { 42045 | pass_result = true; 42046 | 42047 | std::deque<symbol_t> symb_list; 42048 | parser.dec().symbols(symb_list); 42049 | 42050 | for (std::size_t i = 0; i < symb_list.size(); ++i) 42051 | { 42052 | symbol_set.insert(symb_list[i].first); 42053 | } 42054 | } 42055 | 42056 | return pass_result; 42057 | } 42058 | }; 42059 | } 42060 | 42061 | template <typename Allocator, 42062 | template <typename, typename> class Sequence> 42063 | inline bool collect_variables(const std::string& expression, 42064 | Sequence<std::string, Allocator>& symbol_list) 42065 | { 42066 | typedef double T; 42067 | typedef details::collector_helper<T> collect_t; 42068 | 42069 | collect_t::symbol_table_t null_symbol_table; 42070 | 42071 | std::set<std::string> symbol_set; 42072 | 42073 | const bool variable_pass = collect_t::collection_pass 42074 | (expression, symbol_set, true, false, false, null_symbol_table); 42075 | const bool vector_pass = collect_t::collection_pass 42076 | (expression, symbol_set, true, false, true, null_symbol_table); 42077 | 42078 | if (!variable_pass && !vector_pass) 42079 | return false; 42080 | 42081 | std::set<std::string>::iterator itr = symbol_set.begin(); 42082 | 42083 | while (symbol_set.end() != itr) 42084 | { 42085 | symbol_list.push_back(*itr); 42086 | ++itr; 42087 | } 42088 | 42089 | return true; 42090 | } 42091 | 42092 | template <typename T, 42093 | typename Allocator, 42094 | template <typename, typename> class Sequence> 42095 | inline bool collect_variables(const std::string& expression, 42096 | exprtk::symbol_table<T>& extrnl_symbol_table, 42097 | Sequence<std::string, Allocator>& symbol_list) 42098 | { 42099 | typedef details::collector_helper<T> collect_t; 42100 | 42101 | std::set<std::string> symbol_set; 42102 | 42103 | const bool variable_pass = collect_t::collection_pass 42104 | (expression, symbol_set, true, false, false, extrnl_symbol_table); 42105 | const bool vector_pass = collect_t::collection_pass 42106 | (expression, symbol_set, true, false, true, extrnl_symbol_table); 42107 | 42108 | if (!variable_pass && !vector_pass) 42109 | return false; 42110 | 42111 | std::set<std::string>::iterator itr = symbol_set.begin(); 42112 | 42113 | while (symbol_set.end() != itr) 42114 | { 42115 | symbol_list.push_back(*itr); 42116 | ++itr; 42117 | } 42118 | 42119 | return true; 42120 | } 42121 | 42122 | template <typename Allocator, 42123 | template <typename, typename> class Sequence> 42124 | inline bool collect_functions(const std::string& expression, 42125 | Sequence<std::string, Allocator>& symbol_list) 42126 | { 42127 | typedef double T; 42128 | typedef details::collector_helper<T> collect_t; 42129 | 42130 | collect_t::symbol_table_t null_symbol_table; 42131 | 42132 | std::set<std::string> symbol_set; 42133 | 42134 | const bool variable_pass = collect_t::collection_pass 42135 | (expression, symbol_set, false, true, false, null_symbol_table); 42136 | const bool vector_pass = collect_t::collection_pass 42137 | (expression, symbol_set, false, true, true, null_symbol_table); 42138 | 42139 | if (!variable_pass && !vector_pass) 42140 | return false; 42141 | 42142 | std::set<std::string>::iterator itr = symbol_set.begin(); 42143 | 42144 | while (symbol_set.end() != itr) 42145 | { 42146 | symbol_list.push_back(*itr); 42147 | ++itr; 42148 | } 42149 | 42150 | return true; 42151 | } 42152 | 42153 | template <typename T, 42154 | typename Allocator, 42155 | template <typename, typename> class Sequence> 42156 | inline bool collect_functions(const std::string& expression, 42157 | exprtk::symbol_table<T>& extrnl_symbol_table, 42158 | Sequence<std::string, Allocator>& symbol_list) 42159 | { 42160 | typedef details::collector_helper<T> collect_t; 42161 | 42162 | std::set<std::string> symbol_set; 42163 | 42164 | const bool variable_pass = collect_t::collection_pass 42165 | (expression, symbol_set, false, true, false, extrnl_symbol_table); 42166 | const bool vector_pass = collect_t::collection_pass 42167 | (expression, symbol_set, false, true, true, extrnl_symbol_table); 42168 | 42169 | if (!variable_pass && !vector_pass) 42170 | return false; 42171 | 42172 | std::set<std::string>::iterator itr = symbol_set.begin(); 42173 | 42174 | while (symbol_set.end() != itr) 42175 | { 42176 | symbol_list.push_back(*itr); 42177 | ++itr; 42178 | } 42179 | 42180 | return true; 42181 | } 42182 | 42183 | template <typename T> 42184 | inline T integrate(const expression<T>& e, 42185 | T& x, 42186 | const T& r0, const T& r1, 42187 | const std::size_t number_of_intervals = 1000000) 42188 | { 42189 | if (r0 > r1) 42190 | return T(0); 42191 | 42192 | const T h = (r1 - r0) / (T(2) * number_of_intervals); 42193 | T total_area = T(0); 42194 | 42195 | for (std::size_t i = 0; i < number_of_intervals; ++i) 42196 | { 42197 | x = r0 + T(2) * i * h; 42198 | const T y0 = e.value(); x += h; 42199 | const T y1 = e.value(); x += h; 42200 | const T y2 = e.value(); x += h; 42201 | total_area += h * (y0 + T(4) * y1 + y2) / T(3); 42202 | } 42203 | 42204 | return total_area; 42205 | } 42206 | 42207 | template <typename T> 42208 | inline T integrate(const expression<T>& e, 42209 | const std::string& variable_name, 42210 | const T& r0, const T& r1, 42211 | const std::size_t number_of_intervals = 1000000) 42212 | { 42213 | const symbol_table<T>& sym_table = e.get_symbol_table(); 42214 | 42215 | if (!sym_table.valid()) 42216 | { 42217 | return std::numeric_limits<T>::quiet_NaN(); 42218 | } 42219 | 42220 | details::variable_node<T>* var = sym_table.get_variable(variable_name); 42221 | 42222 | if (var) 42223 | { 42224 | T& x = var->ref(); 42225 | const T x_original = x; 42226 | const T result = integrate(e, x, r0, r1, number_of_intervals); 42227 | x = x_original; 42228 | 42229 | return result; 42230 | } 42231 | 42232 | return std::numeric_limits<T>::quiet_NaN(); 42233 | } 42234 | 42235 | template <typename T> 42236 | inline T derivative(const expression<T>& e, 42237 | T& x, 42238 | const T& h = T(0.00000001)) 42239 | { 42240 | const T x_init = x; 42241 | const T _2h = T(2) * h; 42242 | 42243 | x = x_init + _2h; 42244 | const T y0 = e.value(); 42245 | x = x_init + h; 42246 | const T y1 = e.value(); 42247 | x = x_init - h; 42248 | const T y2 = e.value(); 42249 | x = x_init - _2h; 42250 | const T y3 = e.value(); 42251 | x = x_init; 42252 | 42253 | return (-y0 + T(8) * (y1 - y2) + y3) / (T(12) * h); 42254 | } 42255 | 42256 | template <typename T> 42257 | inline T second_derivative(const expression<T>& e, 42258 | T& x, 42259 | const T& h = T(0.00001)) 42260 | { 42261 | const T x_init = x; 42262 | const T _2h = T(2) * h; 42263 | 42264 | const T y = e.value(); 42265 | x = x_init + _2h; 42266 | const T y0 = e.value(); 42267 | x = x_init + h; 42268 | const T y1 = e.value(); 42269 | x = x_init - h; 42270 | const T y2 = e.value(); 42271 | x = x_init - _2h; 42272 | const T y3 = e.value(); 42273 | x = x_init; 42274 | 42275 | return (-y0 + T(16) * (y1 + y2) - T(30) * y - y3) / (T(12) * h * h); 42276 | } 42277 | 42278 | template <typename T> 42279 | inline T third_derivative(const expression<T>& e, 42280 | T& x, 42281 | const T& h = T(0.0001)) 42282 | { 42283 | const T x_init = x; 42284 | const T _2h = T(2) * h; 42285 | 42286 | x = x_init + _2h; 42287 | const T y0 = e.value(); 42288 | x = x_init + h; 42289 | const T y1 = e.value(); 42290 | x = x_init - h; 42291 | const T y2 = e.value(); 42292 | x = x_init - _2h; 42293 | const T y3 = e.value(); 42294 | x = x_init; 42295 | 42296 | return (y0 + T(2) * (y2 - y1) - y3) / (T(2) * h * h * h); 42297 | } 42298 | 42299 | template <typename T> 42300 | inline T derivative(const expression<T>& e, 42301 | const std::string& variable_name, 42302 | const T& h = T(0.00000001)) 42303 | { 42304 | const symbol_table<T>& sym_table = e.get_symbol_table(); 42305 | 42306 | if (!sym_table.valid()) 42307 | { 42308 | return std::numeric_limits<T>::quiet_NaN(); 42309 | } 42310 | 42311 | details::variable_node<T>* var = sym_table.get_variable(variable_name); 42312 | 42313 | if (var) 42314 | { 42315 | T& x = var->ref(); 42316 | const T x_original = x; 42317 | const T result = derivative(e, x, h); 42318 | x = x_original; 42319 | 42320 | return result; 42321 | } 42322 | 42323 | return std::numeric_limits<T>::quiet_NaN(); 42324 | } 42325 | 42326 | template <typename T> 42327 | inline T second_derivative(const expression<T>& e, 42328 | const std::string& variable_name, 42329 | const T& h = T(0.00001)) 42330 | { 42331 | const symbol_table<T>& sym_table = e.get_symbol_table(); 42332 | 42333 | if (!sym_table.valid()) 42334 | { 42335 | return std::numeric_limits<T>::quiet_NaN(); 42336 | } 42337 | 42338 | details::variable_node<T>* var = sym_table.get_variable(variable_name); 42339 | 42340 | if (var) 42341 | { 42342 | T& x = var->ref(); 42343 | const T x_original = x; 42344 | const T result = second_derivative(e, x, h); 42345 | x = x_original; 42346 | 42347 | return result; 42348 | } 42349 | 42350 | return std::numeric_limits<T>::quiet_NaN(); 42351 | } 42352 | 42353 | template <typename T> 42354 | inline T third_derivative(const expression<T>& e, 42355 | const std::string& variable_name, 42356 | const T& h = T(0.0001)) 42357 | { 42358 | const symbol_table<T>& sym_table = e.get_symbol_table(); 42359 | 42360 | if (!sym_table.valid()) 42361 | { 42362 | return std::numeric_limits<T>::quiet_NaN(); 42363 | } 42364 | 42365 | details::variable_node<T>* var = sym_table.get_variable(variable_name); 42366 | 42367 | if (var) 42368 | { 42369 | T& x = var->ref(); 42370 | const T x_original = x; 42371 | const T result = third_derivative(e, x, h); 42372 | x = x_original; 42373 | 42374 | return result; 42375 | } 42376 | 42377 | return std::numeric_limits<T>::quiet_NaN(); 42378 | } 42379 | 42380 | /* 42381 | Note: The following 'compute' routines are simple helpers, 42382 | for quickly setting up the required pieces of code in order 42383 | to evaluate an expression. By virtue of how they operate 42384 | there will be an overhead with regards to their setup and 42385 | teardown and hence should not be used in time critical 42386 | sections of code. 42387 | Furthermore they only assume a small sub set of variables, 42388 | no string variables or user defined functions. 42389 | */ 42390 | template <typename T> 42391 | inline bool compute(const std::string& expression_string, T& result) 42392 | { 42393 | // No variables 42394 | symbol_table<T> symbol_table; 42395 | symbol_table.add_constants(); 42396 | 42397 | expression<T> expression; 42398 | expression.register_symbol_table(symbol_table); 42399 | 42400 | parser<T> parser; 42401 | 42402 | if (parser.compile(expression_string,expression)) 42403 | { 42404 | result = expression.value(); 42405 | 42406 | return true; 42407 | } 42408 | else 42409 | return false; 42410 | } 42411 | 42412 | template <typename T> 42413 | inline bool compute(const std::string& expression_string, 42414 | const T& x, 42415 | T& result) 42416 | { 42417 | // Only 'x' 42418 | static const std::string x_var("x"); 42419 | 42420 | symbol_table<T> symbol_table; 42421 | symbol_table.add_constants(); 42422 | symbol_table.add_constant(x_var,x); 42423 | 42424 | expression<T> expression; 42425 | expression.register_symbol_table(symbol_table); 42426 | 42427 | parser<T> parser; 42428 | 42429 | if (parser.compile(expression_string,expression)) 42430 | { 42431 | result = expression.value(); 42432 | 42433 | return true; 42434 | } 42435 | else 42436 | return false; 42437 | } 42438 | 42439 | template <typename T> 42440 | inline bool compute(const std::string& expression_string, 42441 | const T&x, const T& y, 42442 | T& result) 42443 | { 42444 | // Only 'x' and 'y' 42445 | static const std::string x_var("x"); 42446 | static const std::string y_var("y"); 42447 | 42448 | symbol_table<T> symbol_table; 42449 | symbol_table.add_constants(); 42450 | symbol_table.add_constant(x_var,x); 42451 | symbol_table.add_constant(y_var,y); 42452 | 42453 | expression<T> expression; 42454 | expression.register_symbol_table(symbol_table); 42455 | 42456 | parser<T> parser; 42457 | 42458 | if (parser.compile(expression_string,expression)) 42459 | { 42460 | result = expression.value(); 42461 | 42462 | return true; 42463 | } 42464 | else 42465 | return false; 42466 | } 42467 | 42468 | template <typename T> 42469 | inline bool compute(const std::string& expression_string, 42470 | const T& x, const T& y, const T& z, 42471 | T& result) 42472 | { 42473 | // Only 'x', 'y' or 'z' 42474 | static const std::string x_var("x"); 42475 | static const std::string y_var("y"); 42476 | static const std::string z_var("z"); 42477 | 42478 | symbol_table<T> symbol_table; 42479 | symbol_table.add_constants(); 42480 | symbol_table.add_constant(x_var,x); 42481 | symbol_table.add_constant(y_var,y); 42482 | symbol_table.add_constant(z_var,z); 42483 | 42484 | expression<T> expression; 42485 | expression.register_symbol_table(symbol_table); 42486 | 42487 | parser<T> parser; 42488 | 42489 | if (parser.compile(expression_string,expression)) 42490 | { 42491 | result = expression.value(); 42492 | 42493 | return true; 42494 | } 42495 | else 42496 | return false; 42497 | } 42498 | 42499 | template <typename T, std::size_t N> 42500 | class polynomial : public ifunction<T> 42501 | { 42502 | private: 42503 | 42504 | template <typename Type, std::size_t NumberOfCoefficients> 42505 | struct poly_impl { }; 42506 | 42507 | template <typename Type> 42508 | struct poly_impl <Type,12> 42509 | { 42510 | static inline T evaluate(const Type x, 42511 | const Type c12, const Type c11, const Type c10, const Type c9, const Type c8, 42512 | const Type c7, const Type c6, const Type c5, const Type c4, const Type c3, 42513 | const Type c2, const Type c1, const Type c0) 42514 | { 42515 | // p(x) = c_12x^12 + c_11x^11 + c_10x^10 + c_9x^9 + c_8x^8 + c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42516 | return ((((((((((((c12 * x + c11) * x + c10) * x + c9) * x + c8) * x + c7) * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42517 | } 42518 | }; 42519 | 42520 | template <typename Type> 42521 | struct poly_impl <Type,11> 42522 | { 42523 | static inline T evaluate(const Type x, 42524 | const Type c11, const Type c10, const Type c9, const Type c8, const Type c7, 42525 | const Type c6, const Type c5, const Type c4, const Type c3, const Type c2, 42526 | const Type c1, const Type c0) 42527 | { 42528 | // p(x) = c_11x^11 + c_10x^10 + c_9x^9 + c_8x^8 + c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42529 | return (((((((((((c11 * x + c10) * x + c9) * x + c8) * x + c7) * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42530 | } 42531 | }; 42532 | 42533 | template <typename Type> 42534 | struct poly_impl <Type,10> 42535 | { 42536 | static inline T evaluate(const Type x, 42537 | const Type c10, const Type c9, const Type c8, const Type c7, const Type c6, 42538 | const Type c5, const Type c4, const Type c3, const Type c2, const Type c1, 42539 | const Type c0) 42540 | { 42541 | // p(x) = c_10x^10 + c_9x^9 + c_8x^8 + c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42542 | return ((((((((((c10 * x + c9) * x + c8) * x + c7) * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42543 | } 42544 | }; 42545 | 42546 | template <typename Type> 42547 | struct poly_impl <Type,9> 42548 | { 42549 | static inline T evaluate(const Type x, 42550 | const Type c9, const Type c8, const Type c7, const Type c6, const Type c5, 42551 | const Type c4, const Type c3, const Type c2, const Type c1, const Type c0) 42552 | { 42553 | // p(x) = c_9x^9 + c_8x^8 + c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42554 | return (((((((((c9 * x + c8) * x + c7) * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42555 | } 42556 | }; 42557 | 42558 | template <typename Type> 42559 | struct poly_impl <Type,8> 42560 | { 42561 | static inline T evaluate(const Type x, 42562 | const Type c8, const Type c7, const Type c6, const Type c5, const Type c4, 42563 | const Type c3, const Type c2, const Type c1, const Type c0) 42564 | { 42565 | // p(x) = c_8x^8 + c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42566 | return ((((((((c8 * x + c7) * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42567 | } 42568 | }; 42569 | 42570 | template <typename Type> 42571 | struct poly_impl <Type,7> 42572 | { 42573 | static inline T evaluate(const Type x, 42574 | const Type c7, const Type c6, const Type c5, const Type c4, const Type c3, 42575 | const Type c2, const Type c1, const Type c0) 42576 | { 42577 | // p(x) = c_7x^7 + c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42578 | return (((((((c7 * x + c6) * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42579 | } 42580 | }; 42581 | 42582 | template <typename Type> 42583 | struct poly_impl <Type,6> 42584 | { 42585 | static inline T evaluate(const Type x, 42586 | const Type c6, const Type c5, const Type c4, const Type c3, const Type c2, 42587 | const Type c1, const Type c0) 42588 | { 42589 | // p(x) = c_6x^6 + c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42590 | return ((((((c6 * x + c5) * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42591 | } 42592 | }; 42593 | 42594 | template <typename Type> 42595 | struct poly_impl <Type,5> 42596 | { 42597 | static inline T evaluate(const Type x, 42598 | const Type c5, const Type c4, const Type c3, const Type c2, 42599 | const Type c1, const Type c0) 42600 | { 42601 | // p(x) = c_5x^5 + c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42602 | return (((((c5 * x + c4) * x + c3) * x + c2) * x + c1) * x + c0); 42603 | } 42604 | }; 42605 | 42606 | template <typename Type> 42607 | struct poly_impl <Type,4> 42608 | { 42609 | static inline T evaluate(const Type x, const Type c4, const Type c3, const Type c2, const Type c1, const Type c0) 42610 | { 42611 | // p(x) = c_4x^4 + c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42612 | return ((((c4 * x + c3) * x + c2) * x + c1) * x + c0); 42613 | } 42614 | }; 42615 | 42616 | template <typename Type> 42617 | struct poly_impl <Type,3> 42618 | { 42619 | static inline T evaluate(const Type x, const Type c3, const Type c2, const Type c1, const Type c0) 42620 | { 42621 | // p(x) = c_3x^3 + c_2x^2 + c_1x^1 + c_0x^0 42622 | return (((c3 * x + c2) * x + c1) * x + c0); 42623 | } 42624 | }; 42625 | 42626 | template <typename Type> 42627 | struct poly_impl <Type,2> 42628 | { 42629 | static inline T evaluate(const Type x, const Type c2, const Type c1, const Type c0) 42630 | { 42631 | // p(x) = c_2x^2 + c_1x^1 + c_0x^0 42632 | return ((c2 * x + c1) * x + c0); 42633 | } 42634 | }; 42635 | 42636 | template <typename Type> 42637 | struct poly_impl <Type,1> 42638 | { 42639 | static inline T evaluate(const Type x, const Type c1, const Type c0) 42640 | { 42641 | // p(x) = c_1x^1 + c_0x^0 42642 | return (c1 * x + c0); 42643 | } 42644 | }; 42645 | 42646 | public: 42647 | 42648 | using ifunction<T>::operator(); 42649 | 42650 | polynomial() 42651 | : ifunction<T>((N+2 <= 20) ? (N + 2) : std::numeric_limits<std::size_t>::max()) 42652 | { 42653 | disable_has_side_effects(*this); 42654 | } 42655 | 42656 | virtual ~polynomial() exprtk_override 42657 | {} 42658 | 42659 | #define poly_rtrn(NN) \ 42660 | return (NN != N) ? std::numeric_limits<T>::quiet_NaN() : 42661 | 42662 | inline virtual T operator() (const T& x, const T& c1, const T& c0) exprtk_override 42663 | { 42664 | poly_rtrn(1) (poly_impl<T,1>::evaluate(x, c1, c0)); 42665 | } 42666 | 42667 | inline virtual T operator() (const T& x, const T& c2, const T& c1, const T& c0) exprtk_override 42668 | { 42669 | poly_rtrn(2) (poly_impl<T,2>::evaluate(x, c2, c1, c0)); 42670 | } 42671 | 42672 | inline virtual T operator() (const T& x, const T& c3, const T& c2, const T& c1, const T& c0) exprtk_override 42673 | { 42674 | poly_rtrn(3) (poly_impl<T,3>::evaluate(x, c3, c2, c1, c0)); 42675 | } 42676 | 42677 | inline virtual T operator() (const T& x, const T& c4, const T& c3, const T& c2, const T& c1, 42678 | const T& c0) exprtk_override 42679 | { 42680 | poly_rtrn(4) (poly_impl<T,4>::evaluate(x, c4, c3, c2, c1, c0)); 42681 | } 42682 | 42683 | inline virtual T operator() (const T& x, const T& c5, const T& c4, const T& c3, const T& c2, 42684 | const T& c1, const T& c0) exprtk_override 42685 | { 42686 | poly_rtrn(5) (poly_impl<T,5>::evaluate(x, c5, c4, c3, c2, c1, c0)); 42687 | } 42688 | 42689 | inline virtual T operator() (const T& x, const T& c6, const T& c5, const T& c4, const T& c3, 42690 | const T& c2, const T& c1, const T& c0) exprtk_override 42691 | { 42692 | poly_rtrn(6) (poly_impl<T,6>::evaluate(x, c6, c5, c4, c3, c2, c1, c0)); 42693 | } 42694 | 42695 | inline virtual T operator() (const T& x, const T& c7, const T& c6, const T& c5, const T& c4, 42696 | const T& c3, const T& c2, const T& c1, const T& c0) exprtk_override 42697 | { 42698 | poly_rtrn(7) (poly_impl<T,7>::evaluate(x, c7, c6, c5, c4, c3, c2, c1, c0)); 42699 | } 42700 | 42701 | inline virtual T operator() (const T& x, const T& c8, const T& c7, const T& c6, const T& c5, 42702 | const T& c4, const T& c3, const T& c2, const T& c1, const T& c0) exprtk_override 42703 | { 42704 | poly_rtrn(8) (poly_impl<T,8>::evaluate(x, c8, c7, c6, c5, c4, c3, c2, c1, c0)); 42705 | } 42706 | 42707 | inline virtual T operator() (const T& x, const T& c9, const T& c8, const T& c7, const T& c6, 42708 | const T& c5, const T& c4, const T& c3, const T& c2, const T& c1, 42709 | const T& c0) exprtk_override 42710 | { 42711 | poly_rtrn(9) (poly_impl<T,9>::evaluate(x, c9, c8, c7, c6, c5, c4, c3, c2, c1, c0)); 42712 | } 42713 | 42714 | inline virtual T operator() (const T& x, const T& c10, const T& c9, const T& c8, const T& c7, 42715 | const T& c6, const T& c5, const T& c4, const T& c3, const T& c2, 42716 | const T& c1, const T& c0) exprtk_override 42717 | { 42718 | poly_rtrn(10) (poly_impl<T,10>::evaluate(x, c10, c9, c8, c7, c6, c5, c4, c3, c2, c1, c0)); 42719 | } 42720 | 42721 | inline virtual T operator() (const T& x, const T& c11, const T& c10, const T& c9, const T& c8, 42722 | const T& c7, const T& c6, const T& c5, const T& c4, const T& c3, 42723 | const T& c2, const T& c1, const T& c0) exprtk_override 42724 | { 42725 | poly_rtrn(11) (poly_impl<T,11>::evaluate(x, c11, c10, c9, c8, c7, c6, c5, c4, c3, c2, c1, c0)); 42726 | } 42727 | 42728 | inline virtual T operator() (const T& x, const T& c12, const T& c11, const T& c10, const T& c9, 42729 | const T& c8, const T& c7, const T& c6, const T& c5, const T& c4, 42730 | const T& c3, const T& c2, const T& c1, const T& c0) exprtk_override 42731 | { 42732 | poly_rtrn(12) (poly_impl<T,12>::evaluate(x, c12, c11, c10, c9, c8, c7, c6, c5, c4, c3, c2, c1, c0)); 42733 | } 42734 | 42735 | #undef poly_rtrn 42736 | 42737 | inline virtual T operator() () exprtk_override 42738 | { 42739 | return std::numeric_limits<T>::quiet_NaN(); 42740 | } 42741 | 42742 | inline virtual T operator() (const T&) exprtk_override 42743 | { 42744 | return std::numeric_limits<T>::quiet_NaN(); 42745 | } 42746 | 42747 | inline virtual T operator() (const T&, const T&) exprtk_override 42748 | { 42749 | return std::numeric_limits<T>::quiet_NaN(); 42750 | } 42751 | }; 42752 | 42753 | template <typename T> 42754 | class function_compositor 42755 | { 42756 | public: 42757 | 42758 | typedef exprtk::expression<T> expression_t; 42759 | typedef exprtk::symbol_table<T> symbol_table_t; 42760 | typedef exprtk::parser<T> parser_t; 42761 | typedef typename parser_t::settings_store settings_t; 42762 | 42763 | struct function 42764 | { 42765 | function() 42766 | {} 42767 | 42768 | explicit function(const std::string& n) 42769 | : name_(n) 42770 | {} 42771 | 42772 | function(const std::string& name, 42773 | const std::string& expression) 42774 | : name_(name) 42775 | , expression_(expression) 42776 | {} 42777 | 42778 | function(const std::string& name, 42779 | const std::string& expression, 42780 | const std::string& v0) 42781 | : name_(name) 42782 | , expression_(expression) 42783 | { 42784 | v_.push_back(v0); 42785 | } 42786 | 42787 | function(const std::string& name, 42788 | const std::string& expression, 42789 | const std::string& v0, const std::string& v1) 42790 | : name_(name) 42791 | , expression_(expression) 42792 | { 42793 | v_.push_back(v0); v_.push_back(v1); 42794 | } 42795 | 42796 | function(const std::string& name, 42797 | const std::string& expression, 42798 | const std::string& v0, const std::string& v1, 42799 | const std::string& v2) 42800 | : name_(name) 42801 | , expression_(expression) 42802 | { 42803 | v_.push_back(v0); v_.push_back(v1); 42804 | v_.push_back(v2); 42805 | } 42806 | 42807 | function(const std::string& name, 42808 | const std::string& expression, 42809 | const std::string& v0, const std::string& v1, 42810 | const std::string& v2, const std::string& v3) 42811 | : name_(name) 42812 | , expression_(expression) 42813 | { 42814 | v_.push_back(v0); v_.push_back(v1); 42815 | v_.push_back(v2); v_.push_back(v3); 42816 | } 42817 | 42818 | function(const std::string& name, 42819 | const std::string& expression, 42820 | const std::string& v0, const std::string& v1, 42821 | const std::string& v2, const std::string& v3, 42822 | const std::string& v4) 42823 | : name_(name) 42824 | , expression_(expression) 42825 | { 42826 | v_.push_back(v0); v_.push_back(v1); 42827 | v_.push_back(v2); v_.push_back(v3); 42828 | v_.push_back(v4); 42829 | } 42830 | 42831 | inline function& name(const std::string& n) 42832 | { 42833 | name_ = n; 42834 | return (*this); 42835 | } 42836 | 42837 | inline function& expression(const std::string& e) 42838 | { 42839 | expression_ = e; 42840 | return (*this); 42841 | } 42842 | 42843 | inline function& var(const std::string& v) 42844 | { 42845 | v_.push_back(v); 42846 | return (*this); 42847 | } 42848 | 42849 | inline function& vars(const std::string& v0, 42850 | const std::string& v1) 42851 | { 42852 | v_.push_back(v0); 42853 | v_.push_back(v1); 42854 | return (*this); 42855 | } 42856 | 42857 | inline function& vars(const std::string& v0, 42858 | const std::string& v1, 42859 | const std::string& v2) 42860 | { 42861 | v_.push_back(v0); 42862 | v_.push_back(v1); 42863 | v_.push_back(v2); 42864 | return (*this); 42865 | } 42866 | 42867 | inline function& vars(const std::string& v0, 42868 | const std::string& v1, 42869 | const std::string& v2, 42870 | const std::string& v3) 42871 | { 42872 | v_.push_back(v0); 42873 | v_.push_back(v1); 42874 | v_.push_back(v2); 42875 | v_.push_back(v3); 42876 | return (*this); 42877 | } 42878 | 42879 | inline function& vars(const std::string& v0, 42880 | const std::string& v1, 42881 | const std::string& v2, 42882 | const std::string& v3, 42883 | const std::string& v4) 42884 | { 42885 | v_.push_back(v0); 42886 | v_.push_back(v1); 42887 | v_.push_back(v2); 42888 | v_.push_back(v3); 42889 | v_.push_back(v4); 42890 | return (*this); 42891 | } 42892 | 42893 | std::string name_; 42894 | std::string expression_; 42895 | std::deque<std::string> v_; 42896 | }; 42897 | 42898 | private: 42899 | 42900 | struct base_func : public exprtk::ifunction<T> 42901 | { 42902 | typedef const T& type; 42903 | typedef exprtk::ifunction<T> function_t; 42904 | typedef std::vector<T*> varref_t; 42905 | typedef std::vector<T> var_t; 42906 | typedef std::vector<std::string> str_t; 42907 | typedef std::pair<T*,std::size_t> lvarref_t; 42908 | typedef std::vector<lvarref_t> lvr_vec_t; 42909 | typedef std::vector<std::string*> lstr_vec_t; 42910 | 42911 | using exprtk::ifunction<T>::operator(); 42912 | 42913 | explicit base_func(const std::size_t& pc = 0) 42914 | : exprtk::ifunction<T>(pc) 42915 | , local_var_stack_size(0) 42916 | , stack_depth(0) 42917 | { 42918 | v.resize(pc); 42919 | } 42920 | 42921 | virtual ~base_func() 42922 | {} 42923 | 42924 | #define exprtk_assign(Index) \ 42925 | (*v[Index]) = v##Index; \ 42926 | 42927 | inline void update(const T& v0) 42928 | { 42929 | exprtk_assign(0) 42930 | } 42931 | 42932 | inline void update(const T& v0, const T& v1) 42933 | { 42934 | exprtk_assign(0) exprtk_assign(1) 42935 | } 42936 | 42937 | inline void update(const T& v0, const T& v1, const T& v2) 42938 | { 42939 | exprtk_assign(0) exprtk_assign(1) 42940 | exprtk_assign(2) 42941 | } 42942 | 42943 | inline void update(const T& v0, const T& v1, const T& v2, const T& v3) 42944 | { 42945 | exprtk_assign(0) exprtk_assign(1) 42946 | exprtk_assign(2) exprtk_assign(3) 42947 | } 42948 | 42949 | inline void update(const T& v0, const T& v1, const T& v2, const T& v3, const T& v4) 42950 | { 42951 | exprtk_assign(0) exprtk_assign(1) 42952 | exprtk_assign(2) exprtk_assign(3) 42953 | exprtk_assign(4) 42954 | } 42955 | 42956 | inline void update(const T& v0, const T& v1, const T& v2, const T& v3, const T& v4, const T& v5) 42957 | { 42958 | exprtk_assign(0) exprtk_assign(1) 42959 | exprtk_assign(2) exprtk_assign(3) 42960 | exprtk_assign(4) exprtk_assign(5) 42961 | } 42962 | 42963 | #ifdef exprtk_assign 42964 | #undef exprtk_assign 42965 | #endif 42966 | 42967 | inline function_t& setup(expression_t& expr) 42968 | { 42969 | expression = expr; 42970 | 42971 | typedef typename expression_t::control_block ctrlblk_t; 42972 | typedef typename ctrlblk_t::local_data_list_t ldl_t; 42973 | typedef typename ctrlblk_t::data_type data_t; 42974 | typedef typename ldl_t::value_type ldl_value_type; 42975 | 42976 | const ldl_t ldl = expr.local_data_list(); 42977 | 42978 | std::vector<std::pair<std::size_t,data_t> > index_list; 42979 | 42980 | for (std::size_t i = 0; i < ldl.size(); ++i) 42981 | { 42982 | exprtk_debug(("base_func::setup() - element[%02d] type: %s size: %d\n", 42983 | static_cast<int>(i), 42984 | expression_t::control_block::to_str(ldl[i].type).c_str(), 42985 | static_cast<int>(ldl[i].size))); 42986 | 42987 | switch (ldl[i].type) 42988 | { 42989 | case ctrlblk_t::e_unknown : continue; 42990 | case ctrlblk_t::e_expr : continue; 42991 | case ctrlblk_t::e_vecholder : continue; 42992 | default : break; 42993 | } 42994 | 42995 | if (ldl[i].size) 42996 | { 42997 | index_list.push_back(std::make_pair(i,ldl[i].type)); 42998 | } 42999 | } 43000 | 43001 | std::size_t input_param_count = 0; 43002 | 43003 | for (std::size_t i = 0; i < index_list.size(); ++i) 43004 | { 43005 | const std::size_t index = index_list[i].first; 43006 | const ldl_value_type& local_var = ldl[index]; 43007 | 43008 | assert(local_var.pointer); 43009 | 43010 | if (i < (index_list.size() - v.size())) 43011 | { 43012 | if (local_var.type == ctrlblk_t::e_string) 43013 | { 43014 | local_str_vars.push_back( 43015 | reinterpret_cast<std::string*>(local_var.pointer)); 43016 | } 43017 | else if ( 43018 | (local_var.type == ctrlblk_t::e_data ) || 43019 | (local_var.type == ctrlblk_t::e_vecdata) 43020 | ) 43021 | { 43022 | local_vars.push_back(std::make_pair( 43023 | reinterpret_cast<T*>(local_var.pointer), 43024 | local_var.size)); 43025 | 43026 | local_var_stack_size += local_var.size; 43027 | } 43028 | } 43029 | else 43030 | { 43031 | v[input_param_count++] = reinterpret_cast<T*>(local_var.pointer); 43032 | } 43033 | } 43034 | 43035 | clear_stack(); 43036 | 43037 | return (*this); 43038 | } 43039 | 43040 | inline void pre() 43041 | { 43042 | if (stack_depth++) 43043 | { 43044 | if (!v.empty()) 43045 | { 43046 | var_t var_stack(v.size(),T(0)); 43047 | copy(v,var_stack); 43048 | input_params_stack.push_back(var_stack); 43049 | } 43050 | 43051 | if (!local_vars.empty()) 43052 | { 43053 | var_t local_vec_frame(local_var_stack_size,T(0)); 43054 | copy(local_vars,local_vec_frame); 43055 | local_var_stack.push_back(local_vec_frame); 43056 | } 43057 | 43058 | if (!local_str_vars.empty()) 43059 | { 43060 | str_t local_str_frame(local_str_vars.size()); 43061 | copy(local_str_vars,local_str_frame); 43062 | local_str_stack.push_back(local_str_frame); 43063 | } 43064 | } 43065 | } 43066 | 43067 | inline void post() 43068 | { 43069 | if (--stack_depth) 43070 | { 43071 | if (!v.empty()) 43072 | { 43073 | copy(input_params_stack.back(), v); 43074 | input_params_stack.pop_back(); 43075 | } 43076 | 43077 | if (!local_vars.empty()) 43078 | { 43079 | copy(local_var_stack.back(), local_vars); 43080 | local_var_stack.pop_back(); 43081 | } 43082 | 43083 | if (!local_str_vars.empty()) 43084 | { 43085 | copy(local_str_stack.back(), local_str_vars); 43086 | local_str_stack.pop_back(); 43087 | } 43088 | } 43089 | } 43090 | 43091 | void copy(const varref_t& src_v, var_t& dest_v) 43092 | { 43093 | for (std::size_t i = 0; i < src_v.size(); ++i) 43094 | { 43095 | dest_v[i] = (*src_v[i]); 43096 | } 43097 | } 43098 | 43099 | void copy(const lstr_vec_t& src_v, str_t& dest_v) 43100 | { 43101 | for (std::size_t i = 0; i < src_v.size(); ++i) 43102 | { 43103 | dest_v[i] = (*src_v[i]); 43104 | } 43105 | } 43106 | 43107 | void copy(const var_t& src_v, varref_t& dest_v) 43108 | { 43109 | for (std::size_t i = 0; i < src_v.size(); ++i) 43110 | { 43111 | (*dest_v[i]) = src_v[i]; 43112 | } 43113 | } 43114 | 43115 | void copy(const lvr_vec_t& src_v, var_t& dest_v) 43116 | { 43117 | typename var_t::iterator itr = dest_v.begin(); 43118 | typedef typename std::iterator_traits<typename var_t::iterator>::difference_type diff_t; 43119 | 43120 | for (std::size_t i = 0; i < src_v.size(); ++i) 43121 | { 43122 | lvarref_t vr = src_v[i]; 43123 | 43124 | if (1 == vr.second) 43125 | *itr++ = (*vr.first); 43126 | else 43127 | { 43128 | std::copy(vr.first, vr.first + vr.second, itr); 43129 | itr += static_cast<diff_t>(vr.second); 43130 | } 43131 | } 43132 | } 43133 | 43134 | void copy(const var_t& src_v, lvr_vec_t& dest_v) 43135 | { 43136 | typename var_t::const_iterator itr = src_v.begin(); 43137 | typedef typename std::iterator_traits<typename var_t::iterator>::difference_type diff_t; 43138 | 43139 | for (std::size_t i = 0; i < dest_v.size(); ++i) 43140 | { 43141 | lvarref_t& vr = dest_v[i]; 43142 | 43143 | assert(vr.first != 0); 43144 | assert(vr.second > 0); 43145 | 43146 | if (1 == vr.second) 43147 | (*vr.first) = *itr++; 43148 | else 43149 | { 43150 | std::copy(itr, itr + static_cast<diff_t>(vr.second), vr.first); 43151 | itr += static_cast<diff_t>(vr.second); 43152 | } 43153 | } 43154 | } 43155 | 43156 | void copy(const str_t& src_str, lstr_vec_t& dest_str) 43157 | { 43158 | assert(src_str.size() == dest_str.size()); 43159 | 43160 | for (std::size_t i = 0; i < dest_str.size(); ++i) 43161 | { 43162 | *dest_str[i] = src_str[i]; 43163 | } 43164 | } 43165 | 43166 | inline void clear_stack() 43167 | { 43168 | for (std::size_t i = 0; i < v.size(); ++i) 43169 | { 43170 | (*v[i]) = 0; 43171 | } 43172 | } 43173 | 43174 | inline virtual T value(expression_t& e) 43175 | { 43176 | return e.value(); 43177 | } 43178 | 43179 | expression_t expression; 43180 | varref_t v; 43181 | lvr_vec_t local_vars; 43182 | lstr_vec_t local_str_vars; 43183 | std::size_t local_var_stack_size; 43184 | std::size_t stack_depth; 43185 | std::deque<var_t> input_params_stack; 43186 | std::deque<var_t> local_var_stack; 43187 | std::deque<str_t> local_str_stack; 43188 | }; 43189 | 43190 | typedef std::map<std::string,base_func*> funcparam_t; 43191 | 43192 | typedef const T& type; 43193 | 43194 | template <typename BaseFuncType> 43195 | struct scoped_bft 43196 | { 43197 | explicit scoped_bft(BaseFuncType& bft) 43198 | : bft_(bft) 43199 | { 43200 | bft_.pre (); 43201 | } 43202 | 43203 | ~scoped_bft() 43204 | { 43205 | bft_.post(); 43206 | } 43207 | 43208 | BaseFuncType& bft_; 43209 | 43210 | private: 43211 | 43212 | scoped_bft(const scoped_bft&) exprtk_delete; 43213 | scoped_bft& operator=(const scoped_bft&) exprtk_delete; 43214 | }; 43215 | 43216 | struct func_0param : public base_func 43217 | { 43218 | using exprtk::ifunction<T>::operator(); 43219 | 43220 | func_0param() : base_func(0) {} 43221 | 43222 | inline T operator() () exprtk_override 43223 | { 43224 | scoped_bft<func_0param> sb(*this); 43225 | return this->value(base_func::expression); 43226 | } 43227 | }; 43228 | 43229 | struct func_1param : public base_func 43230 | { 43231 | using exprtk::ifunction<T>::operator(); 43232 | 43233 | func_1param() : base_func(1) {} 43234 | 43235 | inline T operator() (type v0) exprtk_override 43236 | { 43237 | scoped_bft<func_1param> sb(*this); 43238 | base_func::update(v0); 43239 | return this->value(base_func::expression); 43240 | } 43241 | }; 43242 | 43243 | struct func_2param : public base_func 43244 | { 43245 | using exprtk::ifunction<T>::operator(); 43246 | 43247 | func_2param() : base_func(2) {} 43248 | 43249 | inline T operator() (type v0, type v1) exprtk_override 43250 | { 43251 | scoped_bft<func_2param> sb(*this); 43252 | base_func::update(v0, v1); 43253 | return this->value(base_func::expression); 43254 | } 43255 | }; 43256 | 43257 | struct func_3param : public base_func 43258 | { 43259 | using exprtk::ifunction<T>::operator(); 43260 | 43261 | func_3param() : base_func(3) {} 43262 | 43263 | inline T operator() (type v0, type v1, type v2) exprtk_override 43264 | { 43265 | scoped_bft<func_3param> sb(*this); 43266 | base_func::update(v0, v1, v2); 43267 | return this->value(base_func::expression); 43268 | } 43269 | }; 43270 | 43271 | struct func_4param : public base_func 43272 | { 43273 | using exprtk::ifunction<T>::operator(); 43274 | 43275 | func_4param() : base_func(4) {} 43276 | 43277 | inline T operator() (type v0, type v1, type v2, type v3) exprtk_override 43278 | { 43279 | scoped_bft<func_4param> sb(*this); 43280 | base_func::update(v0, v1, v2, v3); 43281 | return this->value(base_func::expression); 43282 | } 43283 | }; 43284 | 43285 | struct func_5param : public base_func 43286 | { 43287 | using exprtk::ifunction<T>::operator(); 43288 | 43289 | func_5param() : base_func(5) {} 43290 | 43291 | inline T operator() (type v0, type v1, type v2, type v3, type v4) exprtk_override 43292 | { 43293 | scoped_bft<func_5param> sb(*this); 43294 | base_func::update(v0, v1, v2, v3, v4); 43295 | return this->value(base_func::expression); 43296 | } 43297 | }; 43298 | 43299 | struct func_6param : public base_func 43300 | { 43301 | using exprtk::ifunction<T>::operator(); 43302 | 43303 | func_6param() : base_func(6) {} 43304 | 43305 | inline T operator() (type v0, type v1, type v2, type v3, type v4, type v5) exprtk_override 43306 | { 43307 | scoped_bft<func_6param> sb(*this); 43308 | base_func::update(v0, v1, v2, v3, v4, v5); 43309 | return this->value(base_func::expression); 43310 | } 43311 | }; 43312 | 43313 | static T return_value(expression_t& e) 43314 | { 43315 | typedef exprtk::results_context<T> results_context_t; 43316 | typedef typename results_context_t::type_store_t type_t; 43317 | typedef typename type_t::scalar_view scalar_t; 43318 | 43319 | const T result = e.value(); 43320 | 43321 | if (e.return_invoked()) 43322 | { 43323 | // Due to the post compilation checks, it can be safely 43324 | // assumed that there will be at least one parameter 43325 | // and that the first parameter will always be scalar. 43326 | return scalar_t(e.results()[0])(); 43327 | } 43328 | 43329 | return result; 43330 | } 43331 | 43332 | #define def_fp_retval(N) \ 43333 | struct func_##N##param_retval exprtk_final : public func_##N##param \ 43334 | { \ 43335 | inline T value(expression_t& e) exprtk_override \ 43336 | { \ 43337 | return return_value(e); \ 43338 | } \ 43339 | }; \ 43340 | 43341 | def_fp_retval(0) 43342 | def_fp_retval(1) 43343 | def_fp_retval(2) 43344 | def_fp_retval(3) 43345 | def_fp_retval(4) 43346 | def_fp_retval(5) 43347 | def_fp_retval(6) 43348 | 43349 | #undef def_fp_retval 43350 | 43351 | template <typename Allocator, 43352 | template <typename, typename> class Sequence> 43353 | inline bool add(const std::string& name, 43354 | const std::string& expression, 43355 | const Sequence<std::string,Allocator>& var_list, 43356 | const bool override = false) 43357 | { 43358 | const typename std::map<std::string,expression_t>::iterator itr = expr_map_.find(name); 43359 | 43360 | if (expr_map_.end() != itr) 43361 | { 43362 | if (!override) 43363 | { 43364 | exprtk_debug(("Compositor error(add): function '%s' already defined\n", 43365 | name.c_str())); 43366 | 43367 | return false; 43368 | } 43369 | 43370 | remove(name, var_list.size()); 43371 | } 43372 | 43373 | if (compile_expression(name, expression, var_list)) 43374 | { 43375 | const std::size_t n = var_list.size(); 43376 | 43377 | fp_map_[n][name]->setup(expr_map_[name]); 43378 | 43379 | return true; 43380 | } 43381 | else 43382 | { 43383 | exprtk_debug(("Compositor error(add): Failed to compile function '%s'\n", 43384 | name.c_str())); 43385 | 43386 | return false; 43387 | } 43388 | } 43389 | 43390 | public: 43391 | 43392 | function_compositor() 43393 | : parser_(settings_t::default_compile_all_opts + 43394 | settings_t::e_disable_zero_return) 43395 | , fp_map_(7) 43396 | , load_variables_(false) 43397 | , load_vectors_(false) 43398 | {} 43399 | 43400 | explicit function_compositor(const symbol_table_t& st) 43401 | : symbol_table_(st) 43402 | , parser_(settings_t::default_compile_all_opts + 43403 | settings_t::e_disable_zero_return) 43404 | , fp_map_(7) 43405 | , load_variables_(false) 43406 | , load_vectors_(false) 43407 | {} 43408 | 43409 | ~function_compositor() 43410 | { 43411 | clear(); 43412 | } 43413 | 43414 | inline symbol_table_t& symbol_table() 43415 | { 43416 | return symbol_table_; 43417 | } 43418 | 43419 | inline const symbol_table_t& symbol_table() const 43420 | { 43421 | return symbol_table_; 43422 | } 43423 | 43424 | inline void add_auxiliary_symtab(symbol_table_t& symtab) 43425 | { 43426 | auxiliary_symtab_list_.push_back(&symtab); 43427 | } 43428 | 43429 | void load_variables(const bool load = true) 43430 | { 43431 | load_variables_ = load; 43432 | } 43433 | 43434 | void load_vectors(const bool load = true) 43435 | { 43436 | load_vectors_ = load; 43437 | } 43438 | 43439 | inline void register_loop_runtime_check(loop_runtime_check& lrtchk) 43440 | { 43441 | parser_.register_loop_runtime_check(lrtchk); 43442 | } 43443 | 43444 | inline void register_vector_access_runtime_check(vector_access_runtime_check& vartchk) 43445 | { 43446 | parser_.register_vector_access_runtime_check(vartchk); 43447 | } 43448 | 43449 | inline void register_compilation_timeout_check(compilation_check& compchk) 43450 | { 43451 | parser_.register_compilation_timeout_check(compchk); 43452 | } 43453 | 43454 | inline void clear_loop_runtime_check() 43455 | { 43456 | parser_.clear_loop_runtime_check(); 43457 | } 43458 | 43459 | inline void clear_vector_access_runtime_check() 43460 | { 43461 | parser_.clear_vector_access_runtime_check(); 43462 | } 43463 | 43464 | inline void clear_compilation_timeout_check() 43465 | { 43466 | parser_.clear_compilation_timeout_check(); 43467 | } 43468 | 43469 | void clear() 43470 | { 43471 | symbol_table_.clear(); 43472 | expr_map_ .clear(); 43473 | 43474 | for (std::size_t i = 0; i < fp_map_.size(); ++i) 43475 | { 43476 | typename funcparam_t::iterator itr = fp_map_[i].begin(); 43477 | typename funcparam_t::iterator end = fp_map_[i].end (); 43478 | 43479 | while (itr != end) 43480 | { 43481 | delete itr->second; 43482 | ++itr; 43483 | } 43484 | 43485 | fp_map_[i].clear(); 43486 | } 43487 | 43488 | clear_loop_runtime_check (); 43489 | clear_vector_access_runtime_check(); 43490 | clear_compilation_timeout_check (); 43491 | } 43492 | 43493 | inline bool add(const function& f, const bool override = false) 43494 | { 43495 | return add(f.name_, f.expression_, f.v_,override); 43496 | } 43497 | 43498 | inline std::string error() const 43499 | { 43500 | if (!error_list_.empty()) 43501 | { 43502 | return error_list_[0].diagnostic; 43503 | } 43504 | else 43505 | return std::string("No Error"); 43506 | } 43507 | 43508 | inline std::size_t error_count() const 43509 | { 43510 | return error_list_.size(); 43511 | } 43512 | 43513 | inline parser_error::type get_error(const std::size_t& index) const 43514 | { 43515 | if (index < error_list_.size()) 43516 | { 43517 | return error_list_[index]; 43518 | } 43519 | 43520 | throw std::invalid_argument("compositor::get_error() - Invalid error index specified"); 43521 | } 43522 | 43523 | private: 43524 | 43525 | template <typename Allocator, 43526 | template <typename, typename> class Sequence> 43527 | bool compile_expression(const std::string& name, 43528 | const std::string& expression, 43529 | const Sequence<std::string,Allocator>& input_var_list, 43530 | bool return_present = false) 43531 | { 43532 | expression_t compiled_expression; 43533 | symbol_table_t local_symbol_table; 43534 | 43535 | local_symbol_table.load_from(symbol_table_); 43536 | local_symbol_table.add_constants(); 43537 | 43538 | if (load_variables_) 43539 | { 43540 | local_symbol_table.load_variables_from(symbol_table_); 43541 | } 43542 | 43543 | if (load_vectors_) 43544 | { 43545 | local_symbol_table.load_vectors_from(symbol_table_); 43546 | } 43547 | 43548 | error_list_.clear(); 43549 | 43550 | if (!valid(name,input_var_list.size())) 43551 | { 43552 | parser_error::type error = 43553 | parser_error::make_error( 43554 | parser_error::e_parser, 43555 | lexer::token(), 43556 | "ERR283 - Function '" + name + "' is an invalid overload", 43557 | exprtk_error_location); 43558 | 43559 | error_list_.push_back(error); 43560 | return false; 43561 | } 43562 | 43563 | if (!forward(name, 43564 | input_var_list.size(), 43565 | local_symbol_table, 43566 | return_present)) 43567 | return false; 43568 | 43569 | compiled_expression.register_symbol_table(local_symbol_table); 43570 | 43571 | for (std::size_t i = 0; i < auxiliary_symtab_list_.size(); ++i) 43572 | { 43573 | compiled_expression.register_symbol_table((*auxiliary_symtab_list_[i])); 43574 | } 43575 | 43576 | std::string mod_expression; 43577 | 43578 | for (std::size_t i = 0; i < input_var_list.size(); ++i) 43579 | { 43580 | mod_expression += " var " + input_var_list[i] + "{};\n" 43581 | } 43582 | 43583 | if ( 43584 | ('{' == details::front(expression)) && 43585 | ('}' == details::back (expression)) 43586 | ) 43587 | mod_expression += "~" + expression + "" 43588 | else 43589 | mod_expression += "~{" + expression + "};" 43590 | 43591 | if (!parser_.compile(mod_expression,compiled_expression)) 43592 | { 43593 | exprtk_debug(("Compositor Error: %s\n", parser_.error().c_str())); 43594 | exprtk_debug(("Compositor modified expression: \n%s\n", mod_expression.c_str())); 43595 | 43596 | remove(name,input_var_list.size()); 43597 | 43598 | for (std::size_t err_index = 0; err_index < parser_.error_count(); ++err_index) 43599 | { 43600 | error_list_.push_back(parser_.get_error(err_index)); 43601 | } 43602 | 43603 | return false; 43604 | } 43605 | 43606 | if (!return_present && parser_.dec().return_present()) 43607 | { 43608 | remove(name,input_var_list.size()); 43609 | return compile_expression(name, expression, input_var_list, true); 43610 | } 43611 | 43612 | // Make sure every return point has a scalar as its first parameter 43613 | if (parser_.dec().return_present()) 43614 | { 43615 | typedef std::vector<std::string> str_list_t; 43616 | 43617 | str_list_t ret_param_list = parser_.dec().return_param_type_list(); 43618 | 43619 | for (std::size_t i = 0; i < ret_param_list.size(); ++i) 43620 | { 43621 | const std::string& params = ret_param_list[i]; 43622 | 43623 | if (params.empty() || ('T' != params[0])) 43624 | { 43625 | exprtk_debug(("Compositor Error: Return statement in function '%s' is invalid\n", 43626 | name.c_str())); 43627 | 43628 | remove(name,input_var_list.size()); 43629 | 43630 | return false; 43631 | } 43632 | } 43633 | } 43634 | 43635 | expr_map_[name] = compiled_expression; 43636 | 43637 | exprtk::ifunction<T>& ifunc = (*(fp_map_[input_var_list.size()])[name]); 43638 | 43639 | if (symbol_table_.add_function(name,ifunc)) 43640 | return true; 43641 | else 43642 | { 43643 | exprtk_debug(("Compositor Error: Failed to add function '%s' to symbol table\n", 43644 | name.c_str())); 43645 | return false; 43646 | } 43647 | } 43648 | 43649 | inline bool symbol_used(const std::string& symbol) const 43650 | { 43651 | return ( 43652 | symbol_table_.is_variable (symbol) || 43653 | symbol_table_.is_stringvar (symbol) || 43654 | symbol_table_.is_function (symbol) || 43655 | symbol_table_.is_vector (symbol) || 43656 | symbol_table_.is_vararg_function(symbol) 43657 | ); 43658 | } 43659 | 43660 | inline bool valid(const std::string& name, 43661 | const std::size_t& arg_count) const 43662 | { 43663 | if (arg_count > 6) 43664 | return false; 43665 | else if (symbol_used(name)) 43666 | return false; 43667 | else if (fp_map_[arg_count].end() != fp_map_[arg_count].find(name)) 43668 | return false; 43669 | else 43670 | return true; 43671 | } 43672 | 43673 | inline bool forward(const std::string& name, 43674 | const std::size_t& arg_count, 43675 | symbol_table_t& sym_table, 43676 | const bool ret_present = false) 43677 | { 43678 | switch (arg_count) 43679 | { 43680 | #define case_stmt(N) \ 43681 | case N : (fp_map_[arg_count])[name] = \ 43682 | (!ret_present) ? static_cast<base_func*> \ 43683 | (new func_##N##param) : \ 43684 | static_cast<base_func*> \ 43685 | (new func_##N##param_retval) ; \ 43686 | break; \ 43687 | 43688 | case_stmt(0) case_stmt(1) case_stmt(2) 43689 | case_stmt(3) case_stmt(4) case_stmt(5) 43690 | case_stmt(6) 43691 | #undef case_stmt 43692 | } 43693 | 43694 | exprtk::ifunction<T>& ifunc = (*(fp_map_[arg_count])[name]); 43695 | 43696 | return sym_table.add_function(name,ifunc); 43697 | } 43698 | 43699 | inline void remove(const std::string& name, const std::size_t& arg_count) 43700 | { 43701 | if (arg_count > 6) 43702 | return; 43703 | 43704 | const typename std::map<std::string,expression_t>::iterator em_itr = expr_map_.find(name); 43705 | 43706 | if (expr_map_.end() != em_itr) 43707 | { 43708 | expr_map_.erase(em_itr); 43709 | } 43710 | 43711 | const typename funcparam_t::iterator fp_itr = fp_map_[arg_count].find(name); 43712 | 43713 | if (fp_map_[arg_count].end() != fp_itr) 43714 | { 43715 | delete fp_itr->second; 43716 | fp_map_[arg_count].erase(fp_itr); 43717 | } 43718 | 43719 | symbol_table_.remove_function(name); 43720 | } 43721 | 43722 | private: 43723 | 43724 | symbol_table_t symbol_table_; 43725 | parser_t parser_; 43726 | std::map<std::string,expression_t> expr_map_; 43727 | std::vector<funcparam_t> fp_map_; 43728 | std::vector<symbol_table_t*> auxiliary_symtab_list_; 43729 | std::deque<parser_error::type> error_list_; 43730 | bool load_variables_; 43731 | bool load_vectors_; 43732 | }; // class function_compositor 43733 | 43734 | } // namespace exprtk 43735 | 43736 | #if defined(_MSC_VER) || defined(_WIN32) || defined(__WIN32__) || defined(WIN32) 43737 | # ifndef NOMINMAX 43738 | # define NOMINMAX 43739 | # endif 43740 | # ifndef WIN32_LEAN_AND_MEAN 43741 | # define WIN32_LEAN_AND_MEAN 43742 | # endif 43743 | # include43744 | # include 43745 | #else 43746 | # include 43747 | # include 43748 | # include 43749 | #endif 43750 | 43751 | namespace exprtk 43752 | { 43753 | class timer 43754 | { 43755 | public: 43756 | 43757 | #if defined(_MSC_VER) || defined(_WIN32) || defined(__WIN32__) || defined(WIN32) 43758 | timer() 43759 | : in_use_(false) 43760 | , start_time_{ {0, 0} } 43761 | , stop_time_ { {0, 0} } 43762 | { 43763 | QueryPerformanceFrequency(&clock_frequency_); 43764 | } 43765 | 43766 | inline void start() 43767 | { 43768 | in_use_ = true; 43769 | QueryPerformanceCounter(&start_time_); 43770 | } 43771 | 43772 | inline void stop() 43773 | { 43774 | QueryPerformanceCounter(&stop_time_); 43775 | in_use_ = false; 43776 | } 43777 | 43778 | inline double time() const 43779 | { 43780 | return (1.0 * (stop_time_.QuadPart - start_time_.QuadPart)) / (1.0 * clock_frequency_.QuadPart); 43781 | } 43782 | 43783 | #else 43784 | 43785 | timer() 43786 | : in_use_(false) 43787 | { 43788 | start_time_.tv_sec = 0; 43789 | start_time_.tv_usec = 0; 43790 | 43791 | stop_time_.tv_sec = 0; 43792 | stop_time_.tv_usec = 0; 43793 | } 43794 | 43795 | inline void start() 43796 | { 43797 | in_use_ = true; 43798 | gettimeofday(&start_time_,0); 43799 | } 43800 | 43801 | inline void stop() 43802 | { 43803 | gettimeofday(&stop_time_, 0); 43804 | in_use_ = false; 43805 | } 43806 | 43807 | inline unsigned long long int usec_time() const 43808 | { 43809 | if (!in_use_) 43810 | { 43811 | if (stop_time_.tv_sec >= start_time_.tv_sec) 43812 | { 43813 | return 1000000LLU * static_cast<details::_uint64_t>(stop_time_.tv_sec - start_time_.tv_sec ) + 43814 | static_cast<details::_uint64_t>(stop_time_.tv_usec - start_time_.tv_usec) ; 43815 | } 43816 | else 43817 | return std::numeric_limits<details::_uint64_t>::max(); 43818 | } 43819 | else 43820 | return std::numeric_limits<details::_uint64_t>::max(); 43821 | } 43822 | 43823 | inline double time() const 43824 | { 43825 | return usec_time() * 0.000001; 43826 | } 43827 | 43828 | #endif 43829 | 43830 | inline bool in_use() const 43831 | { 43832 | return in_use_; 43833 | } 43834 | 43835 | private: 43836 | 43837 | bool in_use_; 43838 | 43839 | #if defined(_MSC_VER) || defined(_WIN32) || defined(__WIN32__) || defined(WIN32) 43840 | LARGE_INTEGER start_time_; 43841 | LARGE_INTEGER stop_time_; 43842 | LARGE_INTEGER clock_frequency_; 43843 | #else 43844 | struct timeval start_time_; 43845 | struct timeval stop_time_; 43846 | #endif 43847 | }; 43848 | 43849 | template <typename T> 43850 | struct type_defs 43851 | { 43852 | typedef symbol_table<T> symbol_table_t; 43853 | typedef expression<T> expression_t; 43854 | typedef parser<T> parser_t; 43855 | typedef parser_error::type error_t; 43856 | typedef function_compositor<T> compositor_t; 43857 | typedef typename compositor_t::function function_t; 43858 | }; 43859 | 43860 | } // namespace exprtk 43861 | 43862 | #ifndef exprtk_disable_rtl_io 43863 | namespace exprtk 43864 | { 43865 | namespace rtl { namespace io { namespace details 43866 | { 43867 | template <typename T> 43868 | inline void print_type(const std::string& fmt, 43869 | const T v, 43870 | exprtk::details::numeric::details::real_type_tag) 43871 | { 43872 | #if defined(__clang__) 43873 | #pragma clang diagnostic push 43874 | #pragma clang diagnostic ignored "-Wformat-nonliteral" 43875 | #elif defined(__GNUC__) || defined(__GNUG__) 43876 | #pragma GCC diagnostic push 43877 | #pragma GCC diagnostic ignored "-Wformat-nonliteral" 43878 | #elif defined(_MSC_VER) 43879 | #endif 43880 | 43881 | printf(fmt.c_str(), v); 43882 | 43883 | #if defined(__clang__) 43884 | #pragma clang diagnostic pop 43885 | #elif defined(__GNUC__) || defined(__GNUG__) 43886 | #pragma GCC diagnostic pop 43887 | #elif defined(_MSC_VER) 43888 | #endif 43889 | } 43890 | 43891 | template <typename T> 43892 | struct print_impl 43893 | { 43894 | typedef typename igeneric_function<T>::generic_type generic_type; 43895 | typedef typename igeneric_function<T>::parameter_list_t parameter_list_t; 43896 | typedef typename generic_type::scalar_view scalar_t; 43897 | typedef typename generic_type::vector_view vector_t; 43898 | typedef typename generic_type::string_view string_t; 43899 | typedef typename exprtk::details::numeric::details::number_type<T>::type num_type; 43900 | 43901 | static void process(const std::string& scalar_format, parameter_list_t parameters) 43902 | { 43903 | for (std::size_t i = 0; i < parameters.size(); ++i) 43904 | { 43905 | generic_type& gt = parameters[i]; 43906 | 43907 | switch (gt.type) 43908 | { 43909 | case generic_type::e_scalar : print(scalar_format,scalar_t(gt)); 43910 | break; 43911 | 43912 | case generic_type::e_vector : print(scalar_format,vector_t(gt)); 43913 | break; 43914 | 43915 | case generic_type::e_string : print(string_t(gt)); 43916 | break; 43917 | 43918 | default : continue; 43919 | } 43920 | } 43921 | } 43922 | 43923 | static inline void print(const std::string& scalar_format, const scalar_t& s) 43924 | { 43925 | print_type(scalar_format,s(),num_type()); 43926 | } 43927 | 43928 | static inline void print(const std::string& scalar_format, const vector_t& v) 43929 | { 43930 | for (std::size_t i = 0; i < v.size(); ++i) 43931 | { 43932 | print_type(scalar_format,v[i],num_type()); 43933 | 43934 | if ((i + 1) < v.size()) 43935 | printf(" "); 43936 | } 43937 | } 43938 | 43939 | static inline void print(const string_t& s) 43940 | { 43941 | printf("%s",to_str(s).c_str()); 43942 | } 43943 | }; 43944 | 43945 | } // namespace exprtk::rtl::io::details 43946 | 43947 | template <typename T> 43948 | struct print exprtk_final : public exprtk::igeneric_function<T> 43949 | { 43950 | typedef typename igeneric_function<T>::parameter_list_t parameter_list_t; 43951 | 43952 | using exprtk::igeneric_function<T>::operator(); 43953 | 43954 | explicit print(const std::string& scalar_format = "%10.5f") 43955 | : scalar_format_(scalar_format) 43956 | { 43957 | exprtk::enable_zero_parameters(*this); 43958 | } 43959 | 43960 | inline T operator() (parameter_list_t parameters) exprtk_override 43961 | { 43962 | details::print_impl<T>::process(scalar_format_,parameters); 43963 | return T(0); 43964 | } 43965 | 43966 | std::string scalar_format_; 43967 | }; 43968 | 43969 | template <typename T> 43970 | struct println exprtk_final : public exprtk::igeneric_function<T> 43971 | { 43972 | typedef typename igeneric_function<T>::parameter_list_t parameter_list_t; 43973 | 43974 | using exprtk::igeneric_function<T>::operator(); 43975 | 43976 | explicit println(const std::string& scalar_format = "%10.5f") 43977 | : scalar_format_(scalar_format) 43978 | { 43979 | exprtk::enable_zero_parameters(*this); 43980 | } 43981 | 43982 | inline T operator() (parameter_list_t parameters) exprtk_override 43983 | { 43984 | details::print_impl<T>::process(scalar_format_,parameters); 43985 | printf("\n"); 43986 | return T(0); 43987 | } 43988 | 43989 | std::string scalar_format_; 43990 | }; 43991 | 43992 | template <typename T> 43993 | struct package 43994 | { 43995 | print <T> p; 43996 | println<T> pl; 43997 | 43998 | bool register_package(exprtk::symbol_table<T>& symtab) 43999 | { 44000 | #define exprtk_register_function(FunctionName, FunctionType) \ 44001 | if (!symtab.add_function(FunctionName,FunctionType)) \ 44002 | { \ 44003 | exprtk_debug(( \ 44004 | "exprtk::rtl::io::register_package - Failed to add function: %s\n", \ 44005 | FunctionName)); \ 44006 | return false; \ 44007 | } \ 44008 | 44009 | exprtk_register_function("print" , p ) 44010 | exprtk_register_function("println", pl) 44011 | #undef exprtk_register_function 44012 | 44013 | return true; 44014 | } 44015 | }; 44016 | 44017 | } // namespace exprtk::rtl::io 44018 | } // namespace exprtk::rtl 44019 | } // namespace exprtk 44020 | #endif 44021 | 44022 | #ifndef exprtk_disable_rtl_io_file 44023 | #include 44024 | namespace exprtk 44025 | { 44026 | namespace rtl { namespace io { namespace file { namespace details 44027 | { 44028 | using ::exprtk::details::char_ptr; 44029 | using ::exprtk::details::char_cptr; 44030 | 44031 | enum file_mode 44032 | { 44033 | e_error = 0, 44034 | e_read = 1, 44035 | e_write = 2, 44036 | e_rdwrt = 4 44037 | }; 44038 | 44039 | struct file_descriptor 44040 | { 44041 | file_descriptor(const std::string& fname, const std::string& access) 44042 | : stream_ptr(0) 44043 | , mode(get_file_mode(access)) 44044 | , file_name(fname) 44045 | {} 44046 | 44047 | void* stream_ptr; 44048 | file_mode mode; 44049 | std::string file_name; 44050 | 44051 | bool open() 44052 | { 44053 | if (e_read == mode) 44054 | { 44055 | std::ifstream* stream = new std::ifstream(file_name.c_str(),std::ios::binary); 44056 | 44057 | if (!(*stream)) 44058 | { 44059 | file_name.clear(); 44060 | delete stream; 44061 | 44062 | return false; 44063 | } 44064 | 44065 | stream_ptr = stream; 44066 | 44067 | return true; 44068 | } 44069 | else if (e_write == mode) 44070 | { 44071 | std::ofstream* stream = new std::ofstream(file_name.c_str(),std::ios::binary); 44072 | 44073 | if (!(*stream)) 44074 | { 44075 | file_name.clear(); 44076 | delete stream; 44077 | 44078 | return false; 44079 | } 44080 | 44081 | stream_ptr = stream; 44082 | 44083 | return true; 44084 | } 44085 | else if (e_rdwrt == mode) 44086 | { 44087 | std::fstream* stream = new std::fstream(file_name.c_str(),std::ios::binary); 44088 | 44089 | if (!(*stream)) 44090 | { 44091 | file_name.clear(); 44092 | delete stream; 44093 | 44094 | return false; 44095 | } 44096 | 44097 | stream_ptr = stream; 44098 | 44099 | return true; 44100 | } 44101 | 44102 | return false; 44103 | } 44104 | 44105 | template <typename Stream, typename Ptr> 44106 | void close(Ptr& p) 44107 | { 44108 | Stream* stream = reinterpret_cast<Stream*>(p); 44109 | stream->close(); 44110 | delete stream; 44111 | p = reinterpret_cast<Ptr>(0); 44112 | } 44113 | 44114 | bool close() 44115 | { 44116 | switch (mode) 44117 | { 44118 | case e_read : close<std::ifstream>(stream_ptr); 44119 | break; 44120 | 44121 | case e_write : close<std::ofstream>(stream_ptr); 44122 | break; 44123 | 44124 | case e_rdwrt : close<std::fstream> (stream_ptr); 44125 | break; 44126 | 44127 | default : return false; 44128 | } 44129 | 44130 | return true; 44131 | } 44132 | 44133 | template <typename View> 44134 | bool write(const View& view, const std::size_t amount, const std::size_t offset = 0) 44135 | { 44136 | switch (mode) 44137 | { 44138 | case e_write : reinterpret_cast<std::ofstream*>(stream_ptr)-> 44139 | write(reinterpret_cast<char_cptr>(view.begin() + offset), amount * sizeof(typename View::value_t)); 44140 | break; 44141 | 44142 | case e_rdwrt : reinterpret_cast<std::fstream*>(stream_ptr)-> 44143 | write(reinterpret_cast<char_cptr>(view.begin() + offset) , amount * sizeof(typename View::value_t)); 44144 | break; 44145 | 44146 | default : return false; 44147 | } 44148 | 44149 | return true; 44150 | } 44151 | 44152 | template <typename View> 44153 | bool read(View& view, const std::size_t amount, const std::size_t offset = 0) 44154 | { 44155 | switch (mode) 44156 | { 44157 | case e_read : reinterpret_cast<std::ifstream*>(stream_ptr)-> 44158 | read(reinterpret_cast<char_ptr>(view.begin() + offset), amount * sizeof(typename View::value_t)); 44159 | break; 44160 | 44161 | case e_rdwrt : reinterpret_cast<std::fstream*>(stream_ptr)-> 44162 | read(reinterpret_cast<char_ptr>(view.begin() + offset) , amount * sizeof(typename View::value_t)); 44163 | break; 44164 | 44165 | default : return false; 44166 | } 44167 | 44168 | return true; 44169 | } 44170 | 44171 | bool getline(std::string& s) 44172 | { 44173 | switch (mode) 44174 | { 44175 | case e_read : return (!!std::getline(*reinterpret_cast<std::ifstream*>(stream_ptr),s)); 44176 | case e_rdwrt : return (!!std::getline(*reinterpret_cast<std::fstream* >(stream_ptr),s)); 44177 | default : return false; 44178 | } 44179 | } 44180 | 44181 | bool eof() const 44182 | { 44183 | switch (mode) 44184 | { 44185 | case e_read : return reinterpret_cast<std::ifstream*>(stream_ptr)->eof(); 44186 | case e_write : return reinterpret_cast<std::ofstream*>(stream_ptr)->eof(); 44187 | case e_rdwrt : return reinterpret_cast<std::fstream* >(stream_ptr)->eof(); 44188 | default : return true; 44189 | } 44190 | } 44191 | 44192 | file_mode get_file_mode(const std::string& access) const 44193 | { 44194 | if (access.empty() || access.size() > 2) 44195 | return e_error; 44196 | 44197 | std::size_t w_cnt = 0; 44198 | std::size_t r_cnt = 0; 44199 | 44200 | for (std::size_t i = 0; i < access.size(); ++i) 44201 | { 44202 | switch (std::tolower(access[i])) 44203 | { 44204 | case 'r' : r_cnt++; break; 44205 | case 'w' : w_cnt++; break; 44206 | default : return e_error; 44207 | } 44208 | } 44209 | 44210 | if ((0 == r_cnt) && (0 == w_cnt)) 44211 | return e_error; 44212 | else if ((r_cnt > 1) || (w_cnt > 1)) 44213 | return e_error; 44214 | else if ((1 == r_cnt) && (1 == w_cnt)) 44215 | return e_rdwrt; 44216 | else if (1 == r_cnt) 44217 | return e_read; 44218 | else 44219 | return e_write; 44220 | } 44221 | }; 44222 | 44223 | template <typename T> 44224 | file_descriptor* make_handle(T v) 44225 | { 44226 | const std::size_t fd_size = sizeof(details::file_descriptor*); 44227 | details::file_descriptor* fd = reinterpret_cast<file_descriptor*>(0); 44228 | 44229 | std::memcpy(reinterpret_cast<char_ptr >(&fd), 44230 | reinterpret_cast<char_cptr>(&v ), 44231 | fd_size); 44232 | return fd; 44233 | } 44234 | 44235 | template <typename T> 44236 | void perform_check() 44237 | { 44238 | #ifdef _MSC_VER 44239 | #pragma warning(push) 44240 | #pragma warning(disable: 4127) 44241 | #endif 44242 | if (sizeof(T) < sizeof(void*)) 44243 | { 44244 | throw std::runtime_error("exprtk::rtl::io::file - Error - pointer size larger than holder."); 44245 | } 44246 | #ifdef _MSC_VER 44247 | #pragma warning(pop) 44248 | #endif 44249 | assert(sizeof(T) <= sizeof(void*)); 44250 | } 44251 | 44252 | } // namespace exprtk::rtl::io::file::details 44253 | 44254 | template <typename T> 44255 | class open exprtk_final : public exprtk::igeneric_function<T> 44256 | { 44257 | public: 44258 | 44259 | typedef typename exprtk::igeneric_function<T> igfun_t; 44260 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44261 | typedef typename igfun_t::generic_type generic_type; 44262 | typedef typename generic_type::string_view string_t; 44263 | 44264 | using igfun_t::operator(); 44265 | 44266 | open() 44267 | : exprtk::igeneric_function<T>("S|SS") 44268 | { details::perform_check<T>(); } 44269 | 44270 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44271 | { 44272 | const std::string file_name = to_str(string_t(parameters[0])); 44273 | 44274 | if (file_name.empty()) 44275 | { 44276 | return T(0); 44277 | } 44278 | 44279 | if ((1 == ps_index) && (0 == string_t(parameters[1]).size())) 44280 | { 44281 | return T(0); 44282 | } 44283 | 44284 | const std::string access = 44285 | (0 == ps_index) ? "r" : to_str(string_t(parameters[1])); 44286 | 44287 | details::file_descriptor* fd = new details::file_descriptor(file_name,access); 44288 | 44289 | if (fd->open()) 44290 | { 44291 | T t = T(0); 44292 | 44293 | const std::size_t fd_size = sizeof(details::file_descriptor*); 44294 | 44295 | std::memcpy(reinterpret_cast<char*>(&t ), 44296 | reinterpret_cast<char*>(&fd), 44297 | fd_size); 44298 | return t; 44299 | } 44300 | else 44301 | { 44302 | delete fd; 44303 | return T(0); 44304 | } 44305 | } 44306 | }; 44307 | 44308 | template <typename T> 44309 | struct close exprtk_final : public exprtk::ifunction<T> 44310 | { 44311 | using exprtk::ifunction<T>::operator(); 44312 | 44313 | close() 44314 | : exprtk::ifunction<T>(1) 44315 | { details::perform_check<T>(); } 44316 | 44317 | inline T operator() (const T& v) exprtk_override 44318 | { 44319 | details::file_descriptor* fd = details::make_handle(v); 44320 | 44321 | if (!fd->close()) 44322 | return T(0); 44323 | 44324 | delete fd; 44325 | 44326 | return T(1); 44327 | } 44328 | }; 44329 | 44330 | template <typename T> 44331 | class write exprtk_final : public exprtk::igeneric_function<T> 44332 | { 44333 | public: 44334 | 44335 | typedef typename exprtk::igeneric_function<T> igfun_t; 44336 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44337 | typedef typename igfun_t::generic_type generic_type; 44338 | typedef typename generic_type::string_view string_t; 44339 | typedef typename generic_type::scalar_view scalar_t; 44340 | typedef typename generic_type::vector_view vector_t; 44341 | 44342 | using igfun_t::operator(); 44343 | 44344 | write() 44345 | : igfun_t("TS|TST|TV|TVT") 44346 | { details::perform_check<T>(); } 44347 | 44348 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44349 | { 44350 | details::file_descriptor* fd = details::make_handle(scalar_t(parameters[0])()); 44351 | 44352 | switch (ps_index) 44353 | { 44354 | case 0 : { 44355 | const string_t buffer(parameters[1]); 44356 | const std::size_t amount = buffer.size(); 44357 | return T(fd->write(buffer, amount) ? 1 : 0); 44358 | } 44359 | 44360 | case 1 : { 44361 | const string_t buffer(parameters[1]); 44362 | const std::size_t amount = 44363 | std::min(buffer.size(), 44364 | static_cast<std::size_t>(scalar_t(parameters[2])())); 44365 | return T(fd->write(buffer, amount) ? 1 : 0); 44366 | } 44367 | 44368 | case 2 : { 44369 | const vector_t vec(parameters[1]); 44370 | const std::size_t amount = vec.size(); 44371 | return T(fd->write(vec, amount) ? 1 : 0); 44372 | } 44373 | 44374 | case 3 : { 44375 | const vector_t vec(parameters[1]); 44376 | const std::size_t amount = 44377 | std::min(vec.size(), 44378 | static_cast<std::size_t>(scalar_t(parameters[2])())); 44379 | return T(fd->write(vec, amount) ? 1 : 0); 44380 | } 44381 | } 44382 | 44383 | return T(0); 44384 | } 44385 | }; 44386 | 44387 | template <typename T> 44388 | class read exprtk_final : public exprtk::igeneric_function<T> 44389 | { 44390 | public: 44391 | 44392 | typedef typename exprtk::igeneric_function<T> igfun_t; 44393 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44394 | typedef typename igfun_t::generic_type generic_type; 44395 | typedef typename generic_type::string_view string_t; 44396 | typedef typename generic_type::scalar_view scalar_t; 44397 | typedef typename generic_type::vector_view vector_t; 44398 | 44399 | using igfun_t::operator(); 44400 | 44401 | read() 44402 | : igfun_t("TS|TST|TV|TVT") 44403 | { details::perform_check<T>(); } 44404 | 44405 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44406 | { 44407 | details::file_descriptor* fd = details::make_handle(scalar_t(parameters[0])()); 44408 | 44409 | switch (ps_index) 44410 | { 44411 | case 0 : { 44412 | string_t buffer(parameters[1]); 44413 | const std::size_t amount = buffer.size(); 44414 | return T(fd->read(buffer,amount) ? 1 : 0); 44415 | } 44416 | 44417 | case 1 : { 44418 | string_t buffer(parameters[1]); 44419 | const std::size_t amount = 44420 | std::min(buffer.size(), 44421 | static_cast<std::size_t>(scalar_t(parameters[2])())); 44422 | return T(fd->read(buffer,amount) ? 1 : 0); 44423 | } 44424 | 44425 | case 2 : { 44426 | vector_t vec(parameters[1]); 44427 | const std::size_t amount = vec.size(); 44428 | return T(fd->read(vec,amount) ? 1 : 0); 44429 | } 44430 | 44431 | case 3 : { 44432 | vector_t vec(parameters[1]); 44433 | const std::size_t amount = 44434 | std::min(vec.size(), 44435 | static_cast<std::size_t>(scalar_t(parameters[2])())); 44436 | return T(fd->read(vec,amount) ? 1 : 0); 44437 | } 44438 | } 44439 | 44440 | return T(0); 44441 | } 44442 | }; 44443 | 44444 | template <typename T> 44445 | class getline exprtk_final : public exprtk::igeneric_function<T> 44446 | { 44447 | public: 44448 | 44449 | typedef typename exprtk::igeneric_function<T> igfun_t; 44450 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44451 | typedef typename igfun_t::generic_type generic_type; 44452 | typedef typename generic_type::string_view string_t; 44453 | typedef typename generic_type::scalar_view scalar_t; 44454 | 44455 | using igfun_t::operator(); 44456 | 44457 | getline() 44458 | : igfun_t("T",igfun_t::e_rtrn_string) 44459 | { details::perform_check<T>(); } 44460 | 44461 | inline T operator() (std::string& result, parameter_list_t parameters) exprtk_override 44462 | { 44463 | details::file_descriptor* fd = details::make_handle(scalar_t(parameters[0])()); 44464 | return T(fd->getline(result) ? 1 : 0); 44465 | } 44466 | }; 44467 | 44468 | template <typename T> 44469 | struct eof exprtk_final : public exprtk::ifunction<T> 44470 | { 44471 | using exprtk::ifunction<T>::operator(); 44472 | 44473 | eof() 44474 | : exprtk::ifunction<T>(1) 44475 | { details::perform_check<T>(); } 44476 | 44477 | inline T operator() (const T& v) exprtk_override 44478 | { 44479 | details::file_descriptor* fd = details::make_handle(v); 44480 | return (fd->eof() ? T(1) : T(0)); 44481 | } 44482 | }; 44483 | 44484 | template <typename T> 44485 | struct package 44486 | { 44487 | open <T> o; 44488 | close <T> c; 44489 | write <T> w; 44490 | read <T> r; 44491 | getline<T> g; 44492 | eof <T> e; 44493 | 44494 | bool register_package(exprtk::symbol_table<T>& symtab) 44495 | { 44496 | #define exprtk_register_function(FunctionName, FunctionType) \ 44497 | if (!symtab.add_function(FunctionName,FunctionType)) \ 44498 | { \ 44499 | exprtk_debug(( \ 44500 | "exprtk::rtl::io::file::register_package - Failed to add function: %s\n", \ 44501 | FunctionName)); \ 44502 | return false; \ 44503 | } \ 44504 | 44505 | exprtk_register_function("open" , o) 44506 | exprtk_register_function("close" , c) 44507 | exprtk_register_function("write" , w) 44508 | exprtk_register_function("read" , r) 44509 | exprtk_register_function("getline" , g) 44510 | exprtk_register_function("eof" , e) 44511 | #undef exprtk_register_function 44512 | 44513 | return true; 44514 | } 44515 | }; 44516 | 44517 | } // namespace exprtk::rtl::io::file 44518 | } // namespace exprtk::rtl::io 44519 | } // namespace exprtk::rtl 44520 | } // namespace exprtk 44521 | #endif 44522 | 44523 | #ifndef exprtk_disable_rtl_vecops 44524 | namespace exprtk 44525 | { 44526 | namespace rtl { namespace vecops { 44527 | 44528 | namespace helper 44529 | { 44530 | template <typename Vector> 44531 | inline bool invalid_range(const Vector& v, const std::size_t r0, const std::size_t r1) 44532 | { 44533 | if (r0 > (v.size() - 1)) 44534 | return true; 44535 | else if (r1 > (v.size() - 1)) 44536 | return true; 44537 | else if (r1 < r0) 44538 | return true; 44539 | else 44540 | return false; 44541 | } 44542 | 44543 | template <typename T> 44544 | struct load_vector_range 44545 | { 44546 | typedef typename exprtk::igeneric_function<T> igfun_t; 44547 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44548 | typedef typename igfun_t::generic_type generic_type; 44549 | typedef typename generic_type::scalar_view scalar_t; 44550 | typedef typename generic_type::vector_view vector_t; 44551 | 44552 | static inline bool process(parameter_list_t& parameters, 44553 | std::size_t& r0, std::size_t& r1, 44554 | const std::size_t& r0_prmidx, 44555 | const std::size_t& r1_prmidx, 44556 | const std::size_t vec_idx = 0) 44557 | { 44558 | if (r0_prmidx >= parameters.size()) 44559 | return false; 44560 | 44561 | if (r1_prmidx >= parameters.size()) 44562 | return false; 44563 | 44564 | if (!scalar_t(parameters[r0_prmidx]).to_uint(r0)) 44565 | return false; 44566 | 44567 | if (!scalar_t(parameters[r1_prmidx]).to_uint(r1)) 44568 | return false; 44569 | 44570 | return !invalid_range(vector_t(parameters[vec_idx]), r0, r1); 44571 | } 44572 | }; 44573 | } 44574 | 44575 | namespace details 44576 | { 44577 | template <typename T> 44578 | inline void kahan_sum(T& sum, T& error, const T v) 44579 | { 44580 | const T x = v - error; 44581 | const T y = sum + x; 44582 | error = (y - sum) - x; 44583 | sum = y; 44584 | } 44585 | 44586 | } // namespace exprtk::rtl::details 44587 | 44588 | template <typename T> 44589 | class all_true exprtk_final : public exprtk::igeneric_function<T> 44590 | { 44591 | public: 44592 | 44593 | typedef typename exprtk::igeneric_function<T> igfun_t; 44594 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44595 | typedef typename igfun_t::generic_type generic_type; 44596 | typedef typename generic_type::scalar_view scalar_t; 44597 | typedef typename generic_type::vector_view vector_t; 44598 | 44599 | using igfun_t::operator(); 44600 | 44601 | all_true() 44602 | : exprtk::igeneric_function<T>("V|VTT|T*") 44603 | /* 44604 | Overloads: 44605 | 0. V - vector 44606 | 1. VTT - vector, r0, r1 44607 | 2. T* - T....T 44608 | */ 44609 | {} 44610 | 44611 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44612 | { 44613 | if (2 == ps_index) 44614 | { 44615 | for (std::size_t i = 0; i < parameters.size(); ++i) 44616 | { 44617 | if (scalar_t(parameters[i])() == T(0)) 44618 | { 44619 | return T(0); 44620 | } 44621 | } 44622 | } 44623 | else 44624 | { 44625 | const vector_t vec(parameters[0]); 44626 | 44627 | std::size_t r0 = 0; 44628 | std::size_t r1 = vec.size() - 1; 44629 | 44630 | if ( 44631 | (1 == ps_index) && 44632 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 44633 | ) 44634 | { 44635 | return std::numeric_limits<T>::quiet_NaN(); 44636 | } 44637 | 44638 | for (std::size_t i = r0; i <= r1; ++i) 44639 | { 44640 | if (vec[i] == T(0)) 44641 | { 44642 | return T(0); 44643 | } 44644 | } 44645 | } 44646 | 44647 | return T(1); 44648 | } 44649 | }; 44650 | 44651 | template <typename T> 44652 | class all_false exprtk_final : public exprtk::igeneric_function<T> 44653 | { 44654 | public: 44655 | 44656 | typedef typename exprtk::igeneric_function<T> igfun_t; 44657 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44658 | typedef typename igfun_t::generic_type generic_type; 44659 | typedef typename generic_type::scalar_view scalar_t; 44660 | typedef typename generic_type::vector_view vector_t; 44661 | 44662 | using igfun_t::operator(); 44663 | 44664 | all_false() 44665 | : exprtk::igeneric_function<T>("V|VTT|T*") 44666 | /* 44667 | Overloads: 44668 | 0. V - vector 44669 | 1. VTT - vector, r0, r1 44670 | 2. T* - T....T 44671 | */ 44672 | {} 44673 | 44674 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44675 | { 44676 | if (2 == ps_index) 44677 | { 44678 | for (std::size_t i = 0; i < parameters.size(); ++i) 44679 | { 44680 | if (scalar_t(parameters[i])() != T(0)) 44681 | { 44682 | return T(0); 44683 | } 44684 | } 44685 | } 44686 | else 44687 | { 44688 | const vector_t vec(parameters[0]); 44689 | 44690 | std::size_t r0 = 0; 44691 | std::size_t r1 = vec.size() - 1; 44692 | 44693 | if ( 44694 | (1 == ps_index) && 44695 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 44696 | ) 44697 | { 44698 | return std::numeric_limits<T>::quiet_NaN(); 44699 | } 44700 | 44701 | for (std::size_t i = r0; i <= r1; ++i) 44702 | { 44703 | if (vec[i] != T(0)) 44704 | { 44705 | return T(0); 44706 | } 44707 | } 44708 | } 44709 | 44710 | return T(1); 44711 | } 44712 | }; 44713 | 44714 | template <typename T> 44715 | class any_true exprtk_final : public exprtk::igeneric_function<T> 44716 | { 44717 | public: 44718 | 44719 | typedef typename exprtk::igeneric_function<T> igfun_t; 44720 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44721 | typedef typename igfun_t::generic_type generic_type; 44722 | typedef typename generic_type::scalar_view scalar_t; 44723 | typedef typename generic_type::vector_view vector_t; 44724 | 44725 | using igfun_t::operator(); 44726 | 44727 | any_true() 44728 | : exprtk::igeneric_function<T>("V|VTT|T*") 44729 | /* 44730 | Overloads: 44731 | 0. V - vector 44732 | 1. VTT - vector, r0, r1 44733 | 2. T* - T....T 44734 | */ 44735 | {} 44736 | 44737 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44738 | { 44739 | if (2 == ps_index) 44740 | { 44741 | for (std::size_t i = 0; i < parameters.size(); ++i) 44742 | { 44743 | if (scalar_t(parameters[i])() != T(0)) 44744 | { 44745 | return T(1); 44746 | } 44747 | } 44748 | } 44749 | else 44750 | { 44751 | const vector_t vec(parameters[0]); 44752 | 44753 | std::size_t r0 = 0; 44754 | std::size_t r1 = vec.size() - 1; 44755 | 44756 | if ( 44757 | (1 == ps_index) && 44758 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 44759 | ) 44760 | { 44761 | return std::numeric_limits<T>::quiet_NaN(); 44762 | } 44763 | 44764 | for (std::size_t i = r0; i <= r1; ++i) 44765 | { 44766 | if (vec[i] != T(0)) 44767 | { 44768 | return T(1); 44769 | } 44770 | } 44771 | } 44772 | 44773 | return T(0); 44774 | } 44775 | }; 44776 | 44777 | template <typename T> 44778 | class any_false exprtk_final : public exprtk::igeneric_function<T> 44779 | { 44780 | public: 44781 | 44782 | typedef typename exprtk::igeneric_function<T> igfun_t; 44783 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44784 | typedef typename igfun_t::generic_type generic_type; 44785 | typedef typename generic_type::scalar_view scalar_t; 44786 | typedef typename generic_type::vector_view vector_t; 44787 | 44788 | using igfun_t::operator(); 44789 | 44790 | any_false() 44791 | : exprtk::igeneric_function<T>("V|VTT|T*") 44792 | /* 44793 | Overloads: 44794 | 0. V - vector 44795 | 1. VTT - vector, r0, r1 44796 | 2. T* - T....T 44797 | */ 44798 | {} 44799 | 44800 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44801 | { 44802 | if (2 == ps_index) 44803 | { 44804 | for (std::size_t i = 0; i < parameters.size(); ++i) 44805 | { 44806 | if (scalar_t(parameters[i])() == T(0)) 44807 | { 44808 | return T(1); 44809 | } 44810 | } 44811 | } 44812 | else 44813 | { 44814 | const vector_t vec(parameters[0]); 44815 | 44816 | std::size_t r0 = 0; 44817 | std::size_t r1 = vec.size() - 1; 44818 | 44819 | if ( 44820 | (1 == ps_index) && 44821 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 44822 | ) 44823 | { 44824 | return std::numeric_limits<T>::quiet_NaN(); 44825 | } 44826 | 44827 | for (std::size_t i = r0; i <= r1; ++i) 44828 | { 44829 | if (vec[i] == T(0)) 44830 | { 44831 | return T(1); 44832 | } 44833 | } 44834 | } 44835 | 44836 | return T(0); 44837 | } 44838 | }; 44839 | 44840 | template <typename T> 44841 | class count exprtk_final : public exprtk::igeneric_function<T> 44842 | { 44843 | public: 44844 | 44845 | typedef typename exprtk::igeneric_function<T> igfun_t; 44846 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44847 | typedef typename igfun_t::generic_type generic_type; 44848 | typedef typename generic_type::scalar_view scalar_t; 44849 | typedef typename generic_type::vector_view vector_t; 44850 | 44851 | using igfun_t::operator(); 44852 | 44853 | count() 44854 | : exprtk::igeneric_function<T>("V|VTT|T*") 44855 | /* 44856 | Overloads: 44857 | 0. V - vector 44858 | 1. VTT - vector, r0, r1 44859 | 2. T* - T....T 44860 | */ 44861 | {} 44862 | 44863 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44864 | { 44865 | std::size_t cnt = 0; 44866 | 44867 | if (2 == ps_index) 44868 | { 44869 | for (std::size_t i = 0; i < parameters.size(); ++i) 44870 | { 44871 | if (scalar_t(parameters[i])() != T(0)) ++cnt; 44872 | } 44873 | } 44874 | else 44875 | { 44876 | const vector_t vec(parameters[0]); 44877 | 44878 | std::size_t r0 = 0; 44879 | std::size_t r1 = vec.size() - 1; 44880 | 44881 | if ( 44882 | (1 == ps_index) && 44883 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 44884 | ) 44885 | { 44886 | return std::numeric_limits<T>::quiet_NaN(); 44887 | } 44888 | 44889 | for (std::size_t i = r0; i <= r1; ++i) 44890 | { 44891 | if (vec[i] != T(0)) ++cnt; 44892 | } 44893 | } 44894 | 44895 | return T(cnt); 44896 | } 44897 | }; 44898 | 44899 | template <typename T> 44900 | class copy exprtk_final : public exprtk::igeneric_function<T> 44901 | { 44902 | public: 44903 | 44904 | typedef typename exprtk::igeneric_function<T> igfun_t; 44905 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44906 | typedef typename igfun_t::generic_type generic_type; 44907 | typedef typename generic_type::scalar_view scalar_t; 44908 | typedef typename generic_type::vector_view vector_t; 44909 | 44910 | using igfun_t::operator(); 44911 | 44912 | copy() 44913 | : exprtk::igeneric_function<T>("VV|VTTVTT") 44914 | /* 44915 | Overloads: 44916 | 0. VV - x(vector), y(vector) 44917 | 1. VTTVTT - x(vector), xr0, xr1, y(vector), yr0, yr1, 44918 | */ 44919 | {} 44920 | 44921 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44922 | { 44923 | const vector_t x(parameters[0]); 44924 | vector_t y(parameters[(0 == ps_index) ? 1 : 3]); 44925 | 44926 | std::size_t xr0 = 0; 44927 | std::size_t xr1 = x.size() - 1; 44928 | 44929 | std::size_t yr0 = 0; 44930 | std::size_t yr1 = y.size() - 1; 44931 | 44932 | if (1 == ps_index) 44933 | { 44934 | if ( 44935 | !helper::load_vector_range<T>::process(parameters, xr0, xr1, 1, 2, 0) || 44936 | !helper::load_vector_range<T>::process(parameters, yr0, yr1, 4, 5, 3) 44937 | ) 44938 | return T(0); 44939 | } 44940 | 44941 | const std::size_t n = std::min(xr1 - xr0 + 1, yr1 - yr0 + 1); 44942 | 44943 | std::copy( 44944 | x.begin() + xr0, 44945 | x.begin() + xr0 + n, 44946 | y.begin() + yr0); 44947 | 44948 | return T(n); 44949 | } 44950 | }; 44951 | 44952 | template <typename T> 44953 | class rol exprtk_final : public exprtk::igeneric_function<T> 44954 | { 44955 | public: 44956 | 44957 | typedef typename exprtk::igeneric_function<T> igfun_t; 44958 | typedef typename igfun_t::parameter_list_t parameter_list_t; 44959 | typedef typename igfun_t::generic_type generic_type; 44960 | typedef typename generic_type::scalar_view scalar_t; 44961 | typedef typename generic_type::vector_view vector_t; 44962 | 44963 | using igfun_t::operator(); 44964 | 44965 | rol() 44966 | : exprtk::igeneric_function<T>("VT|VTTT") 44967 | /* 44968 | Overloads: 44969 | 0. VT - vector, N 44970 | 1. VTTT - vector, N, r0, r1 44971 | */ 44972 | {} 44973 | 44974 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 44975 | { 44976 | vector_t vec(parameters[0]); 44977 | 44978 | std::size_t n = 0; 44979 | std::size_t r0 = 0; 44980 | std::size_t r1 = vec.size() - 1; 44981 | 44982 | if (!scalar_t(parameters[1]).to_uint(n)) 44983 | return T(0); 44984 | 44985 | if ( 44986 | (1 == ps_index) && 44987 | !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0) 44988 | ) 44989 | return T(0); 44990 | 44991 | const std::size_t dist = r1 - r0 + 1; 44992 | const std::size_t shift = n % dist; 44993 | 44994 | std::rotate( 44995 | vec.begin() + r0, 44996 | vec.begin() + r0 + shift, 44997 | vec.begin() + r1 + 1); 44998 | 44999 | return T(1); 45000 | } 45001 | }; 45002 | 45003 | template <typename T> 45004 | class ror exprtk_final : public exprtk::igeneric_function<T> 45005 | { 45006 | public: 45007 | 45008 | typedef typename exprtk::igeneric_function<T> igfun_t; 45009 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45010 | typedef typename igfun_t::generic_type generic_type; 45011 | typedef typename generic_type::scalar_view scalar_t; 45012 | typedef typename generic_type::vector_view vector_t; 45013 | 45014 | using igfun_t::operator(); 45015 | 45016 | ror() 45017 | : exprtk::igeneric_function<T>("VT|VTTT") 45018 | /* 45019 | Overloads: 45020 | 0. VT - vector, N 45021 | 1. VTTT - vector, N, r0, r1 45022 | */ 45023 | {} 45024 | 45025 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45026 | { 45027 | vector_t vec(parameters[0]); 45028 | 45029 | std::size_t n = 0; 45030 | std::size_t r0 = 0; 45031 | std::size_t r1 = vec.size() - 1; 45032 | 45033 | if (!scalar_t(parameters[1]).to_uint(n)) 45034 | return T(0); 45035 | 45036 | if ( 45037 | (1 == ps_index) && 45038 | !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0) 45039 | ) 45040 | return T(0); 45041 | 45042 | const std::size_t dist = r1 - r0 + 1; 45043 | const std::size_t shift = (dist - (n % dist)) % dist; 45044 | 45045 | std::rotate( 45046 | vec.begin() + r0, 45047 | vec.begin() + r0 + shift, 45048 | vec.begin() + r1 + 1); 45049 | 45050 | return T(1); 45051 | } 45052 | }; 45053 | 45054 | template <typename T> 45055 | class reverse exprtk_final : public exprtk::igeneric_function<T> 45056 | { 45057 | public: 45058 | 45059 | typedef typename exprtk::igeneric_function<T> igfun_t; 45060 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45061 | typedef typename igfun_t::generic_type generic_type; 45062 | typedef typename generic_type::scalar_view scalar_t; 45063 | typedef typename generic_type::vector_view vector_t; 45064 | 45065 | using igfun_t::operator(); 45066 | 45067 | reverse() 45068 | : exprtk::igeneric_function<T>("V|VTT") 45069 | /* 45070 | Overloads: 45071 | 0. V - vector 45072 | 1. VTT - vector, r0, r1 45073 | */ 45074 | {} 45075 | 45076 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45077 | { 45078 | vector_t vec(parameters[0]); 45079 | 45080 | std::size_t r0 = 0; 45081 | std::size_t r1 = vec.size() - 1; 45082 | 45083 | if ( 45084 | (1 == ps_index) && 45085 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 45086 | ) 45087 | return T(0); 45088 | 45089 | std::reverse(vec.begin() + r0, vec.begin() + r1 + 1); 45090 | 45091 | return T(1); 45092 | } 45093 | }; 45094 | 45095 | template <typename T> 45096 | class shift_left exprtk_final : public exprtk::igeneric_function<T> 45097 | { 45098 | public: 45099 | 45100 | typedef typename exprtk::igeneric_function<T> igfun_t; 45101 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45102 | typedef typename igfun_t::generic_type generic_type; 45103 | typedef typename generic_type::scalar_view scalar_t; 45104 | typedef typename generic_type::vector_view vector_t; 45105 | 45106 | using igfun_t::operator(); 45107 | 45108 | shift_left() 45109 | : exprtk::igeneric_function<T>("VT|VTTT") 45110 | /* 45111 | Overloads: 45112 | 0. VT - vector, N 45113 | 1. VTTT - vector, N, r0, r1 45114 | */ 45115 | {} 45116 | 45117 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45118 | { 45119 | vector_t vec(parameters[0]); 45120 | 45121 | std::size_t n = 0; 45122 | std::size_t r0 = 0; 45123 | std::size_t r1 = vec.size() - 1; 45124 | 45125 | if (!scalar_t(parameters[1]).to_uint(n)) 45126 | return T(0); 45127 | 45128 | if ( 45129 | (1 == ps_index) && 45130 | !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0) 45131 | ) 45132 | return T(0); 45133 | 45134 | const std::size_t dist = r1 - r0 + 1; 45135 | 45136 | if (n > dist) 45137 | return T(0); 45138 | 45139 | std::rotate( 45140 | vec.begin() + r0, 45141 | vec.begin() + r0 + n, 45142 | vec.begin() + r1 + 1); 45143 | 45144 | for (std::size_t i = r1 - n + 1ULL; i <= r1; ++i) 45145 | { 45146 | vec[i] = T(0); 45147 | } 45148 | 45149 | return T(1); 45150 | } 45151 | }; 45152 | 45153 | template <typename T> 45154 | class shift_right exprtk_final : public exprtk::igeneric_function<T> 45155 | { 45156 | public: 45157 | 45158 | typedef typename exprtk::igeneric_function<T> igfun_t; 45159 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45160 | typedef typename igfun_t::generic_type generic_type; 45161 | typedef typename generic_type::scalar_view scalar_t; 45162 | typedef typename generic_type::vector_view vector_t; 45163 | 45164 | using igfun_t::operator(); 45165 | 45166 | shift_right() 45167 | : exprtk::igeneric_function<T>("VT|VTTT") 45168 | /* 45169 | Overloads: 45170 | 0. VT - vector, N 45171 | 1. VTTT - vector, N, r0, r1 45172 | */ 45173 | {} 45174 | 45175 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45176 | { 45177 | vector_t vec(parameters[0]); 45178 | 45179 | std::size_t n = 0; 45180 | std::size_t r0 = 0; 45181 | std::size_t r1 = vec.size() - 1; 45182 | 45183 | if (!scalar_t(parameters[1]).to_uint(n)) 45184 | return T(0); 45185 | 45186 | if ( 45187 | (1 == ps_index) && 45188 | !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0) 45189 | ) 45190 | return T(0); 45191 | 45192 | const std::size_t dist = r1 - r0 + 1; 45193 | 45194 | if (n > dist) 45195 | return T(0); 45196 | 45197 | const std::size_t shift = (dist - (n % dist)) % dist; 45198 | 45199 | std::rotate( 45200 | vec.begin() + r0, 45201 | vec.begin() + r0 + shift, 45202 | vec.begin() + r1 + 1); 45203 | 45204 | for (std::size_t i = r0; i < r0 + n; ++i) 45205 | { 45206 | vec[i] = T(0); 45207 | } 45208 | 45209 | return T(1); 45210 | } 45211 | }; 45212 | 45213 | template <typename T> 45214 | class sort exprtk_final : public exprtk::igeneric_function<T> 45215 | { 45216 | public: 45217 | 45218 | typedef typename exprtk::igeneric_function<T> igfun_t; 45219 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45220 | typedef typename igfun_t::generic_type generic_type; 45221 | typedef typename generic_type::string_view string_t; 45222 | typedef typename generic_type::vector_view vector_t; 45223 | 45224 | using igfun_t::operator(); 45225 | 45226 | sort() 45227 | : exprtk::igeneric_function<T>("V|VTT|VS|VSTT") 45228 | /* 45229 | Overloads: 45230 | 0. V - vector 45231 | 1. VTT - vector, r0, r1 45232 | 2. VS - vector, string 45233 | 3. VSTT - vector, string, r0, r1 45234 | */ 45235 | {} 45236 | 45237 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45238 | { 45239 | vector_t vec(parameters[0]); 45240 | 45241 | std::size_t r0 = 0; 45242 | std::size_t r1 = vec.size() - 1; 45243 | 45244 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0)) 45245 | return T(0); 45246 | if ((3 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0)) 45247 | return T(0); 45248 | 45249 | bool ascending = true; 45250 | 45251 | if ((2 == ps_index) || (3 == ps_index)) 45252 | { 45253 | if (exprtk::details::imatch(to_str(string_t(parameters[1])),"ascending")) 45254 | ascending = true; 45255 | else if (exprtk::details::imatch(to_str(string_t(parameters[1])),"descending")) 45256 | ascending = false; 45257 | else 45258 | return T(0); 45259 | } 45260 | 45261 | if (ascending) 45262 | std::sort( 45263 | vec.begin() + r0, 45264 | vec.begin() + r1 + 1, 45265 | std::less<T>()); 45266 | else 45267 | std::sort( 45268 | vec.begin() + r0, 45269 | vec.begin() + r1 + 1, 45270 | std::greater<T>()); 45271 | 45272 | return T(1); 45273 | } 45274 | }; 45275 | 45276 | template <typename T> 45277 | class nthelement exprtk_final : public exprtk::igeneric_function<T> 45278 | { 45279 | public: 45280 | 45281 | typedef typename exprtk::igeneric_function<T> igfun_t; 45282 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45283 | typedef typename igfun_t::generic_type generic_type; 45284 | typedef typename generic_type::scalar_view scalar_t; 45285 | typedef typename generic_type::vector_view vector_t; 45286 | 45287 | using igfun_t::operator(); 45288 | 45289 | nthelement() 45290 | : exprtk::igeneric_function<T>("VT|VTTT") 45291 | /* 45292 | Overloads: 45293 | 0. VT - vector, nth-element 45294 | 1. VTTT - vector, nth-element, r0, r1 45295 | */ 45296 | {} 45297 | 45298 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45299 | { 45300 | vector_t vec(parameters[0]); 45301 | 45302 | std::size_t n = 0; 45303 | std::size_t r0 = 0; 45304 | std::size_t r1 = vec.size() - 1; 45305 | 45306 | if (!scalar_t(parameters[1]).to_uint(n)) 45307 | return T(0); 45308 | 45309 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0)) 45310 | { 45311 | return std::numeric_limits<T>::quiet_NaN(); 45312 | } 45313 | 45314 | std::nth_element( 45315 | vec.begin() + r0, 45316 | vec.begin() + r0 + n , 45317 | vec.begin() + r1 + 1); 45318 | 45319 | return T(1); 45320 | } 45321 | }; 45322 | 45323 | template <typename T> 45324 | class assign exprtk_final : public exprtk::igeneric_function<T> 45325 | { 45326 | public: 45327 | 45328 | typedef typename exprtk::igeneric_function<T> igfun_t; 45329 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45330 | typedef typename igfun_t::generic_type generic_type; 45331 | typedef typename generic_type::scalar_view scalar_t; 45332 | typedef typename generic_type::vector_view vector_t; 45333 | 45334 | using igfun_t::operator(); 45335 | 45336 | assign() 45337 | : exprtk::igeneric_function<T>("VT|VTTT|VTTTT") 45338 | /* 45339 | Overloads: 45340 | 0. VT - vector, V 45341 | 1. VTTT - vector, V, r0, r1 45342 | 2. VTTTT - vector, V, r0, r1, SS 45343 | */ 45344 | {} 45345 | 45346 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45347 | { 45348 | vector_t vec(parameters[0]); 45349 | 45350 | const T assign_value = scalar_t(parameters[1]); 45351 | 45352 | const std::size_t step_size = (2 != ps_index) ? 1 : 45353 | static_cast<std::size_t>(scalar_t(parameters.back())()); 45354 | 45355 | std::size_t r0 = 0; 45356 | std::size_t r1 = vec.size() - 1; 45357 | 45358 | if ( 45359 | ((ps_index == 1) || (ps_index == 2)) && 45360 | !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0) 45361 | ) 45362 | { 45363 | return T(0); 45364 | } 45365 | 45366 | for (std::size_t i = r0; i <= r1; i += step_size) 45367 | { 45368 | vec[i] = assign_value; 45369 | } 45370 | 45371 | return T(1); 45372 | } 45373 | }; 45374 | 45375 | template <typename T> 45376 | class iota exprtk_final : public exprtk::igeneric_function<T> 45377 | { 45378 | public: 45379 | 45380 | typedef typename exprtk::igeneric_function<T> igfun_t; 45381 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45382 | typedef typename igfun_t::generic_type generic_type; 45383 | typedef typename generic_type::scalar_view scalar_t; 45384 | typedef typename generic_type::vector_view vector_t; 45385 | 45386 | using igfun_t::operator(); 45387 | 45388 | iota() 45389 | : exprtk::igeneric_function<T>("VTT|VT|VTTTT|VTTT") 45390 | /* 45391 | Overloads: 45392 | 0. VTT - vector, SV, SS 45393 | 1. VT - vector, SV, SS (+1) 45394 | 2. VTTT - vector, r0, r1, SV, SS 45395 | 3. VTT - vector, r0, r1, SV, SS (+1) 45396 | 45397 | Where: 45398 | 1. SV - Start value 45399 | 2. SS - Step size 45400 | */ 45401 | {} 45402 | 45403 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45404 | { 45405 | vector_t vec(parameters[0]); 45406 | 45407 | const T start_value = (ps_index <= 1) ? 45408 | scalar_t(parameters[1]) : 45409 | scalar_t(parameters[3]) ; 45410 | 45411 | const T step_size = ((0 == ps_index) || (2 == ps_index)) ? 45412 | scalar_t(parameters.back())() : 45413 | T(1) ; 45414 | 45415 | std::size_t r0 = 0; 45416 | std::size_t r1 = vec.size() - 1; 45417 | 45418 | if ( 45419 | ((ps_index == 2) || (ps_index == 3)) && 45420 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 45421 | ) 45422 | { 45423 | return T(0); 45424 | } 45425 | 45426 | for (std::size_t i = r0; i <= r1; ++i) 45427 | { 45428 | vec[i] = start_value + ((i - r0) * step_size); 45429 | } 45430 | 45431 | return T(1); 45432 | } 45433 | }; 45434 | 45435 | template <typename T> 45436 | class sumk exprtk_final : public exprtk::igeneric_function<T> 45437 | { 45438 | public: 45439 | 45440 | typedef typename exprtk::igeneric_function<T> igfun_t; 45441 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45442 | typedef typename igfun_t::generic_type generic_type; 45443 | typedef typename generic_type::scalar_view scalar_t; 45444 | typedef typename generic_type::vector_view vector_t; 45445 | 45446 | using igfun_t::operator(); 45447 | 45448 | sumk() 45449 | : exprtk::igeneric_function<T>("V|VTT|VTTT") 45450 | /* 45451 | Overloads: 45452 | 0. V - vector 45453 | 1. VTT - vector, r0, r1 45454 | 2. VTTT - vector, r0, r1, stride 45455 | */ 45456 | {} 45457 | 45458 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45459 | { 45460 | const vector_t vec(parameters[0]); 45461 | 45462 | const std::size_t stride = (2 != ps_index) ? 1 : 45463 | static_cast<std::size_t>(scalar_t(parameters[3])()); 45464 | 45465 | std::size_t r0 = 0; 45466 | std::size_t r1 = vec.size() - 1; 45467 | 45468 | if ( 45469 | ((1 == ps_index) || (2 == ps_index)) && 45470 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 45471 | ) 45472 | { 45473 | return std::numeric_limits<T>::quiet_NaN(); 45474 | } 45475 | 45476 | T result = T(0); 45477 | T error = T(0); 45478 | 45479 | for (std::size_t i = r0; i <= r1; i += stride) 45480 | { 45481 | details::kahan_sum(result, error, vec[i]); 45482 | } 45483 | 45484 | return result; 45485 | } 45486 | }; 45487 | 45488 | template <typename T> 45489 | class axpy exprtk_final : public exprtk::igeneric_function<T> 45490 | { 45491 | public: 45492 | 45493 | typedef typename exprtk::igeneric_function<T> igfun_t; 45494 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45495 | typedef typename igfun_t::generic_type generic_type; 45496 | typedef typename generic_type::scalar_view scalar_t; 45497 | typedef typename generic_type::vector_view vector_t; 45498 | 45499 | using igfun_t::operator(); 45500 | 45501 | axpy() 45502 | : exprtk::igeneric_function<T>("TVV|TVVTT") 45503 | /* 45504 | y <- ax + y 45505 | Overloads: 45506 | 0. TVV - a, x(vector), y(vector) 45507 | 1. TVVTT - a, x(vector), y(vector), r0, r1 45508 | */ 45509 | {} 45510 | 45511 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45512 | { 45513 | const vector_t x(parameters[1]); 45514 | vector_t y(parameters[2]); 45515 | 45516 | std::size_t r0 = 0; 45517 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45518 | 45519 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 3, 4, 1)) 45520 | return std::numeric_limits<T>::quiet_NaN(); 45521 | else if (helper::invalid_range(y, r0, r1)) 45522 | return std::numeric_limits<T>::quiet_NaN(); 45523 | 45524 | const T a = scalar_t(parameters[0])(); 45525 | 45526 | for (std::size_t i = r0; i <= r1; ++i) 45527 | { 45528 | y[i] = (a * x[i]) + y[i]; 45529 | } 45530 | 45531 | return T(1); 45532 | } 45533 | }; 45534 | 45535 | template <typename T> 45536 | class axpby exprtk_final : public exprtk::igeneric_function<T> 45537 | { 45538 | public: 45539 | 45540 | typedef typename exprtk::igeneric_function<T> igfun_t; 45541 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45542 | typedef typename igfun_t::generic_type generic_type; 45543 | typedef typename generic_type::scalar_view scalar_t; 45544 | typedef typename generic_type::vector_view vector_t; 45545 | 45546 | using igfun_t::operator(); 45547 | 45548 | axpby() 45549 | : exprtk::igeneric_function<T>("TVTV|TVTVTT") 45550 | /* 45551 | y <- ax + by 45552 | Overloads: 45553 | 0. TVTV - a, x(vector), b, y(vector) 45554 | 1. TVTVTT - a, x(vector), b, y(vector), r0, r1 45555 | */ 45556 | {} 45557 | 45558 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45559 | { 45560 | const vector_t x(parameters[1]); 45561 | vector_t y(parameters[3]); 45562 | 45563 | std::size_t r0 = 0; 45564 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45565 | 45566 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 4, 5, 1)) 45567 | return std::numeric_limits<T>::quiet_NaN(); 45568 | else if (helper::invalid_range(y, r0, r1)) 45569 | return std::numeric_limits<T>::quiet_NaN(); 45570 | 45571 | const T a = scalar_t(parameters[0])(); 45572 | const T b = scalar_t(parameters[2])(); 45573 | 45574 | for (std::size_t i = r0; i <= r1; ++i) 45575 | { 45576 | y[i] = (a * x[i]) + (b * y[i]); 45577 | } 45578 | 45579 | return T(1); 45580 | } 45581 | }; 45582 | 45583 | template <typename T> 45584 | class axpyz exprtk_final : public exprtk::igeneric_function<T> 45585 | { 45586 | public: 45587 | 45588 | typedef typename exprtk::igeneric_function<T> igfun_t; 45589 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45590 | typedef typename igfun_t::generic_type generic_type; 45591 | typedef typename generic_type::scalar_view scalar_t; 45592 | typedef typename generic_type::vector_view vector_t; 45593 | 45594 | using igfun_t::operator(); 45595 | 45596 | axpyz() 45597 | : exprtk::igeneric_function<T>("TVVV|TVVVTT") 45598 | /* 45599 | z <- ax + y 45600 | Overloads: 45601 | 0. TVVV - a, x(vector), y(vector), z(vector) 45602 | 1. TVVVTT - a, x(vector), y(vector), z(vector), r0, r1 45603 | */ 45604 | {} 45605 | 45606 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45607 | { 45608 | const vector_t x(parameters[1]); 45609 | const vector_t y(parameters[2]); 45610 | vector_t z(parameters[3]); 45611 | 45612 | std::size_t r0 = 0; 45613 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45614 | 45615 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 4, 5, 1)) 45616 | return std::numeric_limits<T>::quiet_NaN(); 45617 | else if (helper::invalid_range(y, r0, r1)) 45618 | return std::numeric_limits<T>::quiet_NaN(); 45619 | else if (helper::invalid_range(z, r0, r1)) 45620 | return std::numeric_limits<T>::quiet_NaN(); 45621 | 45622 | const T a = scalar_t(parameters[0])(); 45623 | 45624 | for (std::size_t i = r0; i <= r1; ++i) 45625 | { 45626 | z[i] = (a * x[i]) + y[i]; 45627 | } 45628 | 45629 | return T(1); 45630 | } 45631 | }; 45632 | 45633 | template <typename T> 45634 | class axpbyz exprtk_final : public exprtk::igeneric_function<T> 45635 | { 45636 | public: 45637 | 45638 | typedef typename exprtk::igeneric_function<T> igfun_t; 45639 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45640 | typedef typename igfun_t::generic_type generic_type; 45641 | typedef typename generic_type::scalar_view scalar_t; 45642 | typedef typename generic_type::vector_view vector_t; 45643 | 45644 | using igfun_t::operator(); 45645 | 45646 | axpbyz() 45647 | : exprtk::igeneric_function<T>("TVTVV|TVTVVTT") 45648 | /* 45649 | z <- ax + by 45650 | Overloads: 45651 | 0. TVTVV - a, x(vector), b, y(vector), z(vector) 45652 | 1. TVTVVTT - a, x(vector), b, y(vector), z(vector), r0, r1 45653 | */ 45654 | {} 45655 | 45656 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45657 | { 45658 | const vector_t x(parameters[1]); 45659 | const vector_t y(parameters[3]); 45660 | vector_t z(parameters[4]); 45661 | 45662 | std::size_t r0 = 0; 45663 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45664 | 45665 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 5, 6, 1)) 45666 | return std::numeric_limits<T>::quiet_NaN(); 45667 | else if (helper::invalid_range(y, r0, r1)) 45668 | return std::numeric_limits<T>::quiet_NaN(); 45669 | else if (helper::invalid_range(z, r0, r1)) 45670 | return std::numeric_limits<T>::quiet_NaN(); 45671 | 45672 | const T a = scalar_t(parameters[0])(); 45673 | const T b = scalar_t(parameters[2])(); 45674 | 45675 | for (std::size_t i = r0; i <= r1; ++i) 45676 | { 45677 | z[i] = (a * x[i]) + (b * y[i]); 45678 | } 45679 | 45680 | return T(1); 45681 | } 45682 | }; 45683 | 45684 | template <typename T> 45685 | class axpbsy exprtk_final : public exprtk::igeneric_function<T> 45686 | { 45687 | public: 45688 | 45689 | typedef typename exprtk::igeneric_function<T> igfun_t; 45690 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45691 | typedef typename igfun_t::generic_type generic_type; 45692 | typedef typename generic_type::scalar_view scalar_t; 45693 | typedef typename generic_type::vector_view vector_t; 45694 | 45695 | using igfun_t::operator(); 45696 | 45697 | axpbsy() 45698 | : exprtk::igeneric_function<T>("TVTTV|TVTTVTT") 45699 | /* 45700 | y <- ax + by 45701 | Overloads: 45702 | 0. TVTVV - a, x(vector), b, shift, y(vector), z(vector) 45703 | 1. TVTVVTT - a, x(vector), b, shift, y(vector), z(vector), r0, r1 45704 | */ 45705 | {} 45706 | 45707 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45708 | { 45709 | const vector_t x(parameters[1]); 45710 | vector_t y(parameters[4]); 45711 | 45712 | std::size_t r0 = 0; 45713 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45714 | 45715 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 5, 6, 1)) 45716 | return std::numeric_limits<T>::quiet_NaN(); 45717 | else if (helper::invalid_range(y, r0, r1)) 45718 | return std::numeric_limits<T>::quiet_NaN(); 45719 | 45720 | const T a = scalar_t(parameters[0])(); 45721 | const T b = scalar_t(parameters[2])(); 45722 | 45723 | const std::size_t s = static_cast<std::size_t>(scalar_t(parameters[3])()); 45724 | 45725 | for (std::size_t i = r0; i <= r1; ++i) 45726 | { 45727 | y[i] = (a * x[i]) + (b * y[i + s]); 45728 | } 45729 | 45730 | return T(1); 45731 | } 45732 | }; 45733 | 45734 | template <typename T> 45735 | class axpbsyz exprtk_final : public exprtk::igeneric_function<T> 45736 | { 45737 | public: 45738 | 45739 | typedef typename exprtk::igeneric_function<T> igfun_t; 45740 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45741 | typedef typename igfun_t::generic_type generic_type; 45742 | typedef typename generic_type::scalar_view scalar_t; 45743 | typedef typename generic_type::vector_view vector_t; 45744 | 45745 | using igfun_t::operator(); 45746 | 45747 | axpbsyz() 45748 | : exprtk::igeneric_function<T>("TVTTVV|TVTTVVTT") 45749 | /* 45750 | z <- ax + by 45751 | Overloads: 45752 | 0. TVTVV - a, x(vector), b, shift, y(vector), z(vector) 45753 | 1. TVTVVTT - a, x(vector), b, shift, y(vector), z(vector), r0, r1 45754 | */ 45755 | {} 45756 | 45757 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45758 | { 45759 | const vector_t x(parameters[1]); 45760 | const vector_t y(parameters[4]); 45761 | vector_t z(parameters[5]); 45762 | 45763 | std::size_t r0 = 0; 45764 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45765 | 45766 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 6, 7, 1)) 45767 | return std::numeric_limits<T>::quiet_NaN(); 45768 | else if (helper::invalid_range(y, r0, r1)) 45769 | return std::numeric_limits<T>::quiet_NaN(); 45770 | else if (helper::invalid_range(z, r0, r1)) 45771 | return std::numeric_limits<T>::quiet_NaN(); 45772 | 45773 | const T a = scalar_t(parameters[0])(); 45774 | const T b = scalar_t(parameters[2])(); 45775 | 45776 | const std::size_t s = static_cast<std::size_t>(scalar_t(parameters[3])()); 45777 | 45778 | for (std::size_t i = r0; i <= r1; ++i) 45779 | { 45780 | z[i] = (a * x[i]) + (b * y[i + s]); 45781 | } 45782 | 45783 | return T(1); 45784 | } 45785 | }; 45786 | 45787 | template <typename T> 45788 | class axpbz exprtk_final : public exprtk::igeneric_function<T> 45789 | { 45790 | public: 45791 | 45792 | typedef typename exprtk::igeneric_function<T> igfun_t; 45793 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45794 | typedef typename igfun_t::generic_type generic_type; 45795 | typedef typename generic_type::scalar_view scalar_t; 45796 | typedef typename generic_type::vector_view vector_t; 45797 | 45798 | using igfun_t::operator(); 45799 | 45800 | axpbz() 45801 | : exprtk::igeneric_function<T>("TVTV|TVTVTT") 45802 | /* 45803 | z <- ax + b 45804 | Overloads: 45805 | 0. TVTV - a, x(vector), b, z(vector) 45806 | 1. TVTVTT - a, x(vector), b, z(vector), r0, r1 45807 | */ 45808 | {} 45809 | 45810 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45811 | { 45812 | const vector_t x(parameters[1]); 45813 | vector_t z(parameters[3]); 45814 | 45815 | std::size_t r0 = 0; 45816 | std::size_t r1 = x.size() - 1; 45817 | 45818 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 4, 5, 1)) 45819 | return std::numeric_limits<T>::quiet_NaN(); 45820 | else if (helper::invalid_range(z, r0, r1)) 45821 | return std::numeric_limits<T>::quiet_NaN(); 45822 | 45823 | const T a = scalar_t(parameters[0])(); 45824 | const T b = scalar_t(parameters[2])(); 45825 | 45826 | for (std::size_t i = r0; i <= r1; ++i) 45827 | { 45828 | z[i] = (a * x[i]) + b; 45829 | } 45830 | 45831 | return T(1); 45832 | } 45833 | }; 45834 | 45835 | template <typename T> 45836 | class diff exprtk_final : public exprtk::igeneric_function<T> 45837 | { 45838 | public: 45839 | 45840 | typedef typename exprtk::igeneric_function<T> igfun_t; 45841 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45842 | typedef typename igfun_t::generic_type generic_type; 45843 | typedef typename generic_type::scalar_view scalar_t; 45844 | typedef typename generic_type::vector_view vector_t; 45845 | 45846 | using igfun_t::operator(); 45847 | 45848 | diff() 45849 | : exprtk::igeneric_function<T>("VV|VVT") 45850 | /* 45851 | x_(i - stride) - x_i 45852 | Overloads: 45853 | 0. VV - x(vector), y(vector) 45854 | 1. VVT - x(vector), y(vector), stride 45855 | */ 45856 | {} 45857 | 45858 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45859 | { 45860 | const vector_t x(parameters[0]); 45861 | vector_t y(parameters[1]); 45862 | 45863 | const std::size_t r0 = 0; 45864 | const std::size_t r1 = std::min(x.size(),y.size()) - 1; 45865 | 45866 | const std::size_t stride = (1 != ps_index) ? 1 : 45867 | std::min(r1,static_cast<std::size_t>(scalar_t(parameters[2])())); 45868 | 45869 | for (std::size_t i = 0; i < stride; ++i) 45870 | { 45871 | y[i] = std::numeric_limits<T>::quiet_NaN(); 45872 | } 45873 | 45874 | for (std::size_t i = (r0 + stride); i <= r1; ++i) 45875 | { 45876 | y[i] = x[i] - x[i - stride]; 45877 | } 45878 | 45879 | return T(1); 45880 | } 45881 | }; 45882 | 45883 | template <typename T> 45884 | class dot exprtk_final : public exprtk::igeneric_function<T> 45885 | { 45886 | public: 45887 | 45888 | typedef typename exprtk::igeneric_function<T> igfun_t; 45889 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45890 | typedef typename igfun_t::generic_type generic_type; 45891 | typedef typename generic_type::scalar_view scalar_t; 45892 | typedef typename generic_type::vector_view vector_t; 45893 | 45894 | using igfun_t::operator(); 45895 | 45896 | dot() 45897 | : exprtk::igeneric_function<T>("VV|VVTT") 45898 | /* 45899 | Overloads: 45900 | 0. VV - x(vector), y(vector) 45901 | 1. VVTT - x(vector), y(vector), r0, r1 45902 | */ 45903 | {} 45904 | 45905 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45906 | { 45907 | const vector_t x(parameters[0]); 45908 | const vector_t y(parameters[1]); 45909 | 45910 | std::size_t r0 = 0; 45911 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45912 | 45913 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0)) 45914 | return std::numeric_limits<T>::quiet_NaN(); 45915 | else if (helper::invalid_range(y, r0, r1)) 45916 | return std::numeric_limits<T>::quiet_NaN(); 45917 | 45918 | T result = T(0); 45919 | 45920 | for (std::size_t i = r0; i <= r1; ++i) 45921 | { 45922 | result += (x[i] * y[i]); 45923 | } 45924 | 45925 | return result; 45926 | } 45927 | }; 45928 | 45929 | template <typename T> 45930 | class dotk exprtk_final : public exprtk::igeneric_function<T> 45931 | { 45932 | public: 45933 | 45934 | typedef typename exprtk::igeneric_function<T> igfun_t; 45935 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45936 | typedef typename igfun_t::generic_type generic_type; 45937 | typedef typename generic_type::scalar_view scalar_t; 45938 | typedef typename generic_type::vector_view vector_t; 45939 | 45940 | using igfun_t::operator(); 45941 | 45942 | dotk() 45943 | : exprtk::igeneric_function<T>("VV|VVTT") 45944 | /* 45945 | Overloads: 45946 | 0. VV - x(vector), y(vector) 45947 | 1. VVTT - x(vector), y(vector), r0, r1 45948 | */ 45949 | {} 45950 | 45951 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 45952 | { 45953 | const vector_t x(parameters[0]); 45954 | const vector_t y(parameters[1]); 45955 | 45956 | std::size_t r0 = 0; 45957 | std::size_t r1 = std::min(x.size(),y.size()) - 1; 45958 | 45959 | if ((1 == ps_index) && !helper::load_vector_range<T>::process(parameters, r0, r1, 2, 3, 0)) 45960 | return std::numeric_limits<T>::quiet_NaN(); 45961 | else if (helper::invalid_range(y, r0, r1)) 45962 | return std::numeric_limits<T>::quiet_NaN(); 45963 | 45964 | T result = T(0); 45965 | T error = T(0); 45966 | 45967 | for (std::size_t i = r0; i <= r1; ++i) 45968 | { 45969 | details::kahan_sum(result, error, (x[i] * y[i])); 45970 | } 45971 | 45972 | return result; 45973 | } 45974 | }; 45975 | 45976 | template <typename T> 45977 | class threshold_below exprtk_final : public exprtk::igeneric_function<T> 45978 | { 45979 | public: 45980 | 45981 | typedef typename exprtk::igeneric_function<T> igfun_t; 45982 | typedef typename igfun_t::parameter_list_t parameter_list_t; 45983 | typedef typename igfun_t::generic_type generic_type; 45984 | typedef typename generic_type::scalar_view scalar_t; 45985 | typedef typename generic_type::vector_view vector_t; 45986 | 45987 | using igfun_t::operator(); 45988 | 45989 | threshold_below() 45990 | : exprtk::igeneric_function<T>("VTT|VTTTT") 45991 | /* 45992 | Overloads: 45993 | 0. VTT - vector, TV, SV 45994 | 1. VTTTT - vector, r0, r1, TV, SV 45995 | 45996 | Where: 45997 | TV - Threshold value 45998 | SV - Snap-to value 45999 | */ 46000 | {} 46001 | 46002 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 46003 | { 46004 | vector_t vec(parameters[0]); 46005 | 46006 | const T threshold_value = (0 == ps_index) ? 46007 | scalar_t(parameters[1]) : 46008 | scalar_t(parameters[3]) ; 46009 | 46010 | const T snap_value = scalar_t(parameters.back()); 46011 | 46012 | std::size_t r0 = 0; 46013 | std::size_t r1 = vec.size() - 1; 46014 | 46015 | if ( 46016 | (1 == ps_index) && 46017 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 46018 | ) 46019 | { 46020 | return T(0); 46021 | } 46022 | 46023 | for (std::size_t i = r0; i <= r1; ++i) 46024 | { 46025 | if (vec[i] < threshold_value) 46026 | { 46027 | vec[i] = snap_value; 46028 | } 46029 | } 46030 | 46031 | return T(1); 46032 | } 46033 | }; 46034 | 46035 | template <typename T> 46036 | class threshold_above exprtk_final : public exprtk::igeneric_function<T> 46037 | { 46038 | public: 46039 | 46040 | typedef typename exprtk::igeneric_function<T> igfun_t; 46041 | typedef typename igfun_t::parameter_list_t parameter_list_t; 46042 | typedef typename igfun_t::generic_type generic_type; 46043 | typedef typename generic_type::scalar_view scalar_t; 46044 | typedef typename generic_type::vector_view vector_t; 46045 | 46046 | using igfun_t::operator(); 46047 | 46048 | threshold_above() 46049 | : exprtk::igeneric_function<T>("VTT|VTTTT") 46050 | /* 46051 | Overloads: 46052 | 0. VTT - vector, TV, SV 46053 | 1. VTTTT - vector, r0, r1, TV, SV 46054 | 46055 | Where: 46056 | TV - Threshold value 46057 | SV - Snap-to value 46058 | */ 46059 | {} 46060 | 46061 | inline T operator() (const std::size_t& ps_index, parameter_list_t parameters) exprtk_override 46062 | { 46063 | vector_t vec(parameters[0]); 46064 | 46065 | const T threshold_value = (0 == ps_index) ? 46066 | scalar_t(parameters[1]) : 46067 | scalar_t(parameters[3]) ; 46068 | 46069 | const T snap_value = scalar_t(parameters.back()); 46070 | 46071 | std::size_t r0 = 0; 46072 | std::size_t r1 = vec.size() - 1; 46073 | 46074 | if ( 46075 | (1 == ps_index) && 46076 | !helper::load_vector_range<T>::process(parameters, r0, r1, 1, 2, 0) 46077 | ) 46078 | { 46079 | return T(0); 46080 | } 46081 | 46082 | for (std::size_t i = r0; i <= r1; ++i) 46083 | { 46084 | if (vec[i] > threshold_value) 46085 | { 46086 | vec[i] = snap_value; 46087 | } 46088 | } 46089 | 46090 | return T(1); 46091 | } 46092 | }; 46093 | 46094 | template <typename T> 46095 | struct package 46096 | { 46097 | all_true <T> at; 46098 | all_false <T> af; 46099 | any_true <T> nt; 46100 | any_false <T> nf; 46101 | count <T> c; 46102 | copy <T> cp; 46103 | rol <T> rl; 46104 | ror <T> rr; 46105 | reverse <T> rev; 46106 | shift_left <T> sl; 46107 | shift_right <T> sr; 46108 | sort <T> st; 46109 | nthelement <T> ne; 46110 | assign <T> an; 46111 | iota <T> ia; 46112 | sumk <T> sk; 46113 | axpy <T> b1_axpy; 46114 | axpby <T> b1_axpby; 46115 | axpyz <T> b1_axpyz; 46116 | axpbyz <T> b1_axpbyz; 46117 | axpbsy <T> b1_axpbsy; 46118 | axpbsyz <T> b1_axpbsyz; 46119 | axpbz <T> b1_axpbz; 46120 | diff <T> df; 46121 | dot <T> dt; 46122 | dotk <T> dtk; 46123 | threshold_above<T> ta; 46124 | threshold_below<T> tb; 46125 | 46126 | bool register_package(exprtk::symbol_table<T>& symtab) 46127 | { 46128 | #define exprtk_register_function(FunctionName, FunctionType) \ 46129 | if (!symtab.add_function(FunctionName,FunctionType)) \ 46130 | { \ 46131 | exprtk_debug(( \ 46132 | "exprtk::rtl::vecops::register_package - Failed to add function: %s\n", \ 46133 | FunctionName)); \ 46134 | return false; \ 46135 | } \ 46136 | 46137 | exprtk_register_function("all_true" , at ) 46138 | exprtk_register_function("all_false" , af ) 46139 | exprtk_register_function("any_true" , nt ) 46140 | exprtk_register_function("any_false" , nf ) 46141 | exprtk_register_function("count" , c ) 46142 | exprtk_register_function("copy" , cp ) 46143 | exprtk_register_function("rotate_left" , rl ) 46144 | exprtk_register_function("rol" , rl ) 46145 | exprtk_register_function("rotate_right" , rr ) 46146 | exprtk_register_function("ror" , rr ) 46147 | exprtk_register_function("reverse" , rev ) 46148 | exprtk_register_function("shftl" , sl ) 46149 | exprtk_register_function("shftr" , sr ) 46150 | exprtk_register_function("sort" , st ) 46151 | exprtk_register_function("nth_element" , ne ) 46152 | exprtk_register_function("assign" , an ) 46153 | exprtk_register_function("iota" , ia ) 46154 | exprtk_register_function("sumk" , sk ) 46155 | exprtk_register_function("axpy" , b1_axpy ) 46156 | exprtk_register_function("axpby" , b1_axpby ) 46157 | exprtk_register_function("axpyz" , b1_axpyz ) 46158 | exprtk_register_function("axpbyz" , b1_axpbyz ) 46159 | exprtk_register_function("axpbsy" , b1_axpbsy ) 46160 | exprtk_register_function("axpbsyz" , b1_axpbsyz) 46161 | exprtk_register_function("axpbz" , b1_axpbz ) 46162 | exprtk_register_function("diff" , df ) 46163 | exprtk_register_function("dot" , dt ) 46164 | exprtk_register_function("dotk" , dtk ) 46165 | exprtk_register_function("threshold_above" , ta ) 46166 | exprtk_register_function("threshold_below" , tb ) 46167 | #undef exprtk_register_function 46168 | 46169 | return true; 46170 | } 46171 | }; 46172 | 46173 | } // namespace exprtk::rtl::vecops 46174 | } // namespace exprtk::rtl 46175 | } // namespace exprtk 46176 | #endif 46177 | 46178 | namespace exprtk 46179 | { 46180 | namespace information 46181 | { 46182 | using ::exprtk::details::char_cptr; 46183 | 46184 | static char_cptr library = "Mathematical Expression Toolkit" 46185 | static char_cptr version = "2.718281828459045235360287471352662497757" 46186 | "24709369995957496696762772407663035354759" 46187 | "45713821785251664274274663919320030599218" 46188 | "17413596629043572900334295260595630738132" 46189 | static char_cptr date = "20240101" 46190 | static char_cptr min_cpp = "199711L" 46191 | 46192 | static inline std::string data() 46193 | { 46194 | static const std::string info_str = std::string(library) + 46195 | std::string(" v") + std::string(version) + 46196 | std::string(" (") + date + std::string(")") + 46197 | std::string(" (") + min_cpp + std::string(")"); 46198 | return info_str; 46199 | } 46200 | 46201 | } // namespace information 46202 | 46203 | #ifdef exprtk_debug 46204 | #undef exprtk_debug 46205 | #endif 46206 | 46207 | #ifdef exprtk_error_location 46208 | #undef exprtk_error_location 46209 | #endif 46210 | 46211 | #ifdef exprtk_fallthrough 46212 | #undef exprtk_fallthrough 46213 | #endif 46214 | 46215 | #ifdef exprtk_override 46216 | #undef exprtk_override 46217 | #endif 46218 | 46219 | #ifdef exprtk_final 46220 | #undef exprtk_final 46221 | #endif 46222 | 46223 | #ifdef exprtk_delete 46224 | #undef exprtk_delete 46225 | #endif 46226 | 46227 | } // namespace exprtk 46228 | 46229 | #endif