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