mirror of
https://github.com/Kitware/CMake.git
synced 2025-10-21 06:10:16 +08:00
math: Diagnose divide-by-zero
This commit is contained in:
@@ -89,6 +89,7 @@ Modify cmExprParser.cxx:
|
|||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
/*-------------------------------------------------------------------------*/
|
/*-------------------------------------------------------------------------*/
|
||||||
#define YYDEBUG 1
|
#define YYDEBUG 1
|
||||||
@@ -108,7 +109,7 @@ static void cmExpr_yyerror(yyscan_t yyscanner, const char* message);
|
|||||||
# pragma warning (disable: 4065) /* Switch statement contains default but no case. */
|
# pragma warning (disable: 4065) /* Switch statement contains default but no case. */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#line 112 "cmExprParser.cxx" /* yacc.c:339 */
|
#line 113 "cmExprParser.cxx" /* yacc.c:339 */
|
||||||
|
|
||||||
# ifndef YY_NULLPTR
|
# ifndef YY_NULLPTR
|
||||||
# if defined __cplusplus && 201103L <= __cplusplus
|
# if defined __cplusplus && 201103L <= __cplusplus
|
||||||
@@ -187,7 +188,7 @@ int cmExpr_yyparse (yyscan_t yyscanner);
|
|||||||
|
|
||||||
/* Copy the second part of user declarations. */
|
/* Copy the second part of user declarations. */
|
||||||
|
|
||||||
#line 191 "cmExprParser.cxx" /* yacc.c:358 */
|
#line 192 "cmExprParser.cxx" /* yacc.c:358 */
|
||||||
|
|
||||||
#ifdef short
|
#ifdef short
|
||||||
# undef short
|
# undef short
|
||||||
@@ -486,9 +487,9 @@ static const yytype_uint8 yytranslate[] =
|
|||||||
/* YYRLINE[YYN] -- Source line where rule number YYN was defined. */
|
/* YYRLINE[YYN] -- Source line where rule number YYN was defined. */
|
||||||
static const yytype_uint8 yyrline[] =
|
static const yytype_uint8 yyrline[] =
|
||||||
{
|
{
|
||||||
0, 74, 74, 79, 82, 87, 90, 95, 98, 103,
|
0, 75, 75, 80, 83, 88, 91, 96, 99, 104,
|
||||||
106, 109, 114, 117, 120, 125, 128, 131, 134, 139,
|
107, 110, 115, 118, 121, 126, 129, 132, 138, 143,
|
||||||
142, 145, 150, 153
|
146, 149, 154, 157
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -1284,183 +1285,186 @@ yyreduce:
|
|||||||
switch (yyn)
|
switch (yyn)
|
||||||
{
|
{
|
||||||
case 2:
|
case 2:
|
||||||
#line 74 "cmExprParser.y" /* yacc.c:1646 */
|
#line 75 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
cmExpr_yyget_extra(yyscanner)->SetResult((yyvsp[0].Number));
|
cmExpr_yyget_extra(yyscanner)->SetResult((yyvsp[0].Number));
|
||||||
}
|
}
|
||||||
#line 1292 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1293 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 3:
|
case 3:
|
||||||
#line 79 "cmExprParser.y" /* yacc.c:1646 */
|
#line 80 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1300 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1301 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 4:
|
case 4:
|
||||||
#line 82 "cmExprParser.y" /* yacc.c:1646 */
|
#line 83 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) | (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) | (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1308 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1309 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 5:
|
case 5:
|
||||||
#line 87 "cmExprParser.y" /* yacc.c:1646 */
|
#line 88 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1316 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1317 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 6:
|
case 6:
|
||||||
#line 90 "cmExprParser.y" /* yacc.c:1646 */
|
#line 91 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) ^ (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) ^ (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1324 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1325 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 7:
|
case 7:
|
||||||
#line 95 "cmExprParser.y" /* yacc.c:1646 */
|
#line 96 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1332 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1333 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 8:
|
case 8:
|
||||||
#line 98 "cmExprParser.y" /* yacc.c:1646 */
|
#line 99 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) & (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) & (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1340 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1341 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 9:
|
case 9:
|
||||||
#line 103 "cmExprParser.y" /* yacc.c:1646 */
|
#line 104 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1348 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1349 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 10:
|
case 10:
|
||||||
#line 106 "cmExprParser.y" /* yacc.c:1646 */
|
#line 107 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) << (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) << (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1356 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1357 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 11:
|
case 11:
|
||||||
#line 109 "cmExprParser.y" /* yacc.c:1646 */
|
#line 110 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) >> (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) >> (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1364 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1365 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 12:
|
case 12:
|
||||||
#line 114 "cmExprParser.y" /* yacc.c:1646 */
|
#line 115 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1372 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1373 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 13:
|
case 13:
|
||||||
#line 117 "cmExprParser.y" /* yacc.c:1646 */
|
#line 118 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) + (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) + (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1380 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1381 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 14:
|
case 14:
|
||||||
#line 120 "cmExprParser.y" /* yacc.c:1646 */
|
#line 121 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) - (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) - (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1388 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1389 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 15:
|
case 15:
|
||||||
#line 125 "cmExprParser.y" /* yacc.c:1646 */
|
#line 126 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1396 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1397 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 16:
|
case 16:
|
||||||
#line 128 "cmExprParser.y" /* yacc.c:1646 */
|
#line 129 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) * (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) * (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1404 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1405 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 17:
|
case 17:
|
||||||
#line 131 "cmExprParser.y" /* yacc.c:1646 */
|
#line 132 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
|
if (yyvsp[0].Number == 0) {
|
||||||
|
throw std::overflow_error("divide by zero");
|
||||||
|
}
|
||||||
(yyval.Number) = (yyvsp[-2].Number) / (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) / (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1412 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1416 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 18:
|
case 18:
|
||||||
#line 134 "cmExprParser.y" /* yacc.c:1646 */
|
#line 138 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-2].Number) % (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[-2].Number) % (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1420 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1424 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 19:
|
case 19:
|
||||||
#line 139 "cmExprParser.y" /* yacc.c:1646 */
|
#line 143 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1428 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1432 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 20:
|
case 20:
|
||||||
#line 142 "cmExprParser.y" /* yacc.c:1646 */
|
#line 146 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = + (yyvsp[0].Number);
|
(yyval.Number) = + (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1436 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1440 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 21:
|
case 21:
|
||||||
#line 145 "cmExprParser.y" /* yacc.c:1646 */
|
#line 149 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = - (yyvsp[0].Number);
|
(yyval.Number) = - (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1444 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1448 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 22:
|
case 22:
|
||||||
#line 150 "cmExprParser.y" /* yacc.c:1646 */
|
#line 154 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[0].Number);
|
(yyval.Number) = (yyvsp[0].Number);
|
||||||
}
|
}
|
||||||
#line 1452 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1456 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 23:
|
case 23:
|
||||||
#line 153 "cmExprParser.y" /* yacc.c:1646 */
|
#line 157 "cmExprParser.y" /* yacc.c:1646 */
|
||||||
{
|
{
|
||||||
(yyval.Number) = (yyvsp[-1].Number);
|
(yyval.Number) = (yyvsp[-1].Number);
|
||||||
}
|
}
|
||||||
#line 1460 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1464 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
||||||
#line 1464 "cmExprParser.cxx" /* yacc.c:1646 */
|
#line 1468 "cmExprParser.cxx" /* yacc.c:1646 */
|
||||||
default: break;
|
default: break;
|
||||||
}
|
}
|
||||||
/* User semantic actions sometimes alter yychar, and that requires
|
/* User semantic actions sometimes alter yychar, and that requires
|
||||||
@@ -1690,7 +1694,7 @@ yyreturn:
|
|||||||
#endif
|
#endif
|
||||||
return yyresult;
|
return yyresult;
|
||||||
}
|
}
|
||||||
#line 158 "cmExprParser.y" /* yacc.c:1906 */
|
#line 162 "cmExprParser.y" /* yacc.c:1906 */
|
||||||
|
|
||||||
/* End of grammar */
|
/* End of grammar */
|
||||||
|
|
||||||
|
@@ -18,6 +18,7 @@ Modify cmExprParser.cxx:
|
|||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
/*-------------------------------------------------------------------------*/
|
/*-------------------------------------------------------------------------*/
|
||||||
#define YYDEBUG 1
|
#define YYDEBUG 1
|
||||||
@@ -129,6 +130,9 @@ term:
|
|||||||
$<Number>$ = $<Number>1 * $<Number>3;
|
$<Number>$ = $<Number>1 * $<Number>3;
|
||||||
}
|
}
|
||||||
| term exp_DIVIDE unary {
|
| term exp_DIVIDE unary {
|
||||||
|
if (yyvsp[0].Number == 0) {
|
||||||
|
throw std::overflow_error("divide by zero");
|
||||||
|
}
|
||||||
$<Number>$ = $<Number>1 / $<Number>3;
|
$<Number>$ = $<Number>1 / $<Number>3;
|
||||||
}
|
}
|
||||||
| term exp_MOD unary {
|
| term exp_MOD unary {
|
||||||
|
1
Tests/RunCMake/math/MATH-DivideByZero-result.txt
Normal file
1
Tests/RunCMake/math/MATH-DivideByZero-result.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
1
|
4
Tests/RunCMake/math/MATH-DivideByZero-stderr.txt
Normal file
4
Tests/RunCMake/math/MATH-DivideByZero-stderr.txt
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
^CMake Error at MATH-DivideByZero.cmake:1 \(math\):
|
||||||
|
math cannot evaluate the expression: "100/0": divide by zero.
|
||||||
|
Call Stack \(most recent call first\):
|
||||||
|
CMakeLists.txt:3 \(include\)$
|
1
Tests/RunCMake/math/MATH-DivideByZero.cmake
Normal file
1
Tests/RunCMake/math/MATH-DivideByZero.cmake
Normal file
@@ -0,0 +1 @@
|
|||||||
|
math(EXPR var "100/0")
|
@@ -2,3 +2,4 @@ include(RunCMake)
|
|||||||
|
|
||||||
run_cmake(MATH)
|
run_cmake(MATH)
|
||||||
run_cmake(MATH-InvalidExpression)
|
run_cmake(MATH-InvalidExpression)
|
||||||
|
run_cmake(MATH-DivideByZero)
|
||||||
|
Reference in New Issue
Block a user