15a9630b7Scgyurgyik //===-- Unittests for memrchr ---------------------------------------------===//
25a9630b7Scgyurgyik //
35a9630b7Scgyurgyik // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
45a9630b7Scgyurgyik // See https://llvm.org/LICENSE.txt for license information.
55a9630b7Scgyurgyik // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65a9630b7Scgyurgyik //
75a9630b7Scgyurgyik //===----------------------------------------------------------------------===//
85a9630b7Scgyurgyik
95a9630b7Scgyurgyik #include "src/string/memrchr.h"
10af1315c2SSiva Chandra Reddy #include "test/UnitTest/Test.h"
115a9630b7Scgyurgyik #include <stddef.h>
125a9630b7Scgyurgyik
135a9630b7Scgyurgyik // A helper function that calls memrchr and abstracts away the explicit cast for
145a9630b7Scgyurgyik // readability purposes.
call_memrchr(const void * src,int c,size_t size)155a9630b7Scgyurgyik const char *call_memrchr(const void *src, int c, size_t size) {
16*b6bc9d72SGuillaume Chatelet return reinterpret_cast<const char *>(LIBC_NAMESPACE::memrchr(src, c, size));
175a9630b7Scgyurgyik }
185a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsCharacterAfterNullTerminator)191df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsCharacterAfterNullTerminator) {
205a9630b7Scgyurgyik // memrchr should continue searching after a null terminator.
215a9630b7Scgyurgyik const size_t size = 6;
225a9630b7Scgyurgyik const unsigned char src[size] = {'a', '\0', 'b', 'c', 'd', '\0'};
235a9630b7Scgyurgyik // Should return 'b', 'c', 'd', '\0' even when after null terminator.
245a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'b', size), "bcd");
255a9630b7Scgyurgyik }
265a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsCharacterInNonNullTerminatedCollection)271df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsCharacterInNonNullTerminatedCollection) {
285a9630b7Scgyurgyik const size_t size = 3;
295a9630b7Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c'};
305a9630b7Scgyurgyik // Should return 'b', 'c'.
315a9630b7Scgyurgyik const char *ret = call_memrchr(src, 'b', size);
325a9630b7Scgyurgyik ASSERT_EQ(ret[0], 'b');
335a9630b7Scgyurgyik ASSERT_EQ(ret[1], 'c');
345a9630b7Scgyurgyik }
355a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsFirstCharacter)361df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsFirstCharacter) {
375a9630b7Scgyurgyik const size_t size = 6;
385a9630b7Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
395a9630b7Scgyurgyik // Should return original array since 'a' is the first character.
405a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'a', size), "abcde");
415a9630b7Scgyurgyik }
425a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsMiddleCharacter)431df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsMiddleCharacter) {
445a9630b7Scgyurgyik const size_t size = 6;
455a9630b7Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
465a9630b7Scgyurgyik // Should return characters after (and including) 'c'.
475a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'c', size), "cde");
485a9630b7Scgyurgyik }
495a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsLastCharacterThatIsNotNullTerminator)501df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsLastCharacterThatIsNotNullTerminator) {
515a9630b7Scgyurgyik const size_t size = 6;
525a9630b7Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
535a9630b7Scgyurgyik // Should return 'e' and null-terminator.
545a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'e', size), "e");
555a9630b7Scgyurgyik }
565a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,FindsNullTerminator)571df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, FindsNullTerminator) {
585a9630b7Scgyurgyik const size_t size = 6;
595a9630b7Scgyurgyik const unsigned char src[size] = {'a', 'b', 'c', 'd', 'e', '\0'};
605a9630b7Scgyurgyik // Should return null terminator.
615a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, '\0', size), "");
625a9630b7Scgyurgyik }
635a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,CharacterNotWithinStringShouldReturnNullptr)641df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, CharacterNotWithinStringShouldReturnNullptr) {
655a9630b7Scgyurgyik const size_t size = 4;
665a9630b7Scgyurgyik const unsigned char src[size] = {'1', '2', '3', '?'};
675a9630b7Scgyurgyik // Since 'z' is not within 'characters', should return nullptr.
685a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'z', size), nullptr);
695a9630b7Scgyurgyik }
705a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,CharacterNotWithinSizeShouldReturnNullptr)711df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, CharacterNotWithinSizeShouldReturnNullptr) {
725a9630b7Scgyurgyik const unsigned char src[5] = {'1', '2', '3', '4', '\0'};
735a9630b7Scgyurgyik // Since '4' is not within the first 2 characters, this should return nullptr.
745a9630b7Scgyurgyik const size_t size = 2;
755a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, '4', size), nullptr);
765a9630b7Scgyurgyik }
775a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,ShouldFindLastOfDuplicates)781df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, ShouldFindLastOfDuplicates) {
795a9630b7Scgyurgyik size_t size = 12; // 11 characters + null terminator.
805a9630b7Scgyurgyik const char *dups = "abc1def1ghi";
815a9630b7Scgyurgyik // 1 is duplicated in 'dups', but it should find the last copy.
825a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(dups, '1', size), "1ghi");
835a9630b7Scgyurgyik
845a9630b7Scgyurgyik const char *repeated = "XXXXX";
855a9630b7Scgyurgyik size = 6; // 5 characters + null terminator.
865a9630b7Scgyurgyik // Should return the last X with the null terminator.
875a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(repeated, 'X', size), "X");
885a9630b7Scgyurgyik }
895a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,EmptyStringShouldOnlyMatchNullTerminator)901df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, EmptyStringShouldOnlyMatchNullTerminator) {
915a9630b7Scgyurgyik const size_t size = 1; // Null terminator.
925a9630b7Scgyurgyik const char *empty_string = "";
935a9630b7Scgyurgyik // Null terminator should match.
945a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(empty_string, '\0', size), "");
955a9630b7Scgyurgyik // All other characters should not match.
965a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(empty_string, 'A', size), nullptr);
975a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(empty_string, '9', size), nullptr);
985a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(empty_string, '?', size), nullptr);
995a9630b7Scgyurgyik }
1005a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,SignedCharacterFound)1011df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, SignedCharacterFound) {
1025a9630b7Scgyurgyik char c = -1;
1035a9630b7Scgyurgyik const size_t size = 1;
1045a9630b7Scgyurgyik char src[size] = {c};
1055a9630b7Scgyurgyik const char *actual = call_memrchr(src, c, size);
1065a9630b7Scgyurgyik // Should find the last character 'c'.
1075a9630b7Scgyurgyik ASSERT_EQ(actual[0], c);
1085a9630b7Scgyurgyik }
1095a9630b7Scgyurgyik
TEST(LlvmLibcMemRChrTest,ZeroLengthShouldReturnNullptr)1101df0dbfcSMichael Jones TEST(LlvmLibcMemRChrTest, ZeroLengthShouldReturnNullptr) {
1115a9630b7Scgyurgyik const unsigned char src[4] = {'a', 'b', 'c', '\0'};
1125a9630b7Scgyurgyik // This will iterate over exactly zero characters, so should return nullptr.
1135a9630b7Scgyurgyik ASSERT_STREQ(call_memrchr(src, 'd', 0), nullptr);
1145a9630b7Scgyurgyik }
115