Skip to content

Commit

Permalink
Fix overflow
Browse files Browse the repository at this point in the history
  • Loading branch information
rui-mo committed Apr 11, 2024
1 parent ecf8118 commit 04b7fcf
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 10 deletions.
24 changes: 24 additions & 0 deletions velox/functions/sparksql/tests/SparkCastExprTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
* limitations under the License.
*/

#include "velox/common/base/tests/GTestUtils.h"
#include "velox/functions/prestosql/tests/CastBaseTest.h"
#include "velox/functions/sparksql/Register.h"
#include "velox/parse/TypeResolver.h"
Expand Down Expand Up @@ -453,6 +454,29 @@ TEST_F(SparkCastExprTest, overflow) {
testCast(
makeNullableFlatVector<int64_t>({214748364890}, DECIMAL(12, 2)),
makeNullableFlatVector<int64_t>({2147483648}));

VELOX_ASSERT_THROW(
(testCast<std::string, int8_t>("tinyint", {"166"}, {std::nullopt})),
"Cannot cast VARCHAR '166' to TINYINT. Value is too large for type.");
testTryCast<std::string, int32_t>("tinyint", {"166"}, {std::nullopt});

VELOX_ASSERT_THROW(
(testCast<std::string, int16_t>("smallint", {"52769"}, {std::nullopt})),
"Cannot cast VARCHAR '52769' to SMALLINT. Value is too large for type.");
testTryCast<std::string, int32_t>("tinyint", {"52769"}, {std::nullopt});

VELOX_ASSERT_THROW(
(testCast<std::string, int32_t>(
"integer", {"17515055537"}, {std::nullopt})),
"Cannot cast VARCHAR '17515055537' to INTEGER. Value is too large for type.");
testTryCast<std::string, int32_t>("integer", {"17515055537"}, {std::nullopt});

VELOX_ASSERT_THROW(
(testCast<std::string, int64_t>(
"integer", {"9663372036854775809"}, {std::nullopt})),
"Cannot cast VARCHAR '9663372036854775809' to BIGINT. Value is too large for type.");
testTryCast<std::string, int32_t>(
"integer", {"9663372036854775809"}, {std::nullopt});
}

TEST_F(SparkCastExprTest, timestampToString) {
Expand Down
26 changes: 16 additions & 10 deletions velox/type/Conversions.h
Original file line number Diff line number Diff line change
Expand Up @@ -195,11 +195,14 @@ struct Converter<
VELOX_USER_FAIL("Encountered a non-digit character");
}
if (!decimalPoint) {
result = result * 10 - (v[index] - '0');
}
// Overflow check
if (result > 0) {
VELOX_USER_FAIL("Value is too large for type");
bool overflow = __builtin_mul_overflow(result, 10, &result);
if (overflow) {
VELOX_USER_FAIL("Value is too large for type.");
}
overflow = __builtin_sub_overflow(result, v[index] - '0', &result);
if (overflow) {
VELOX_USER_FAIL("Value is too large for type.");
}
}
}
} else {
Expand All @@ -215,11 +218,14 @@ struct Converter<
VELOX_USER_FAIL("Encountered a non-digit character");
}
if (!decimalPoint) {
result = result * 10 + (v[index] - '0');
}
// Overflow check
if (result < 0) {
VELOX_USER_FAIL("Value is too large for type");
bool overflow = __builtin_mul_overflow(result, 10, &result);
if (overflow) {
VELOX_USER_FAIL("Value is too large for type.");
}
overflow = __builtin_add_overflow(result, v[index] - '0', &result);
if (overflow) {
VELOX_USER_FAIL("Value is too large for type.");
}
}
}
}
Expand Down

0 comments on commit 04b7fcf

Please sign in to comment.