| //===-- Unittests for wcstok ----------------------------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "hdr/types/size_t.h" |
| #include "hdr/types/wchar_t.h" |
| #include "src/wchar/wcstok.h" |
| #include "test/UnitTest/Test.h" |
| |
| TEST(LlvmLibcWCSTokReentrantTest, NoTokenFound) { |
| { // Empty source and delimiter string. |
| wchar_t empty[] = L""; |
| wchar_t *reserve = nullptr; |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(empty, L"", &reserve), nullptr); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(empty, L"", &reserve), nullptr); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L"", &reserve), nullptr); |
| } |
| { // Empty source and single character delimiter string. |
| wchar_t empty[] = L""; |
| wchar_t *reserve = nullptr; |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(empty, L"_", &reserve), nullptr); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(empty, L"_", &reserve), nullptr); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L"_", &reserve), nullptr); |
| } |
| { // Same character source and delimiter string. |
| wchar_t single[] = L"_"; |
| wchar_t *reserve = nullptr; |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(single, L"_", &reserve), nullptr); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(single, L"_", &reserve), nullptr); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L"_", &reserve), nullptr); |
| } |
| { // Multiple character source and single character delimiter string. |
| wchar_t multiple[] = L"1,2"; |
| wchar_t *reserve = nullptr; |
| wchar_t *tok = LIBC_NAMESPACE::wcstok(multiple, L":", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L','); |
| ASSERT_TRUE(tok[2] == L'2'); |
| ASSERT_TRUE(tok[3] == L'\0'); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| tok = LIBC_NAMESPACE::wcstok(multiple, L":", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L','); |
| ASSERT_TRUE(tok[2] == L'2'); |
| ASSERT_TRUE(tok[3] == L'\0'); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L":", &reserve), nullptr); |
| } |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, DelimiterAsFirstCharacterShouldBeIgnored) { |
| wchar_t src[] = L".123"; |
| wchar_t *reserve = nullptr; |
| wchar_t *tok = LIBC_NAMESPACE::wcstok(src, L".", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'3'); |
| ASSERT_TRUE(tok[3] == L'\0'); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| tok = LIBC_NAMESPACE::wcstok(src, L".", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'3'); |
| ASSERT_TRUE(tok[3] == L'\0'); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L".", &reserve), nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, DelimiterIsMiddleCharacter) { |
| wchar_t src[] = L"12,34"; |
| wchar_t *reserve = nullptr; |
| wchar_t *tok = LIBC_NAMESPACE::wcstok(src, L",", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'\0'); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| tok = LIBC_NAMESPACE::wcstok(src, L",", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'\0'); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L",", &reserve), nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, DelimiterAsLastCharacterShouldBeIgnored) { |
| wchar_t src[] = L"1234:"; |
| wchar_t *reserve = nullptr; |
| wchar_t *tok = LIBC_NAMESPACE::wcstok(src, L":", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'3'); |
| ASSERT_TRUE(tok[3] == L'4'); |
| ASSERT_TRUE(tok[4] == L'\0'); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| tok = LIBC_NAMESPACE::wcstok(src, L":", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'3'); |
| ASSERT_TRUE(tok[3] == L'4'); |
| ASSERT_TRUE(tok[4] == L'\0'); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L":", &reserve), nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, ShouldNotGoPastNullTerminator) { |
| wchar_t src[] = {L'1', L'2', L'\0', L',', L'3'}; |
| wchar_t *reserve = nullptr; |
| wchar_t *tok = LIBC_NAMESPACE::wcstok(src, L",", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'\0'); |
| // Another call to ensure that 'reserve' is not in a bad state. |
| tok = LIBC_NAMESPACE::wcstok(src, L",", &reserve); |
| ASSERT_TRUE(tok[0] == L'1'); |
| ASSERT_TRUE(tok[1] == L'2'); |
| ASSERT_TRUE(tok[2] == L'\0'); |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(nullptr, L",", &reserve), nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, |
| ShouldReturnNullptrWhenBothSrcAndSaveptrAreNull) { |
| wchar_t *src = nullptr; |
| wchar_t *reserve = nullptr; |
| // Ensure that instead of crashing if src and reserve are null, nullptr is |
| // returned |
| ASSERT_EQ(LIBC_NAMESPACE::wcstok(src, L",", &reserve), nullptr); |
| // And that neither src nor reserve are changed when that happens |
| ASSERT_EQ(src, nullptr); |
| ASSERT_EQ(reserve, nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, |
| SubsequentCallsShouldFindFollowingDelimiters) { |
| wchar_t src[] = L"12,34.56"; |
| wchar_t *reserve = nullptr; |
| wchar_t *token = LIBC_NAMESPACE::wcstok(src, L",.", &reserve); |
| ASSERT_TRUE(token[0] == L'1'); |
| ASSERT_TRUE(token[1] == L'2'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| |
| token = LIBC_NAMESPACE::wcstok(nullptr, L",.", &reserve); |
| ASSERT_TRUE(token[0] == L'3'); |
| ASSERT_TRUE(token[1] == L'4'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| |
| token = LIBC_NAMESPACE::wcstok(nullptr, L",.", &reserve); |
| ASSERT_TRUE(token[0] == L'5'); |
| ASSERT_TRUE(token[1] == L'6'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| token = LIBC_NAMESPACE::wcstok(nullptr, L"_:,_", &reserve); |
| ASSERT_EQ(token, nullptr); |
| // Subsequent calls after hitting the end of the string should also return |
| // nullptr. |
| token = LIBC_NAMESPACE::wcstok(nullptr, L"_:,_", &reserve); |
| ASSERT_EQ(token, nullptr); |
| } |
| |
| TEST(LlvmLibcWCSTokReentrantTest, DelimitersShouldNotBeIncludedInToken) { |
| wchar_t src[] = L"__ab__:_cd__:__ef__:__"; |
| wchar_t *reserve = nullptr; |
| wchar_t *token = LIBC_NAMESPACE::wcstok(src, L"_:", &reserve); |
| ASSERT_TRUE(token[0] == L'a'); |
| ASSERT_TRUE(token[1] == L'b'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| |
| token = LIBC_NAMESPACE::wcstok(nullptr, L":_", &reserve); |
| ASSERT_TRUE(token[0] == L'c'); |
| ASSERT_TRUE(token[1] == L'd'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| |
| token = LIBC_NAMESPACE::wcstok(nullptr, L"_:,", &reserve); |
| ASSERT_TRUE(token[0] == L'e'); |
| ASSERT_TRUE(token[1] == L'f'); |
| ASSERT_TRUE(token[2] == L'\0'); |
| |
| token = LIBC_NAMESPACE::wcstok(nullptr, L"_:,_", &reserve); |
| ASSERT_EQ(token, nullptr); |
| } |