root/arch/arm/boot/compressed/string.c
// SPDX-License-Identifier: GPL-2.0
/*
 * arch/arm/boot/compressed/string.c
 *
 * Small subset of simple string routines
 */

#define __NO_FORTIFY
#include <linux/string.h>

/*
 * The decompressor is built without KASan but uses the same redirects as the
 * rest of the kernel when CONFIG_KASAN is enabled, defining e.g. memcpy()
 * to __memcpy() but since we are not linking with the main kernel string
 * library in the decompressor, that will lead to link failures.
 *
 * Undefine KASan's versions, define the wrapped functions and alias them to
 * the right names so that when e.g. __memcpy() appear in the code, it will
 * still be linked to this local version of memcpy().
 */
#ifdef CONFIG_KASAN
#undef memcpy
#undef memmove
#undef memset
void *__memcpy(void *__dest, __const void *__src, size_t __n) __alias(memcpy);
void *__memmove(void *__dest, __const void *__src, size_t count) __alias(memmove);
void *__memset(void *s, int c, size_t count) __alias(memset);
#endif

void *memcpy(void *__dest, __const void *__src, size_t __n)
{
        int i = 0;
        unsigned char *d = (unsigned char *)__dest, *s = (unsigned char *)__src;

        for (i = __n >> 3; i > 0; i--) {
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
        }

        if (__n & 1 << 2) {
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
                *d++ = *s++;
        }

        if (__n & 1 << 1) {
                *d++ = *s++;
                *d++ = *s++;
        }

        if (__n & 1)
                *d++ = *s++;

        return __dest;
}

void *memmove(void *__dest, __const void *__src, size_t count)
{
        unsigned char *d = __dest;
        const unsigned char *s = __src;

        if (__dest == __src)
                return __dest;

        if (__dest < __src)
                return memcpy(__dest, __src, count);

        while (count--)
                d[count] = s[count];
        return __dest;
}

size_t strlen(const char *s)
{
        const char *sc = s;

        while (*sc != '\0')
                sc++;
        return sc - s;
}

size_t strnlen(const char *s, size_t count)
{
        const char *sc;

        for (sc = s; count-- && *sc != '\0'; ++sc)
                /* nothing */;
        return sc - s;
}

int memcmp(const void *cs, const void *ct, size_t count)
{
        const unsigned char *su1 = cs, *su2 = ct, *end = su1 + count;
        int res = 0;

        while (su1 < end) {
                res = *su1++ - *su2++;
                if (res)
                        break;
        }
        return res;
}

int strcmp(const char *cs, const char *ct)
{
        unsigned char c1, c2;
        int res = 0;

        do {
                c1 = *cs++;
                c2 = *ct++;
                res = c1 - c2;
                if (res)
                        break;
        } while (c1);
        return res;
}

void *memchr(const void *s, int c, size_t count)
{
        const unsigned char *p = s;

        while (count--)
                if ((unsigned char)c == *p++)
                        return (void *)(p - 1);
        return NULL;
}

char *strchr(const char *s, int c)
{
        while (*s != (char)c)
                if (*s++ == '\0')
                        return NULL;
        return (char *)s;
}

char *strrchr(const char *s, int c)
{
        const char *last = NULL;
        do {
                if (*s == (char)c)
                        last = s;
        } while (*s++);
        return (char *)last;
}

#undef memset

void *memset(void *s, int c, size_t count)
{
        char *xs = s;
        while (count--)
                *xs++ = c;
        return s;
}