diff options
author | mmentovai@google.com <mmentovai@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-08-14 01:25:32 +0000 |
---|---|---|
committer | mmentovai@google.com <mmentovai@google.com@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-08-14 01:25:32 +0000 |
commit | 44cd16fcc85697b5094250a89decaab5c692b392 (patch) | |
tree | 37e4fb0dbee34f2cc673c287b54515a71b1f9629 /base | |
parent | 5f800edb90578f08b3462e1dc96b78bf0ec56d36 (diff) | |
download | chromium_src-44cd16fcc85697b5094250a89decaab5c692b392.zip chromium_src-44cd16fcc85697b5094250a89decaab5c692b392.tar.gz chromium_src-44cd16fcc85697b5094250a89decaab5c692b392.tar.bz2 |
DCHECK vswprintf format string cross-platform portability. Use %ls, not %s, for wchar_t* fields.
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@849 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'base')
-rw-r--r-- | base/string_util.cc | 42 | ||||
-rw-r--r-- | base/string_util.h | 23 | ||||
-rw-r--r-- | base/string_util_posix.h | 3 | ||||
-rw-r--r-- | base/string_util_unittest.cc | 36 | ||||
-rw-r--r-- | base/string_util_win.h | 10 |
5 files changed, 109 insertions, 5 deletions
diff --git a/base/string_util.cc b/base/string_util.cc index 7208cd0..0a930d9 100644 --- a/base/string_util.cc +++ b/base/string_util.cc @@ -259,6 +259,48 @@ class WStringToDoubleTraits { } // namespace +namespace base { + +bool IsWprintfFormatPortable(const wchar_t* format) { + for (const wchar_t* position = format; *position != '\0'; ++position) { + + if (*position == '%') { + bool in_specification = true; + bool modifier_l = false; + while (in_specification) { + // Eat up characters until reaching a known specifier. + if (*++position == '\0') { + // The format string ended in the middle of a specification. Call + // it portable because no unportable specifications were found. The + // string is equally broken on all platforms. + return true; + } + + if (*position == 'l') { + // 'l' is the only thing that can save the 's' and 'c' specifiers. + modifier_l = true; + } else if (((*position == 's' || *position == 'c') && !modifier_l) || + *position == 'S' || *position == 'C' || *position == 'F' || + *position == 'D' || *position == 'O' || *position == 'U') { + // Not portable. + return false; + } + + if (wcschr(L"diouxXeEfgGaAcspn%", *position)) { + // Portable, keep scanning the rest of the format string. + in_specification = false; + } + } + } + + } + + return true; +} + +} // namespace base + + const std::string& EmptyString() { return *Singleton<std::string>::get(); } diff --git a/base/string_util.h b/base/string_util.h index 5faa9cf..ca0496c 100644 --- a/base/string_util.h +++ b/base/string_util.h @@ -92,6 +92,29 @@ inline int swprintf(wchar_t* buffer, size_t size, const wchar_t* format, ...) { size_t strlcpy(char* dst, const char* src, size_t dst_size); size_t wcslcpy(wchar_t* dst, const wchar_t* src, size_t dst_size); +// Scan a wprintf format string to determine whether it's portable across a +// variety of systems. This function only checks that the conversion +// specifiers used by the format string are supported and have the same meaning +// on a variety of systems. It doesn't check for other errors that might occur +// within a format string. +// +// Nonportable conversion specifiers for wprintf are: +// - 's' and 'c' without an 'l' length modifier. %s and %c operate on char +// data on all systems except Windows, which treat them as wchar_t data. +// Use %ls and %lc for wchar_t data instead. +// - 'S' and 'C', which operate on wchar_t data on all systems except Windows, +// which treat them as char data. Use %ls and %lc for wchar_t data +// instead. +// - 'F', which is not identified by Windows wprintf documentation. +// - 'D', 'O', and 'U', which are deprecated and not available on all systems. +// Use %ld, %lo, and %lu instead. +// +// Note that there is no portable conversion specifier for char data when +// working with wprintf. +// +// This function is intended to be called from base::vswprintf. +bool IsWprintfFormatPortable(const wchar_t* format); + } // namespace base #if defined(OS_WIN) diff --git a/base/string_util_posix.h b/base/string_util_posix.h index b3df2ad..a9e3646 100644 --- a/base/string_util_posix.h +++ b/base/string_util_posix.h @@ -35,6 +35,8 @@ #include <string.h> #include <wchar.h> +#include "base/logging.h" + namespace base { inline int strncasecmp(const char* string1, const char* string2, size_t count) { @@ -48,6 +50,7 @@ inline int vsnprintf(char* buffer, size_t size, inline int vswprintf(wchar_t* buffer, size_t size, const wchar_t* format, va_list arguments) { + DCHECK(IsWprintfFormatPortable(format)); return ::vswprintf(buffer, size, format, arguments); } diff --git a/base/string_util_unittest.cc b/base/string_util_unittest.cc index 7cc9764..c286222 100644 --- a/base/string_util_unittest.cc +++ b/base/string_util_unittest.cc @@ -978,7 +978,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%2ls %1c", 123, L"hello", 'w')); + EXPECT_EQ(L"123hello w", StringPrintf(L"%3d%2ls %1lc", 123, L"hello", 'w')); } TEST(StringUtilTest, StringAppendfStringEmptyParam) { @@ -997,7 +997,7 @@ TEST(StringUtilTest, StringAppendfEmptyString) { EXPECT_EQ("Hello", value); std::wstring valuew(L"Hello"); - StringAppendF(&valuew, L"%s", L""); + StringAppendF(&valuew, L"%ls", L""); EXPECT_EQ(L"Hello", valuew); } @@ -1306,3 +1306,35 @@ TEST(StringUtilTest, LcpyTest) { EXPECT_EQ(0, memcmp(wdst, L"ab", sizeof(wchar_t) * 3)); } } + +TEST(StringUtilTest, WprintfFormatPortabilityTest) { + struct TestData { + const wchar_t* input; + bool portable; + } cases[] = { + { L"%ls", true }, + { L"%s", false }, + { L"%S", false }, + { L"%lS", false }, + { L"Hello, %s", false }, + { L"%lc", true }, + { L"%c", false }, + { L"%C", false }, + { L"%lC", false }, + { L"%ls %s", false }, + { L"%s %ls", false }, + { L"%s %ls %s", false }, + { L"%f", true }, + { L"%f %F", false }, + { L"%d %D", false }, + { L"%o %O", false }, + { L"%u %U", false }, + { L"%f %d %o %u", true }, + { L"%-8d (%02.1f%)", true }, + { L"% 10s", false }, + { L"% 10ls", true } + }; + for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) { + EXPECT_EQ(cases[i].portable, base::IsWprintfFormatPortable(cases[i].input)); + } +} diff --git a/base/string_util_win.h b/base/string_util_win.h index bec84042..80ebd38 100644 --- a/base/string_util_win.h +++ b/base/string_util_win.h @@ -27,14 +27,16 @@ // (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_STRING_UTIL_WIN_H__ -#define BASE_STRING_UTIL_WIN_H__ +#ifndef BASE_STRING_UTIL_WIN_H_ +#define BASE_STRING_UTIL_WIN_H_ #include <stdarg.h> #include <stdio.h> #include <string.h> #include <wchar.h> +#include "base/logging.h" + namespace base { inline int strncasecmp(const char* s1, const char* s2, size_t count) { @@ -51,6 +53,8 @@ inline int vsnprintf(char* buffer, size_t size, inline int vswprintf(wchar_t* buffer, size_t size, const wchar_t* format, va_list arguments) { + DCHECK(IsWprintfFormatPortable(format)); + int length = _vsnwprintf_s(buffer, size, size - 1, format, arguments); if (length < 0) return _vscwprintf(format, arguments); @@ -59,4 +63,4 @@ inline int vswprintf(wchar_t* buffer, size_t size, } // namespace base -#endif // BASE_STRING_UTIL_WIN_H__ +#endif // BASE_STRING_UTIL_WIN_H_ |