Coverage Report

Created: 2025-07-11 06:27

/src/exprtk_fuzzer.cpp
Line
Count
Source (jump to first uncovered line)
1
// Copyright 2020 Google LLC
2
//
3
// Licensed under the Apache License, Version 2.0 (the "License");
4
// you may not use this file except in compliance with the License.
5
// You may obtain a copy of the License at
6
//
7
//      http://www.apache.org/licenses/LICENSE-2.0
8
//
9
// Unless required by applicable law or agreed to in writing, software
10
// distributed under the License is distributed on an "AS IS" BASIS,
11
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
// See the License for the specific language governing permissions and
13
// limitations under the License.
14
15
16
#include <chrono>
17
#include <cstdint>
18
#include <string>
19
20
#define exprtk_enable_range_runtime_checks
21
#include "exprtk.hpp"
22
23
24
constexpr auto max_test_duration = std::chrono::seconds(58); // OSSFuzz test time is 60seconds
25
const     auto global_timeout_tp = std::chrono::steady_clock::now() + max_test_duration;
26
27
struct timeout_rtc_handler : public exprtk::loop_runtime_check
28
{
29
   timeout_rtc_handler()
30
9.32k
   : exprtk::loop_runtime_check()
31
9.32k
   {}
32
33
   class timeout_exception : public std::runtime_error
34
   {
35
   public:
36
       timeout_exception(const std::string& what = "")
37
0
       : std::runtime_error(what)
38
0
       {}
39
   };
40
41
   static constexpr std::size_t max_iterations = 5000000;
42
43
   using time_point_t = std::chrono::time_point<std::chrono::steady_clock>;
44
45
   void set_timeout_time(const time_point_t& timeout_tp)
46
4.03k
   {
47
4.03k
      timeout_tp_ = timeout_tp;
48
4.03k
   }
49
50
   bool check() override
51
0
   {
52
0
      if (++iterations_ >= max_iterations)
53
0
      {
54
0
         if (std::chrono::steady_clock::now() >= timeout_tp_)
55
0
         {
56
0
            return false;
57
0
         }
58
59
0
         iterations_ = 0;
60
0
      }
61
62
0
      return true;
63
0
   }
64
65
   void handle_runtime_violation(const violation_context& /*context*/) override
66
0
   {
67
0
      throw timeout_exception("ExprTk Loop run-time timeout violation.");
68
0
   }
69
70
   std::size_t iterations_ = 0;
71
   time_point_t timeout_tp_;
72
};
73
74
struct compilation_timeout_check final : public exprtk::compilation_check
75
{
76
   static constexpr std::size_t max_iters_per_check = 500;
77
78
   bool continue_compilation(compilation_context& context) override
79
320k
   {
80
320k
      if (++iterations_ >= max_iters_per_check)
81
8.03k
      {
82
8.03k
         if (std::chrono::steady_clock::now() >= timeout_tp_)
83
14
         {
84
14
            context.error_message = "Compilation has timed-out";
85
14
            return false;
86
14
         }
87
88
8.02k
         iterations_ = 0;
89
8.02k
      }
90
91
320k
      return true;
92
320k
   }
93
94
   using time_point_t = std::chrono::time_point<std::chrono::steady_clock>;
95
96
   void set_timeout_time(const time_point_t& timeout_tp)
97
9.32k
   {
98
9.32k
      timeout_tp_ = timeout_tp;
99
9.32k
   }
100
101
   std::size_t iterations_ = max_iters_per_check;
102
   time_point_t timeout_tp_;
103
};
104
105
struct vector_access_rtc final : public exprtk::vector_access_runtime_check
106
{
107
   bool handle_runtime_violation(violation_context& /*context*/) override
108
0
   {
109
0
      throw std::runtime_error("Runtime vector access violation.");
110
0
      return false;
111
0
   }
112
};
113
114
struct assert_handler final : public exprtk::assert_check
115
{
116
   void handle_assert(const assert_context& /*context*/) override
117
0
   {
118
0
      throw std::runtime_error("assert: vector access violation.");
119
0
   }
120
};
121
122
template <typename T>
123
void run(const std::string& expression_string)
124
9.32k
{
125
9.32k
   using symbol_table_t       = exprtk::symbol_table<T>;
126
9.32k
   using expression_t         = exprtk::expression<T>;
127
9.32k
   using parser_t             = exprtk::parser<T>;
128
9.32k
   using loop_runtime_check_t = exprtk::loop_runtime_check;
129
130
9.32k
   T x = T(1.2345);
131
9.32k
   T y = T(2.2345);
132
9.32k
   T z = T(3.2345);
133
9.32k
   T w = T(4.2345);
134
135
9.32k
   symbol_table_t symbol_table;
136
9.32k
   symbol_table.add_variable("x",x);
137
9.32k
   symbol_table.add_variable("y",y);
138
9.32k
   symbol_table.add_variable("z",z);
139
9.32k
   symbol_table.add_variable("w",w);
140
9.32k
   symbol_table.add_constants();
141
142
9.32k
   expression_t expression;
143
9.32k
   expression.register_symbol_table(symbol_table);
144
145
9.32k
   timeout_rtc_handler loop_runtime_check;
146
9.32k
   loop_runtime_check.loop_set = loop_runtime_check_t::e_all_loops;
147
9.32k
   loop_runtime_check.max_loop_iterations = 100000;
148
149
9.32k
   compilation_timeout_check compilation_timeout_chck;
150
9.32k
   vector_access_rtc         vector_rtc;
151
9.32k
   assert_handler            asrt_handler;
152
153
9.32k
   parser_t parser;
154
155
9.32k
   parser.settings().set_max_stack_depth(400);
156
9.32k
   parser.settings().set_max_node_depth (400);
157
9.32k
   parser.settings().set_max_local_vector_size(10000000); // double: 80MB float: 40MB
158
159
9.32k
   parser.register_compilation_timeout_check  (compilation_timeout_chck);
160
9.32k
   parser.register_loop_runtime_check         (loop_runtime_check      );
161
9.32k
   parser.register_vector_access_runtime_check(vector_rtc              );
162
9.32k
   parser.register_assert_check               (asrt_handler            );
163
164
9.32k
   compilation_timeout_chck.set_timeout_time(global_timeout_tp);
165
166
9.32k
   if (parser.compile(expression_string, expression))
167
4.04k
   {
168
4.04k
      const std::size_t max_expression_size = 64 * 1024;
169
170
4.04k
      if (expression_string.size() <= max_expression_size)
171
4.03k
      {
172
4.03k
         loop_runtime_check.set_timeout_time(global_timeout_tp);
173
174
4.03k
         try
175
4.03k
         {
176
4.03k
            expression.value();
177
4.03k
         }
178
4.03k
         catch (std::runtime_error& rte)
179
4.03k
         {}
180
4.03k
         catch (...)
181
4.03k
         {}
182
183
4.03k
         parser.clear_loop_runtime_check();
184
4.03k
      }
185
4.04k
   }
186
9.32k
}
void run<double>(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&)
Line
Count
Source
124
4.66k
{
125
4.66k
   using symbol_table_t       = exprtk::symbol_table<T>;
126
4.66k
   using expression_t         = exprtk::expression<T>;
127
4.66k
   using parser_t             = exprtk::parser<T>;
128
4.66k
   using loop_runtime_check_t = exprtk::loop_runtime_check;
129
130
4.66k
   T x = T(1.2345);
131
4.66k
   T y = T(2.2345);
132
4.66k
   T z = T(3.2345);
133
4.66k
   T w = T(4.2345);
134
135
4.66k
   symbol_table_t symbol_table;
136
4.66k
   symbol_table.add_variable("x",x);
137
4.66k
   symbol_table.add_variable("y",y);
138
4.66k
   symbol_table.add_variable("z",z);
139
4.66k
   symbol_table.add_variable("w",w);
140
4.66k
   symbol_table.add_constants();
141
142
4.66k
   expression_t expression;
143
4.66k
   expression.register_symbol_table(symbol_table);
144
145
4.66k
   timeout_rtc_handler loop_runtime_check;
146
4.66k
   loop_runtime_check.loop_set = loop_runtime_check_t::e_all_loops;
147
4.66k
   loop_runtime_check.max_loop_iterations = 100000;
148
149
4.66k
   compilation_timeout_check compilation_timeout_chck;
150
4.66k
   vector_access_rtc         vector_rtc;
151
4.66k
   assert_handler            asrt_handler;
152
153
4.66k
   parser_t parser;
154
155
4.66k
   parser.settings().set_max_stack_depth(400);
156
4.66k
   parser.settings().set_max_node_depth (400);
157
4.66k
   parser.settings().set_max_local_vector_size(10000000); // double: 80MB float: 40MB
158
159
4.66k
   parser.register_compilation_timeout_check  (compilation_timeout_chck);
160
4.66k
   parser.register_loop_runtime_check         (loop_runtime_check      );
161
4.66k
   parser.register_vector_access_runtime_check(vector_rtc              );
162
4.66k
   parser.register_assert_check               (asrt_handler            );
163
164
4.66k
   compilation_timeout_chck.set_timeout_time(global_timeout_tp);
165
166
4.66k
   if (parser.compile(expression_string, expression))
167
2.02k
   {
168
2.02k
      const std::size_t max_expression_size = 64 * 1024;
169
170
2.02k
      if (expression_string.size() <= max_expression_size)
171
2.02k
      {
172
2.02k
         loop_runtime_check.set_timeout_time(global_timeout_tp);
173
174
2.02k
         try
175
2.02k
         {
176
2.02k
            expression.value();
177
2.02k
         }
178
2.02k
         catch (std::runtime_error& rte)
179
2.02k
         {}
180
2.02k
         catch (...)
181
2.02k
         {}
182
183
2.02k
         parser.clear_loop_runtime_check();
184
2.02k
      }
185
2.02k
   }
186
4.66k
}
void run<float>(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&)
Line
Count
Source
124
4.66k
{
125
4.66k
   using symbol_table_t       = exprtk::symbol_table<T>;
126
4.66k
   using expression_t         = exprtk::expression<T>;
127
4.66k
   using parser_t             = exprtk::parser<T>;
128
4.66k
   using loop_runtime_check_t = exprtk::loop_runtime_check;
129
130
4.66k
   T x = T(1.2345);
131
4.66k
   T y = T(2.2345);
132
4.66k
   T z = T(3.2345);
133
4.66k
   T w = T(4.2345);
134
135
4.66k
   symbol_table_t symbol_table;
136
4.66k
   symbol_table.add_variable("x",x);
137
4.66k
   symbol_table.add_variable("y",y);
138
4.66k
   symbol_table.add_variable("z",z);
139
4.66k
   symbol_table.add_variable("w",w);
140
4.66k
   symbol_table.add_constants();
141
142
4.66k
   expression_t expression;
143
4.66k
   expression.register_symbol_table(symbol_table);
144
145
4.66k
   timeout_rtc_handler loop_runtime_check;
146
4.66k
   loop_runtime_check.loop_set = loop_runtime_check_t::e_all_loops;
147
4.66k
   loop_runtime_check.max_loop_iterations = 100000;
148
149
4.66k
   compilation_timeout_check compilation_timeout_chck;
150
4.66k
   vector_access_rtc         vector_rtc;
151
4.66k
   assert_handler            asrt_handler;
152
153
4.66k
   parser_t parser;
154
155
4.66k
   parser.settings().set_max_stack_depth(400);
156
4.66k
   parser.settings().set_max_node_depth (400);
157
4.66k
   parser.settings().set_max_local_vector_size(10000000); // double: 80MB float: 40MB
158
159
4.66k
   parser.register_compilation_timeout_check  (compilation_timeout_chck);
160
4.66k
   parser.register_loop_runtime_check         (loop_runtime_check      );
161
4.66k
   parser.register_vector_access_runtime_check(vector_rtc              );
162
4.66k
   parser.register_assert_check               (asrt_handler            );
163
164
4.66k
   compilation_timeout_chck.set_timeout_time(global_timeout_tp);
165
166
4.66k
   if (parser.compile(expression_string, expression))
167
2.01k
   {
168
2.01k
      const std::size_t max_expression_size = 64 * 1024;
169
170
2.01k
      if (expression_string.size() <= max_expression_size)
171
2.01k
      {
172
2.01k
         loop_runtime_check.set_timeout_time(global_timeout_tp);
173
174
2.01k
         try
175
2.01k
         {
176
2.01k
            expression.value();
177
2.01k
         }
178
2.01k
         catch (std::runtime_error& rte)
179
2.01k
         {}
180
2.01k
         catch (...)
181
2.01k
         {}
182
183
2.01k
         parser.clear_loop_runtime_check();
184
2.01k
      }
185
2.01k
   }
186
4.66k
}
187
188
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size)
189
4.66k
{
190
4.66k
   const std::string expression(reinterpret_cast<const char*>(data), size);
191
192
4.66k
   run<double>(expression);
193
4.66k
   run<float> (expression);
194
195
4.66k
   return 0;
196
4.66k
}