summaryrefslogtreecommitdiffstats
path: root/base
diff options
context:
space:
mode:
authormmentovai@google.com <mmentovai@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2008-08-12 18:48:58 +0000
committermmentovai@google.com <mmentovai@google.com@0039d316-1c4b-4281-b951-d872f2087c98>2008-08-12 18:48:58 +0000
commit60632023552251e52b1948724e28a93d595cb72e (patch)
tree0ae64732e2659331eb26659d6c0d2ebaf1073d20 /base
parentb8c541b9d4f8769f5a191340f479c957f9886b38 (diff)
downloadchromium_src-60632023552251e52b1948724e28a93d595cb72e.zip
chromium_src-60632023552251e52b1948724e28a93d595cb72e.tar.gz
chromium_src-60632023552251e52b1948724e28a93d595cb72e.tar.bz2
Fix string_util and its tests for the Mac, GCC, UTF-32 wchar_t platforms, and POSIX systems generally.
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@736 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'base')
-rw-r--r--base/base.xcodeproj/project.pbxproj8
-rw-r--r--base/basictypes.h9
-rw-r--r--base/port.h25
-rw-r--r--base/string_util.cc91
-rw-r--r--base/string_util.h6
-rw-r--r--base/string_util_icu.cc6
-rw-r--r--base/string_util_unittest.cc147
7 files changed, 178 insertions, 114 deletions
diff --git a/base/base.xcodeproj/project.pbxproj b/base/base.xcodeproj/project.pbxproj
index f9f8844..f64ff74 100644
--- a/base/base.xcodeproj/project.pbxproj
+++ b/base/base.xcodeproj/project.pbxproj
@@ -45,6 +45,8 @@
7BAF50760E50B8F100CA8A07 /* file_version_info_mac.mm in Sources */ = {isa = PBXBuildFile; fileRef = 7BAF50740E50B8F100CA8A07 /* file_version_info_mac.mm */; };
7BAF50780E50B8F100CA8A07 /* file_version_info_mac.mm in Sources */ = {isa = PBXBuildFile; fileRef = 7BAF50740E50B8F100CA8A07 /* file_version_info_mac.mm */; };
7BAF50790E50B8F100CA8A07 /* file_version_info_unittest.cc in Sources */ = {isa = PBXBuildFile; fileRef = 7BAF50750E50B8F100CA8A07 /* file_version_info_unittest.cc */; };
+ 7BAF50AA0E50BACB00CA8A07 /* string_util_posix.h in Headers */ = {isa = PBXBuildFile; fileRef = 7BAF50A90E50BACB00CA8A07 /* string_util_posix.h */; };
+ 7BAF50B20E50BAE700CA8A07 /* string_util_unittest.cc in Sources */ = {isa = PBXBuildFile; fileRef = 7BAF50B10E50BAE700CA8A07 /* string_util_unittest.cc */; };
820EB4F00E3A610A009668FC /* linked_ptr.h in Headers */ = {isa = PBXBuildFile; fileRef = 820EB4EF0E3A610A009668FC /* linked_ptr.h */; };
820EB4F70E3A613F009668FC /* string_piece.cc in Sources */ = {isa = PBXBuildFile; fileRef = 820EB4F50E3A613F009668FC /* string_piece.cc */; };
820EB4F80E3A613F009668FC /* string_piece.h in Headers */ = {isa = PBXBuildFile; fileRef = 820EB4F60E3A613F009668FC /* string_piece.h */; };
@@ -363,6 +365,8 @@
7BAF501B0E50B84200CA8A07 /* base_paths.cc */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = base_paths.cc; sourceTree = "<group>"; };
7BAF50740E50B8F100CA8A07 /* file_version_info_mac.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = file_version_info_mac.mm; sourceTree = "<group>"; };
7BAF50750E50B8F100CA8A07 /* file_version_info_unittest.cc */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = file_version_info_unittest.cc; sourceTree = "<group>"; };
+ 7BAF50A90E50BACB00CA8A07 /* string_util_posix.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = string_util_posix.h; sourceTree = "<group>"; };
+ 7BAF50B10E50BAE700CA8A07 /* string_util_unittest.cc */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = string_util_unittest.cc; sourceTree = "<group>"; };
7BD9E84E0DA447F800FC7A01 /* singleton.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = singleton.h; sourceTree = "<group>"; };
7BEB81100D9AD288009BA8DD /* prcpucfg_mac.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = prcpucfg_mac.h; path = third_party/nspr/prcpucfg_mac.h; sourceTree = "<group>"; };
7BEB81490D9B0F33009BA8DD /* time_posix.cc */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = time_posix.cc; sourceTree = "<group>"; };
@@ -750,6 +754,8 @@
820EB4F90E3A6178009668FC /* string_util_icu.cc */,
7BEB834D0D9C4BE0009BA8DD /* string_util_mac.cc */,
7BEE52C10D9D84FD0067FF23 /* string_util_mac.h */,
+ 7BAF50A90E50BACB00CA8A07 /* string_util_posix.h */,
+ 7BAF50B10E50BAE700CA8A07 /* string_util_unittest.cc */,
821B91680DAABD7F00F350D7 /* string16.h */,
7B4C5F470E4B6BF900679E8F /* sys_string_conversions.h */,
7B4C5F480E4B6BF900679E8F /* sys_string_conversions_mac.cc */,
@@ -946,6 +952,7 @@
A5A0276C0E4BA33700498DA9 /* build_config.h in Headers */,
7BAF4F0B0E50A2FD00CA8A07 /* notimplemented.h in Headers */,
7BAF4F7C0E50A3BD00CA8A07 /* logging.h in Headers */,
+ 7BAF50AA0E50BACB00CA8A07 /* string_util_posix.h in Headers */,
);
runOnlyForDeploymentPostprocessing = 0;
};
@@ -1288,6 +1295,7 @@
7BAF501D0E50B84200CA8A07 /* base_paths.cc in Sources */,
7BAF50780E50B8F100CA8A07 /* file_version_info_mac.mm in Sources */,
7BAF50790E50B8F100CA8A07 /* file_version_info_unittest.cc in Sources */,
+ 7BAF50B20E50BAE700CA8A07 /* string_util_unittest.cc in Sources */,
E4AFA4A80E50D84900201347 /* prtime.cc in Sources */,
E4AFA4A90E50D85500201347 /* time.cc in Sources */,
E4AFA4AA0E50D85800201347 /* time_posix.cc in Sources */,
diff --git a/base/basictypes.h b/base/basictypes.h
index 26b4c3d..10a557e 100644
--- a/base/basictypes.h
+++ b/base/basictypes.h
@@ -26,10 +26,9 @@
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-// Copied from base/basictypes.h with some modifications
-#ifndef BASE_BASICTYPES_H__
-#define BASE_BASICTYPES_H__
+#ifndef BASE_BASICTYPES_H_
+#define BASE_BASICTYPES_H_
#include <assert.h> // for use with down_cast<>
#include <limits.h> // So we can set the bounds of our types
@@ -416,7 +415,7 @@ inline Dest bit_cast(const Source& source) {
// static MyClass my_variable_name(base::LINKER_INITIALIZED);
namespace base {
enum LinkerInitialized { LINKER_INITIALIZED };
-}
+} // base
-#endif // BASE_BASICTYPES_H__
+#endif // BASE_BASICTYPES_H_
diff --git a/base/port.h b/base/port.h
index 12b2a90..885e526 100644
--- a/base/port.h
+++ b/base/port.h
@@ -27,9 +27,10 @@
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-#ifndef BASE_PORT_H__
-#define BASE_PORT_H__
+#ifndef BASE_PORT_H_
+#define BASE_PORT_H_
+#include <stdarg.h>
#include "build/build_config.h"
#ifdef COMPILER_MSVC
@@ -56,4 +57,22 @@
#define GG_UINT32_C(x) (x ## U)
#define GG_UINT64_C(x) GG_ULONGLONG(x)
-#endif // BASE_PORT_H__
+namespace base {
+
+// It's possible for functions that use a va_list, such as StringPrintf, to
+// invalidate the data in it upon use. The fix is to make a copy of the
+// structure before using it and use that copy instead. va_copy is provided
+// for this purpose. MSVC does not provide va_copy, so define an
+// implementation here. It is not guaranteed that assignment is a copy, so the
+// StringUtil.VariableArgsFunc unit test tests this capability.
+inline void va_copy(va_list& a, va_list& b) {
+#if defined(COMPILER_GCC)
+ ::va_copy(a, b);
+#elif defined(COMPILER_MSVC)
+ a = b;
+#endif
+}
+
+} // namespace base
+
+#endif // BASE_PORT_H_
diff --git a/base/string_util.cc b/base/string_util.cc
index 31f97131b..bd02f3d 100644
--- a/base/string_util.cc
+++ b/base/string_util.cc
@@ -27,8 +27,6 @@
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-// StringPrintf stuff based on strings/stringprintf.cc by Sanjay Ghemawat
-
#include "base/string_util.h"
#include <ctype.h>
@@ -47,7 +45,6 @@
#include "base/basictypes.h"
#include "base/logging.h"
-#include "base/scoped_ptr.h"
#include "base/singleton.h"
namespace {
@@ -763,9 +760,10 @@ void ReplaceSubstringsAfterOffset(std::string* str,
// Overloaded wrappers around vsnprintf and vswprintf. The buf_size parameter
// is the size of the buffer. These return the number of characters in the
-// formatted string excluding the NUL terminator, or if the buffer is not
-// large enough to accommodate the formatted string without truncation, the
-// number of characters that would be in the fully-formatted string.
+// formatted string excluding the NUL terminator. If the buffer is not
+// large enough to accommodate the formatted string without truncation, they
+// return the number of characters that would be in the fully-formatted string
+// (vsnprintf, and vswprintf on Windows), or -1 (vswprintf on POSIX platforms).
inline int vsnprintfT(char* buffer,
size_t buf_size,
const char* format,
@@ -789,46 +787,69 @@ static void StringAppendVT(
va_list ap) {
// First try with a small fixed size buffer.
- // This buffer size should be kept in sync with StringUtilTest.GrowBoundary.
- const int kStackLength = 1024;
- char_type stack_buf[kStackLength];
-
- // It's possible for methods that use a va_list to invalidate the data in it
- // upon use. The fix is to make a copy of the structure before using it and
- // use that copy instead. It is not guaranteed that assignment is a copy, and
- // va_copy is not supported by VC, so the UnitTest tests this capability.
- va_list backup_ap = ap;
- int result = vsnprintfT(stack_buf, kStackLength, format, backup_ap);
+ // This buffer size should be kept in sync with StringUtilTest.GrowBoundary
+ // and StringUtilTest.StringPrintfBounds.
+ char_type stack_buf[1024];
+
+ va_list backup_ap;
+ base::va_copy(backup_ap, ap);
+
+#if !defined(OS_WIN)
+ errno = 0;
+#endif
+ int result = vsnprintfT(stack_buf, arraysize(stack_buf), format, backup_ap);
va_end(backup_ap);
- if (result >= 0 && result < kStackLength) {
+ if (result >= 0 && result < static_cast<int>(arraysize(stack_buf))) {
// It fit.
dst->append(stack_buf, result);
return;
}
- int mem_length = result;
+ // Repeatedly increase buffer size until it fits.
+ int mem_length = arraysize(stack_buf);
+ while (true) {
+ if (result < 0) {
+#if !defined(OS_WIN)
+ // On Windows, vsnprintfT always returns the number of characters in a
+ // fully-formatted string, so if we reach this point, something else is
+ // wrong and no amount of buffer-doubling is going to fix it.
+ if (errno != 0 && errno != EOVERFLOW)
+#endif
+ {
+ // If an error other than overflow occurred, it's never going to work.
+ DLOG(WARNING) << "Unable to printf the requested string due to error.";
+ return;
+ }
+ // Try doubling the buffer size.
+ mem_length *= 2;
+ } else {
+ // We need exactly "result + 1" characters.
+ mem_length = result + 1;
+ }
- // vsnprintfT may have failed for some reason other than an insufficient
- // buffer, such as an invalid characer. Check that the requested buffer
- // size is smaller than what was already attempted
- if (mem_length < 0 || mem_length < kStackLength) {
- DLOG(WARNING) << "Unable to compute size of the requested string.";
- return;
- }
+ if (mem_length > 32 * 1024 * 1024) {
+ // That should be plenty, don't try anything larger. This protects
+ // against huge allocations when using vsnprintfT implementations that
+ // return -1 for reasons other than overflow without setting errno.
+ DLOG(WARNING) << "Unable to printf the requested string due to size.";
+ return;
+ }
- mem_length++; // Include the NULL terminator.
- scoped_ptr<char_type> mem_buf(new char_type[mem_length]);
+ std::vector<char_type> mem_buf(mem_length);
- // Do the printf.
- result = vsnprintfT(mem_buf.get(), mem_length, format, ap);
- DCHECK(result < mem_length);
- if (result < 0) {
- DLOG(WARNING) << "Unable to printf the requested string.";
- return;
- }
+ // Restore the va_list before we use it again.
+ base::va_copy(backup_ap, ap);
- dst->append(mem_buf.get(), result);
+ result = vsnprintfT(&mem_buf[0], mem_length, format, ap);
+ va_end(backup_ap);
+
+ if ((result >= 0) && (result < mem_length)) {
+ // It fit.
+ dst->append(&mem_buf[0], result);
+ return;
+ }
+ }
}
std::string Uint64ToString(uint64 value) {
diff --git a/base/string_util.h b/base/string_util.h
index 46269ff..2f0f3d6 100644
--- a/base/string_util.h
+++ b/base/string_util.h
@@ -53,11 +53,15 @@ namespace base {
// s2 > s1 according to a lexicographic comparison.
int strncasecmp(const char* s1, const char* s2, size_t count);
-// Wrapper for vsnprintf that always NUL-terminates and always returns the
+// Wrapper for vsnprintf that always null-terminates and always returns the
// number of characters that would be in an untruncated formatted
// string, even when truncation occurs.
int vsnprintf(char* buffer, size_t size, const char* format, va_list arguments);
+// vswprintf always null-terminates, but when truncation occurs, it will either
+// return -1 or the number of characters that would be in an untruncated
+// formatted string. The actual return value depends on the underlying
+// C library's vswprintf implementation.
int vswprintf(wchar_t* buffer, size_t size,
const wchar_t* format, va_list arguments);
diff --git a/base/string_util_icu.cc b/base/string_util_icu.cc
index 2ab933a0..fca020b 100644
--- a/base/string_util_icu.cc
+++ b/base/string_util_icu.cc
@@ -190,7 +190,7 @@ bool WideToUTF8(const wchar_t* src, size_t src_len, std::string* output) {
// character, assume the rest will be ASCII and use a buffer size the same as
// the input. When it's not ASCII, assume 3-bytes per character as the
// starting point. This will be resized internally later if it's too small.
- if (src[0] < 0x80)
+ if (static_cast<uint32>(src[0]) < 0x80)
output->reserve(src_len);
else
output->reserve(src_len * 3);
@@ -217,7 +217,7 @@ bool UTF8ToWide(const char* src, size_t src_len, std::wstring* output) {
// the input. When it's not ASCII, assume the UTF-8 takes 2 bytes per
// character (this is more conservative than 3 which we use above when
// converting the other way).
- if (src[0] < 0x80)
+ if (static_cast<unsigned char>(src[0]) < 0x80)
output->reserve(src_len);
else
output->reserve(src_len / 2);
@@ -316,7 +316,7 @@ bool WideToCodepage(const std::wstring& wide,
#elif defined(WCHAR_T_IS_UTF32)
// When wchar_t is wider than UChar (16 bits), transform |wide| into a
// UChar* string. Size the UChar* buffer to be large enough to hold twice
- // as many UTF-16 code points as there are UCS-4 characters, in case each
+ // as many UTF-16 code points as there are UTF-16 characters, in case each
// character translates to a UTF-16 surrogate pair, and leave room for a NUL
// terminator.
std::vector<UChar> wide_uchar(wide.length() * 2 + 1);
diff --git a/base/string_util_unittest.cc b/base/string_util_unittest.cc
index 9b0f9b6..fce9d21 100644
--- a/base/string_util_unittest.cc
+++ b/base/string_util_unittest.cc
@@ -27,9 +27,11 @@
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
-#include <sstream>
+#include <math.h>
#include <stdarg.h>
+#include <sstream>
+
#include "base/basictypes.h"
#include "base/logging.h"
#include "base/string_util.h"
@@ -76,7 +78,7 @@ static const struct trim_case_ascii {
TEST(StringUtilTest, TrimWhitespace) {
std::wstring output; // Allow contents to carry over to next testcase
- for (int i = 0; i < arraysize(trim_cases); ++i) {
+ for (size_t i = 0; i < arraysize(trim_cases); ++i) {
const trim_case& value = trim_cases[i];
EXPECT_EQ(value.return_value,
TrimWhitespace(value.input, value.positions, &output));
@@ -94,7 +96,7 @@ TEST(StringUtilTest, TrimWhitespace) {
EXPECT_EQ(L"", output);
std::string output_ascii;
- for (int i = 0; i < arraysize(trim_cases_ascii); ++i) {
+ for (size_t i = 0; i < arraysize(trim_cases_ascii); ++i) {
const trim_case_ascii& value = trim_cases_ascii[i];
EXPECT_EQ(value.return_value,
TrimWhitespace(value.input, value.positions, &output_ascii));
@@ -129,7 +131,7 @@ static const struct collapse_case {
};
TEST(StringUtilTest, CollapseWhitespace) {
- for (int i = 0; i < arraysize(collapse_cases); ++i) {
+ for (size_t i = 0; i < arraysize(collapse_cases); ++i) {
const collapse_case& value = collapse_cases[i];
EXPECT_EQ(value.output, CollapseWhitespace(value.input, value.trim));
}
@@ -148,23 +150,25 @@ static const wchar_t* const kConvertRoundtripCases[] = {
L"\x0020\x0440\x0443\x0441\x0441\x043a\x043e\x043c",
// "전체서비스"
L"\xc804\xccb4\xc11c\xbe44\xc2a4",
- // ????? (Mathematical Alphanumeric Symbols (U+011d40 - U+011d44 : A,B,C,D,E)
- L"\xd807\xdd40\xd807\xdd41\xd807\xdd42\xd807\xdd43\xd807\xdd44",
- // Test a character that takes more than 16-bits. This will depend on whether
+ // Test characters that take more than 16 bits. This will depend on whether
// wchar_t is 16 or 32 bits.
- #if defined(WCHAR_T_IS_UTF16)
- L"\xd800\xdf00",
- #elif defined(WCHAR_T_IS_UTF32)
- "\x10300,
- #endif
+#if defined(WCHAR_T_IS_UTF16)
+ L"\xd800\xdf00",
+ // ????? (Mathematical Alphanumeric Symbols (U+011d40 - U+011d44 : A,B,C,D,E)
+ L"\xd807\xdd40\xd807\xdd41\xd807\xdd42\xd807\xdd43\xd807\xdd44",
+#elif defined(WCHAR_T_IS_UTF32)
+ L"\x10300",
+ // ????? (Mathematical Alphanumeric Symbols (U+011d40 - U+011d44 : A,B,C,D,E)
+ L"\x11d40\x11d41\x11d42\x11d43\x11d44",
+#endif
};
TEST(StringUtilTest, ConvertUTF8AndWide) {
// we round-trip all the wide strings through UTF-8 to make sure everything
// agrees on the conversion. This uses the stream operators to test them
// simultaneously.
- for (int i = 0; i < arraysize(kConvertRoundtripCases); ++i) {
+ for (size_t i = 0; i < arraysize(kConvertRoundtripCases); ++i) {
std::ostringstream utf8;
utf8 << WideToUTF8(kConvertRoundtripCases[i]);
std::wostringstream wide;
@@ -209,7 +213,7 @@ TEST(StringUtilTest, ConvertUTF8ToWide) {
#endif
};
- for (int i = 0; i < arraysize(convert_cases); i++) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(convert_cases); i++) {
std::wstring converted;
EXPECT_EQ(convert_cases[i].success,
UTF8ToWide(convert_cases[i].utf8,
@@ -222,18 +226,18 @@ TEST(StringUtilTest, ConvertUTF8ToWide) {
// Manually test an embedded NULL.
std::wstring converted;
EXPECT_TRUE(UTF8ToWide("\00Z\t", 3, &converted));
- ASSERT_EQ(3, converted.length());
+ ASSERT_EQ(static_cast<std::wstring::size_type>(3), converted.length());
EXPECT_EQ(0, converted[0]);
EXPECT_EQ('Z', converted[1]);
EXPECT_EQ('\t', converted[2]);
// Make sure that conversion replaces, not appends.
EXPECT_TRUE(UTF8ToWide("B", 1, &converted));
- ASSERT_EQ(1, converted.length());
+ ASSERT_EQ(static_cast<std::wstring::size_type>(1), converted.length());
EXPECT_EQ('B', converted[0]);
}
-#if defined(WCHAR_T_IS_UTf16)
+#if defined(WCHAR_T_IS_UTF16)
// This test is only valid when wchar_t == UTF-16.
TEST(StringUtilTest, ConvertUTF16ToUTF8) {
struct UTF16ToUTF8Case {
@@ -268,7 +272,7 @@ TEST(StringUtilTest, ConvertUTF16ToUTF8) {
// This test is only valid when wchar_t == UTF-32.
TEST(StringUtilTest, ConvertUTF32ToUTF8) {
struct UTF8ToWideCase {
- const wchar_t* ucs4;
+ const wchar_t* utf32;
const char* utf8;
bool success;
} convert_cases[] = {
@@ -277,17 +281,17 @@ TEST(StringUtilTest, ConvertUTF32ToUTF8) {
// Test a non-BMP character.
{L"A\x10300z", "A\xF0\x90\x8C\x80z", true},
// Invalid Unicode code points.
- {L"\xffffHello", "Hello, false", false},
- {L"\xfffffffHello", "Hello, false", false},
+ {L"\xffffHello", "Hello", false},
+ {L"\xfffffffHello", "Hello", false},
// The first character is a truncated UTF-16 character.
{L"\xd800\x597d", "\xe5\xa5\xbd", false},
- }
+ };
- for (int i = 0; i < arraysize(convert_cases); i++) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(convert_cases); i++) {
std::string converted;
EXPECT_EQ(convert_cases[i].success,
- WideToUTF8(convert_cases[i].utf16,
- wcslen(convert_cases[i].utf16),
+ WideToUTF8(convert_cases[i].utf32,
+ wcslen(convert_cases[i].utf32),
&converted));
std::string expected(convert_cases[i].utf8);
EXPECT_EQ(expected, converted);
@@ -321,7 +325,7 @@ TEST(StringUtilTest, ConvertMultiString) {
TEST(StringUtilTest, ConvertCodepageUTF8) {
// Make sure WideToCodepage works like WideToUTF8.
- for (int i = 0; i < arraysize(kConvertRoundtripCases); ++i) {
+ for (size_t i = 0; i < arraysize(kConvertRoundtripCases); ++i) {
std::string expected(WideToUTF8(kConvertRoundtripCases[i]));
std::string utf8;
EXPECT_TRUE(WideToCodepage(kConvertRoundtripCases[i], kCodepageUTF8,
@@ -428,7 +432,7 @@ TEST(StringUtilTest, ConvertBetweenCodepageAndWide) {
L"\x0E04\x0E23\x0e31\x0E1A"},
};
- for (int i = 0; i < arraysize(kConvertCodepageCases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kConvertCodepageCases); ++i) {
std::wstring wide;
bool success = CodepageToWide(kConvertCodepageCases[i].encoded,
kConvertCodepageCases[i].codepage_name,
@@ -470,7 +474,7 @@ TEST(StringUtilTest, ConvertBetweenCodepageAndWide) {
EXPECT_TRUE(WideToCodepage(L"a\xd800z", "iso-8859-1",
OnStringUtilConversionError::SKIP, &encoded));
EXPECT_STREQ("az", encoded.c_str());
-#endif // WCHAR_T_IS_UTf16
+#endif // WCHAR_T_IS_UTF16
// Invalid characters should fail.
EXPECT_TRUE(WideToCodepage(L"a\xffffz", "iso-8859-1",
@@ -495,7 +499,7 @@ TEST(StringUtilTest, ConvertASCII) {
L"0123ABCDwxyz \a\b\t\r\n!+,.~"
};
- for (int i = 0; i < arraysize(char_cases); ++i) {
+ for (size_t i = 0; i < arraysize(char_cases); ++i) {
EXPECT_TRUE(IsStringASCII(char_cases[i]));
std::wstring wide = ASCIIToWide(char_cases[i]);
EXPECT_EQ(wchar_cases[i], wide);
@@ -519,9 +523,11 @@ TEST(StringUtilTest, ConvertASCII) {
const int length_with_nul = arraysize(chars_with_nul) - 1;
std::string string_with_nul(chars_with_nul, length_with_nul);
std::wstring wide_with_nul = ASCIIToWide(string_with_nul);
- EXPECT_EQ(length_with_nul, wide_with_nul.length());
+ EXPECT_EQ(static_cast<std::wstring::size_type>(length_with_nul),
+ wide_with_nul.length());
std::string narrow_with_nul = WideToASCII(wide_with_nul);
- EXPECT_EQ(length_with_nul, narrow_with_nul.length());
+ EXPECT_EQ(static_cast<std::string::size_type>(length_with_nul),
+ narrow_with_nul.length());
EXPECT_EQ(0, string_with_nul.compare(narrow_with_nul));
}
@@ -536,7 +542,7 @@ static const struct {
};
TEST(StringUtilTest, LowerCaseEqualsASCII) {
- for (int i = 0; i < arraysize(lowercase_cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(lowercase_cases); ++i) {
EXPECT_TRUE(LowerCaseEqualsASCII(lowercase_cases[i].src_w,
lowercase_cases[i].dst));
EXPECT_TRUE(LowerCaseEqualsASCII(lowercase_cases[i].src_a,
@@ -560,7 +566,7 @@ TEST(StringUtilTest, GetByteDisplayUnits) {
#endif
};
- for (int i = 0; i < arraysize(cases); ++i)
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i)
EXPECT_EQ(cases[i].expected, GetByteDisplayUnits(cases[i].bytes));
}
@@ -590,7 +596,7 @@ TEST(StringUtilTest, FormatBytes) {
#endif
};
- for (int i = 0; i < arraysize(cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) {
EXPECT_EQ(cases[i].expected,
FormatBytes(cases[i].bytes, cases[i].units, false));
EXPECT_EQ(cases[i].expected_with_units,
@@ -620,7 +626,7 @@ TEST(StringUtilTest, ReplaceSubstringsAfterOffset) {
{L"abababab", 2, L"ab", L"c", L"abccc"},
};
- for (int i = 0; i < arraysize(cases); i++) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); i++) {
std::wstring str(cases[i].str);
ReplaceSubstringsAfterOffset(&str, cases[i].start_offset,
cases[i].find_this, cases[i].replace_with);
@@ -640,7 +646,7 @@ TEST(StringUtilTest, IntToString) {
{INT_MIN, "-2147483648"},
};
- for (int i = 0; i < arraysize(cases); ++i)
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i)
EXPECT_EQ(cases[i].output, IntToString(cases[i].input));
}
@@ -655,7 +661,7 @@ TEST(StringUtilTest, Uint64ToString) {
{kuint64max, "18446744073709551615"},
};
- for (int i = 0; i < arraysize(cases); ++i)
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i)
EXPECT_EQ(cases[i].output, Uint64ToString(cases[i].input));
}
@@ -689,7 +695,7 @@ TEST(StringUtilTest, StringToInt) {
{"99999999999", INT_MAX, false},
};
- for (int i = 0; i < arraysize(cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) {
EXPECT_EQ(cases[i].output, StringToInt(cases[i].input));
int output;
EXPECT_EQ(cases[i].success, StringToInt(cases[i].input, &output));
@@ -753,7 +759,7 @@ TEST(StringUtilTest, StringToInt64) {
{"99999999999999999999", kint64max, false},
};
- for (int i = 0; i < arraysize(cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) {
EXPECT_EQ(cases[i].output, StringToInt64(cases[i].input));
int64 output;
EXPECT_EQ(cases[i].success, StringToInt64(cases[i].input, &output));
@@ -814,7 +820,7 @@ TEST(StringUtilTest, HexStringToInt) {
{"", 0, false},
};
- for (int i = 0; i < arraysize(cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) {
EXPECT_EQ(cases[i].output, HexStringToInt(cases[i].input));
int output;
EXPECT_EQ(cases[i].success, HexStringToInt(cases[i].input, &output));
@@ -874,7 +880,7 @@ TEST(StringUtilTest, StringToDouble) {
{"", 0.0, false},
};
- for (int i = 0; i < arraysize(cases); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) {
EXPECT_DOUBLE_EQ(cases[i].output, StringToDouble(cases[i].input));
double output;
EXPECT_EQ(cases[i].success, StringToDouble(cases[i].input, &output));
@@ -908,7 +914,8 @@ static void VariableArgsFunc(const char* format, ...) {
va_list org;
va_start(org, format);
- va_list dup = org;
+ va_list dup;
+ va_copy(dup, org);
int i1 = va_arg(org, int);
int j1 = va_arg(org, int);
char* s1 = va_arg(org, char*);
@@ -940,7 +947,7 @@ TEST(StringUtilTest, StringPrintfEmptyFormat) {
TEST(StringUtilTest, StringPrintfMisc) {
EXPECT_EQ("123hello w", StringPrintf("%3d%2s %1c", 123, "hello", 'w'));
- EXPECT_EQ(L"123hello w", StringPrintf(L"%3d%2s %1c", 123, L"hello", 'w'));
+ EXPECT_EQ(L"123hello w", StringPrintf(L"%3d%2ls %1c", 123, L"hello", 'w'));
}
TEST(StringUtilTest, StringAppendfStringEmptyParam) {
@@ -969,7 +976,7 @@ TEST(StringUtilTest, StringAppendfString) {
EXPECT_EQ("Hello World", value);
std::wstring valuew(L"Hello");
- StringAppendF(&valuew, L" %s", L"World");
+ StringAppendF(&valuew, L" %ls", L"World");
EXPECT_EQ(L"Hello World", valuew);
}
@@ -988,11 +995,11 @@ TEST(StringUtilTest, StringAppendfInt) {
TEST(StringUtilTest, StringPrintfBounds) {
const int src_len = 1026;
char src[src_len];
- for (int i = 0; i < arraysize(src); i++)
+ for (size_t i = 0; i < arraysize(src); i++)
src[i] = 'A';
wchar_t srcw[src_len];
- for (int i = 0; i < arraysize(srcw); i++)
+ for (size_t i = 0; i < arraysize(srcw); i++)
srcw[i] = 'A';
for (int i = 1; i < 3; i++) {
@@ -1003,7 +1010,7 @@ TEST(StringUtilTest, StringPrintfBounds) {
srcw[src_len - i] = 0;
std::wstring outw;
- SStringPrintf(&outw, L"%s", srcw);
+ SStringPrintf(&outw, L"%ls", srcw);
EXPECT_STREQ(srcw, outw.c_str());
}
}
@@ -1011,7 +1018,7 @@ TEST(StringUtilTest, StringPrintfBounds) {
// Test very large sprintfs that will cause the buffer to grow.
TEST(StringUtilTest, Grow) {
char src[1026];
- for (int i = 0; i < arraysize(src); i++)
+ for (size_t i = 0; i < arraysize(src); i++)
src[i] = 'A';
src[1025] = 0;
@@ -1021,7 +1028,11 @@ TEST(StringUtilTest, Grow) {
SStringPrintf(&out, fmt, src, src, src, src, src, src, src);
char* ref = new char[320000];
+#if defined(OS_WIN)
sprintf_s(ref, 320000, fmt, src, src, src, src, src, src, src);
+#elif defined(OS_POSIX)
+ snprintf(ref, 320000, fmt, src, src, src, src, src, src, src);
+#endif
EXPECT_STREQ(ref, out.c_str());
delete ref;
@@ -1053,7 +1064,7 @@ TEST(StringUtilTest, Invalid) {
invalid[1] = 0;
std::wstring out;
- SStringPrintf(&out, L"%s", invalid);
+ SStringPrintf(&out, L"%ls", invalid);
EXPECT_STREQ(L"", out.c_str());
}
@@ -1062,50 +1073,50 @@ TEST(StringUtilTest, SplitString) {
std::vector<std::wstring> r;
SplitString(L"a,b,c", L',', &r);
- EXPECT_EQ(r.size(), 3);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(3), r.size());
EXPECT_EQ(r[0], L"a");
EXPECT_EQ(r[1], L"b");
EXPECT_EQ(r[2], L"c");
r.clear();
SplitString(L"a, b, c", L',', &r);
- EXPECT_EQ(r.size(), 3);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(3), r.size());
EXPECT_EQ(r[0], L"a");
EXPECT_EQ(r[1], L"b");
EXPECT_EQ(r[2], L"c");
r.clear();
SplitString(L"a,,c", L',', &r);
- EXPECT_EQ(r.size(), 3);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(3), r.size());
EXPECT_EQ(r[0], L"a");
EXPECT_EQ(r[1], L"");
EXPECT_EQ(r[2], L"c");
r.clear();
SplitString(L"", L'*', &r);
- EXPECT_EQ(r.size(), 1);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(1), r.size());
EXPECT_EQ(r[0], L"");
r.clear();
SplitString(L"foo", L'*', &r);
- EXPECT_EQ(r.size(), 1);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(1), r.size());
EXPECT_EQ(r[0], L"foo");
r.clear();
SplitString(L"foo ,", L',', &r);
- EXPECT_EQ(r.size(), 2);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(2), r.size());
EXPECT_EQ(r[0], L"foo");
EXPECT_EQ(r[1], L"");
r.clear();
SplitString(L",", L',', &r);
- EXPECT_EQ(r.size(), 2);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(2), r.size());
EXPECT_EQ(r[0], L"");
EXPECT_EQ(r[1], L"");
r.clear();
SplitString(L"\t\ta\t", L'\t', &r);
- EXPECT_EQ(r.size(), 4);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(4), r.size());
EXPECT_EQ(r[0], L"");
EXPECT_EQ(r[1], L"");
EXPECT_EQ(r[2], L"a");
@@ -1113,7 +1124,7 @@ TEST(StringUtilTest, SplitString) {
r.clear();
SplitStringDontTrim(L"\t\ta\t", L'\t', &r);
- EXPECT_EQ(r.size(), 4);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(4), r.size());
EXPECT_EQ(r[0], L"");
EXPECT_EQ(r[1], L"");
EXPECT_EQ(r[2], L"a");
@@ -1121,13 +1132,13 @@ TEST(StringUtilTest, SplitString) {
r.clear();
SplitString(L"\ta\t\nb\tcc", L'\n', &r);
- EXPECT_EQ(r.size(), 2);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(2), r.size());
EXPECT_EQ(r[0], L"a");
EXPECT_EQ(r[1], L"b\tcc");
r.clear();
SplitStringDontTrim(L"\ta\t\nb\tcc", L'\n', &r);
- EXPECT_EQ(r.size(), 2);
+ EXPECT_EQ(static_cast<std::vector<std::wstring>::size_type>(2), r.size());
EXPECT_EQ(r[0], L"\ta\t");
EXPECT_EQ(r[1], L"b\tcc");
r.clear();
@@ -1146,16 +1157,16 @@ TEST(StringUtilTest, GetStringFWithOffsets) {
ReplaceStringPlaceholders(L"Hello, $1. Your number is $2.", L"1", L"2",
&offsets);
- EXPECT_EQ(2, offsets.size());
- EXPECT_EQ(7, offsets[0]);
- EXPECT_EQ(25, offsets[1]);
+ EXPECT_EQ(static_cast<std::vector<size_t>::size_type>(2), offsets.size());
+ EXPECT_EQ(static_cast<size_t>(7), offsets[0]);
+ EXPECT_EQ(static_cast<size_t>(25), offsets[1]);
offsets.clear();
ReplaceStringPlaceholders(L"Hello, $2. Your number is $1.", L"1", L"2",
&offsets);
- EXPECT_EQ(2, offsets.size());
- EXPECT_EQ(25, offsets[0]);
- EXPECT_EQ(7, offsets[1]);
+ EXPECT_EQ(static_cast<std::vector<size_t>::size_type>(2), offsets.size());
+ EXPECT_EQ(static_cast<size_t>(25), offsets[0]);
+ EXPECT_EQ(static_cast<size_t>(7), offsets[1]);
offsets.clear();
}
@@ -1180,10 +1191,12 @@ TEST(StringUtilTest, SplitStringAlongWhitespace) {
{ L"b\tat", 2, L"b", L"at" },
{ L"b\t at", 2, L"b", L"at" },
};
- for (size_t i = 0; i < arraysize(data); ++i) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(data); ++i) {
std::vector<std::wstring> results;
SplitStringAlongWhitespace(data[i].input, &results);
- ASSERT_EQ(data[i].expected_result_count, results.size());
+ ASSERT_EQ(static_cast<std::vector<std::wstring>::size_type>(
+ data[i].expected_result_count),
+ results.size());
if (data[i].expected_result_count > 0)
ASSERT_EQ(data[i].output1, results[0]);
if (data[i].expected_result_count > 1)