1b6a20a49Scgyurgyik //===-- Unittests for memchr ----------------------------------------------===//
2b6a20a49Scgyurgyik //
3b6a20a49Scgyurgyik // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4b6a20a49Scgyurgyik // See https://llvm.org/LICENSE.txt for license information.
5b6a20a49Scgyurgyik // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6b6a20a49Scgyurgyik //
7b6a20a49Scgyurgyik //===----------------------------------------------------------------------===//
8b6a20a49Scgyurgyik
9b6a20a49Scgyurgyik #include "src/string/memchr.h"
10af1315c2SSiva Chandra Reddy #include "test/UnitTest/Test.h"
11b6a20a49Scgyurgyik #include <stddef.h>
12b6a20a49Scgyurgyik
13b6a20a49Scgyurgyik // A helper function that calls memchr and abstracts away the explicit cast for
14b6a20a49Scgyurgyik // readability purposes.
call_memchr(const void * src,int c,size_t size)15b6a20a49Scgyurgyik const char *call_memchr(const void *src, int c, size_t size) {
16*b6bc9d72SGuillaume Chatelet return reinterpret_cast<const char *>(LIBC_NAMESPACE::memchr(src, c, size));
17b6a20a49Scgyurgyik }
18b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsCharacterAfterNullTerminator)191df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsCharacterAfterNullTerminator) {
20b6a20a49Scgyurgyik // memchr should continue searching after a null terminator.
21b6a20a49Scgyurgyik const size_t size = 5;
22b6a20a49Scgyurgyik const unsigned char src[size] = {'a', '\0', 'b', 'c', '\0'};
23b6a20a49Scgyurgyik // Should return 'b', 'c', '\0' even when after null terminator.
24b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, 'b', size), "bc");
25b6a20a49Scgyurgyik }
26b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsCharacterInNonNullTerminatedCollection)271df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsCharacterInNonNullTerminatedCollection) {
28b6a20a49Scgyurgyik const size_t size = 3;
29b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c'};
30b6a20a49Scgyurgyik // Should return 'b', 'c'.
31b6a20a49Scgyurgyik const char *ret = call_memchr(src, 'b', size);
32b6a20a49Scgyurgyik ASSERT_EQ(ret[0], 'b');
33b6a20a49Scgyurgyik ASSERT_EQ(ret[1], 'c');
34b6a20a49Scgyurgyik }
35b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsFirstCharacter)361df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsFirstCharacter) {
37b6a20a49Scgyurgyik const size_t size = 6;
38b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
39b6a20a49Scgyurgyik // Should return original array since 'a' is the first character.
40b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, 'a', size), "abcde");
41b6a20a49Scgyurgyik }
42b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsMiddleCharacter)431df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsMiddleCharacter) {
44b6a20a49Scgyurgyik const size_t size = 6;
45b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
46b6a20a49Scgyurgyik // Should return characters after (and including) 'c'.
47b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, 'c', size), "cde");
48b6a20a49Scgyurgyik }
49b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsLastCharacterThatIsNotNullTerminator)501df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsLastCharacterThatIsNotNullTerminator) {
51b6a20a49Scgyurgyik const size_t size = 6;
52b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
53b6a20a49Scgyurgyik // Should return 'e' and null-terminator.
54b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, 'e', size), "e");
55b6a20a49Scgyurgyik }
56b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,FindsNullTerminator)571df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, FindsNullTerminator) {
58b6a20a49Scgyurgyik const size_t size = 6;
59b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
60b6a20a49Scgyurgyik // Should return null terminator.
61b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, '\0', size), "");
62b6a20a49Scgyurgyik }
63b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,CharacterNotWithinStringShouldReturnNullptr)641df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, CharacterNotWithinStringShouldReturnNullptr) {
65b6a20a49Scgyurgyik const size_t size = 4;
66b6a20a49Scgyurgyik const unsigned char src[size] = {'1', '2', '3', '?'};
67b6a20a49Scgyurgyik // Since 'z' is not within 'characters', should return nullptr.
68b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, 'z', size), nullptr);
69b6a20a49Scgyurgyik }
70b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,CharacterNotWithinSizeShouldReturnNullptr)711df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, CharacterNotWithinSizeShouldReturnNullptr) {
72b6a20a49Scgyurgyik const unsigned char src[5] = {'1', '2', '3', '4', '\0'};
73b6a20a49Scgyurgyik // Since '4' is not the first or second character, this should return nullptr.
74b6a20a49Scgyurgyik const size_t size = 2;
75b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(src, '4', size), nullptr);
76b6a20a49Scgyurgyik }
77b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,TheSourceShouldNotChange)781df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, TheSourceShouldNotChange) {
79b6a20a49Scgyurgyik const size_t size = 6;
80b6a20a49Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
81b6a20a49Scgyurgyik const char *src_copy = reinterpret_cast<const char *>(src);
82b6a20a49Scgyurgyik // When the character is found, the source string should not change.
83*b6bc9d72SGuillaume Chatelet LIBC_NAMESPACE::memchr(src, 'd', size);
84b6a20a49Scgyurgyik ASSERT_STREQ(reinterpret_cast<const char *>(src), src_copy);
85b6a20a49Scgyurgyik // Same case for when the character is not found.
86*b6bc9d72SGuillaume Chatelet LIBC_NAMESPACE::memchr(src, 'z', size);
87b6a20a49Scgyurgyik ASSERT_STREQ(reinterpret_cast<const char *>(src), src_copy);
88b6a20a49Scgyurgyik }
89b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,ShouldFindFirstOfDuplicates)901df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, ShouldFindFirstOfDuplicates) {
91b6a20a49Scgyurgyik const size_t size = 12; // 11 characters + null terminator.
92b6a20a49Scgyurgyik const char *dups = "abc1def1ghi";
93b6a20a49Scgyurgyik // 1 is duplicated in 'dups', but it should find the first copy.
94b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(dups, '1', size), "1def1ghi");
95b6a20a49Scgyurgyik }
96b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,EmptyStringShouldOnlyMatchNullTerminator)971df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, EmptyStringShouldOnlyMatchNullTerminator) {
98b6a20a49Scgyurgyik const size_t size = 1; // Null terminator.
99b6a20a49Scgyurgyik const char *empty_string = "";
100b6a20a49Scgyurgyik // Null terminator should match.
101b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(empty_string, '\0', size), "");
102b6a20a49Scgyurgyik // All other characters should not match.
103b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(empty_string, 'A', size), nullptr);
104b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(empty_string, '9', size), nullptr);
105b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(empty_string, '?', size), nullptr);
106b6a20a49Scgyurgyik }
107b6a20a49Scgyurgyik
TEST(LlvmLibcMemChrTest,SingleRepeatedCharacterShouldReturnFirst)1081df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, SingleRepeatedCharacterShouldReturnFirst) {
109b6a20a49Scgyurgyik const char *dups = "XXXXX";
110b6a20a49Scgyurgyik const size_t size = 6; // 5 characters + null terminator.
111b6a20a49Scgyurgyik // Should return original string since X is first character.
112b6a20a49Scgyurgyik ASSERT_STREQ(call_memchr(dups, 'X', size), dups);
113b6a20a49Scgyurgyik }
114a4f0c58cScgyurgyik
TEST(LlvmLibcMemChrTest,SignedCharacterFound)1151df0dbfcSMichael Jones TEST(LlvmLibcMemChrTest, SignedCharacterFound) {
116a4f0c58cScgyurgyik char c = -1;
117a4f0c58cScgyurgyik const size_t size = 1;
118a4f0c58cScgyurgyik char src[size] = {c};
119a4f0c58cScgyurgyik const char *actual = call_memchr(src, c, size);
120a4f0c58cScgyurgyik // Should find the first character 'c'.
121a4f0c58cScgyurgyik ASSERT_EQ(actual[0], c);
122a4f0c58cScgyurgyik }
123