// Copyright 2015 PDFium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <limits>
#include "core/fxcrt/fx_string.h"
#include "core/fxcrt/fx_system.h"
#include "testing/gtest/include/gtest/gtest.h"
// Unit test covering cases where PDFium replaces well-known library
// functionality on any given platformn.
#if _FX_PLATFORM_ != _FX_PLATFORM_WINDOWS_
namespace {
const char kSentinel = 0x7f;
void Check32BitBase16Itoa(int32_t input, const char* expected_output) {
const size_t kBufLen = 11; // "-" + 8 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_itoa(input, buf, 16);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
void Check32BitBase10Itoa(int32_t input, const char* expected_output) {
const size_t kBufLen = 13; // "-" + 10 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_itoa(input, buf, 10);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
void Check32BitBase2Itoa(int32_t input, const char* expected_output) {
const size_t kBufLen = 35; // "-" + 32 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_itoa(input, buf, 2);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
void Check64BitBase16Itoa(int64_t input, const char* expected_output) {
const size_t kBufLen = 19; // "-" + 16 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_i64toa(input, buf, 16);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
void Check64BitBase10Itoa(int64_t input, const char* expected_output) {
const size_t kBufLen = 22; // "-" + 19 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_i64toa(input, buf, 10);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
void Check64BitBase2Itoa(int64_t input, const char* expected_output) {
const size_t kBufLen = 67; // "-" + 64 digits + NUL + sentinel.
char buf[kBufLen];
buf[kBufLen - 1] = kSentinel;
FXSYS_i64toa(input, buf, 2);
EXPECT_STREQ(expected_output, buf);
EXPECT_EQ(kSentinel, buf[kBufLen - 1]);
}
} // namespace
TEST(fxcrt, FXSYS_itoa_InvalidRadix) {
char buf[32];
FXSYS_itoa(42, buf, 17); // Ours stops at 16.
EXPECT_STREQ("", buf);
FXSYS_itoa(42, buf, 1);
EXPECT_STREQ("", buf);
FXSYS_itoa(42, buf, 0);
EXPECT_STREQ("", buf);
FXSYS_itoa(42, buf, -1);
EXPECT_STREQ("", buf);
}
TEST(fxcrt, FXSYS_itoa) {
Check32BitBase16Itoa(std::numeric_limits<int32_t>::min(), "-80000000");
Check32BitBase10Itoa(std::numeric_limits<int32_t>::min(), "-2147483648");
Check32BitBase2Itoa(std::numeric_limits<int32_t>::min(),
"-10000000000000000000000000000000");
Check32BitBase16Itoa(-1, "-1");
Check32BitBase10Itoa(-1, "-1");
Check32BitBase2Itoa(-1, "-1");
Check32BitBase16Itoa(0, "0");
Check32BitBase10Itoa(0, "0");
Check32BitBase2Itoa(0, "0");
Check32BitBase16Itoa(42, "2a");
Check32BitBase10Itoa(42, "42");
Check32BitBase2Itoa(42, "101010");
Check32BitBase16Itoa(std::numeric_limits<int32_t>::max(), "7fffffff");
Check32BitBase10Itoa(std::numeric_limits<int32_t>::max(), "2147483647");
Check32BitBase2Itoa(std::numeric_limits<int32_t>::max(),
"1111111111111111111111111111111");
}
TEST(fxcrt, FXSYS_i64toa_InvalidRadix) {
char buf[32];
FXSYS_i64toa(42, buf, 17); // Ours stops at 16.
EXPECT_STREQ("", buf);
FXSYS_i64toa(42, buf, 1);
EXPECT_STREQ("", buf);
FXSYS_i64toa(42, buf, 0);
EXPECT_STREQ("", buf);
FXSYS_i64toa(42, buf, -1);
EXPECT_STREQ("", buf);
}
TEST(fxcrt, FXSYS_i64toa) {
Check64BitBase16Itoa(std::numeric_limits<int64_t>::min(),
"-8000000000000000");
Check64BitBase10Itoa(std::numeric_limits<int64_t>::min(),
"-9223372036854775808");
Check64BitBase2Itoa(
std::numeric_limits<int64_t>::min(),
"-1000000000000000000000000000000000000000000000000000000000000000");
Check64BitBase16Itoa(-1, "-1");
Check64BitBase10Itoa(-1, "-1");
Check64BitBase2Itoa(-1, "-1");
Check64BitBase16Itoa(0, "0");
Check64BitBase10Itoa(0, "0");
Check64BitBase2Itoa(0, "0");
Check64BitBase16Itoa(42, "2a");
Check64BitBase10Itoa(42, "42");
Check64BitBase2Itoa(42, "101010");
Check64BitBase16Itoa(std::numeric_limits<int64_t>::max(), "7fffffffffffffff");
Check64BitBase10Itoa(std::numeric_limits<int64_t>::max(),
"9223372036854775807");
Check64BitBase2Itoa(
std::numeric_limits<int64_t>::max(),
"111111111111111111111111111111111111111111111111111111111111111");
}
#endif // _FX_PLATFORM_ != _FX_PLATFORM_WINDOWS_
TEST(fxcrt, FXSYS_wcsftime) {
struct tm good_time = {};
good_time.tm_year = 74; // 1900-based.
good_time.tm_mon = 7; // 0-based.
good_time.tm_mday = 9; // 1-based.
good_time.tm_hour = 11;
good_time.tm_min = 59;
good_time.tm_sec = 59;
wchar_t buf[100] = {};
EXPECT_EQ(19u, FXSYS_wcsftime(buf, FX_ArraySize(buf), L"%Y-%m-%dT%H:%M:%S",
&good_time));
EXPECT_STREQ(L"1974-08-09T11:59:59", buf);
// Ensure wcsftime handles a wide range of years without crashing.
struct tm year_time = {};
year_time.tm_mon = 7; // 0-based.
year_time.tm_mday = 9; // 1-based.
year_time.tm_hour = 11;
year_time.tm_min = 59;
year_time.tm_sec = 59;
for (int year = -2500; year <= 8500; ++year) {
year_time.tm_year = year;
wchar_t year_buf[100] = {};
FXSYS_wcsftime(year_buf, FX_ArraySize(year_buf), L"%Y-%m-%dT%H:%M:%S",
&year_time);
}
// Ensure wcsftime handles bad years, etc. without crashing.
struct tm bad_time = {};
bad_time.tm_year = -1;
bad_time.tm_mon = -1;
bad_time.tm_mday = -1;
bad_time.tm_hour = -1;
bad_time.tm_min = -1;
bad_time.tm_sec = -1;
FXSYS_wcsftime(buf, FX_ArraySize(buf), L"%y-%m-%dT%H:%M:%S", &bad_time);
// Ensure wcsftime handles bad-ish day without crashing (Feb 30).
struct tm feb_time = {};
feb_time.tm_year = 115; // 1900-based.
feb_time.tm_mon = 1; // 0-based.
feb_time.tm_mday = 30; // 1-based.
feb_time.tm_hour = 12;
feb_time.tm_min = 00;
feb_time.tm_sec = 00;
FXSYS_wcsftime(buf, FX_ArraySize(buf), L"%y-%m-%dT%H:%M:%S", &feb_time);
}
TEST(fxcrt, FXSYS_atoi) {
EXPECT_EQ(0, FXSYS_atoi(""));
EXPECT_EQ(0, FXSYS_atoi("0"));
EXPECT_EQ(-1, FXSYS_atoi("-1"));
EXPECT_EQ(2345, FXSYS_atoi("2345"));
EXPECT_EQ(-2147483647, FXSYS_atoi("-2147483647"));
// Handle the sign.
EXPECT_EQ(-2345, FXSYS_atoi("-2345"));
EXPECT_EQ(2345, FXSYS_atoi("+2345"));
// The max value.
EXPECT_EQ(2147483647, FXSYS_atoi("2147483647"));
// The min value. Written in -1 format to avoid "unary minus operator applied
// to unsigned type" warning.
EXPECT_EQ(-2147483647 - 1, FXSYS_atoi("-2147483648"));
// With invalid char.
EXPECT_EQ(9, FXSYS_atoi("9x9"));
// Out of range values.
EXPECT_EQ(2147483647, FXSYS_atoi("2147483623423412348"));
EXPECT_EQ(2147483647, FXSYS_atoi("2147483648"));
EXPECT_EQ(-2147483647 - 1, FXSYS_atoi("-2147483650"));
}
TEST(fxcrt, FXSYS_atoi64) {
EXPECT_EQ(0, FXSYS_atoi64(""));
EXPECT_EQ(0, FXSYS_atoi64("0"));
EXPECT_EQ(-1, FXSYS_atoi64("-1"));
EXPECT_EQ(2345, FXSYS_atoi64("2345"));
EXPECT_EQ(-9223372036854775807LL, FXSYS_atoi64("-9223372036854775807"));
// Handle the sign.
EXPECT_EQ(-2345, FXSYS_atoi64("-2345"));
EXPECT_EQ(2345, FXSYS_atoi64("+2345"));
// The max value.
EXPECT_EQ(9223372036854775807LL, FXSYS_atoi64("9223372036854775807"));
// The min value. Written in -1 format to avoid implicit unsigned warning.
EXPECT_EQ(-9223372036854775807LL - 1LL, FXSYS_atoi64("-9223372036854775808"));
// With invalid char.
EXPECT_EQ(9, FXSYS_atoi64("9x9"));
// Out of range values.
EXPECT_EQ(9223372036854775807LL, FXSYS_atoi64("922337203685471234123475807"));
EXPECT_EQ(9223372036854775807LL, FXSYS_atoi64("9223372036854775808"));
EXPECT_EQ(-9223372036854775807LL - 1LL, FXSYS_atoi64("-9223372036854775810"));
}
TEST(fxcrt, FXSYS_wtoi) {
EXPECT_EQ(0, FXSYS_wtoi(L""));
EXPECT_EQ(0, FXSYS_wtoi(L"0"));
EXPECT_EQ(-1, FXSYS_wtoi(L"-1"));
EXPECT_EQ(2345, FXSYS_wtoi(L"2345"));
EXPECT_EQ(-2147483647, FXSYS_wtoi(L"-2147483647"));
// The max value.
EXPECT_EQ(2147483647, FXSYS_wtoi(L"2147483647"));
// The min value.
EXPECT_EQ(-2147483647 - 1, FXSYS_wtoi(L"-2147483648"));
// Out of range values.
EXPECT_EQ(2147483647, FXSYS_wtoi(L"2147483623423412348"));
EXPECT_EQ(2147483647, FXSYS_wtoi(L"2147483648"));
EXPECT_EQ(-2147483647 - 1, FXSYS_wtoi(L"-2147483652"));
}
TEST(fxcrt, FXSYS_atoui) {
EXPECT_EQ(0u, FXSYS_atoui(""));
EXPECT_EQ(0u, FXSYS_atoui("0"));
EXPECT_EQ(4294967295, FXSYS_atoui("-1"));
EXPECT_EQ(2345u, FXSYS_atoui("2345"));
// Handle the sign.
EXPECT_EQ(4294964951, FXSYS_atoui("-2345"));
EXPECT_EQ(2345u, FXSYS_atoui("+2345"));
// The max value.
EXPECT_EQ(4294967295, FXSYS_atoui("4294967295"));
EXPECT_EQ(9u, FXSYS_atoui("9x9"));
// Out of range values.
EXPECT_EQ(4294967295, FXSYS_atoui("2147483623423412348"));
EXPECT_EQ(4294967295, FXSYS_atoui("4294967296"));
EXPECT_EQ(4294967295, FXSYS_atoui("-4294967345"));
}