root/crypto/libecc/src/utils/utils.c
/*
 *  Copyright (C) 2017 - This file is part of libecc project
 *
 *  Authors:
 *      Ryad BENADJILA <ryadbenadjila@gmail.com>
 *      Arnaud EBALARD <arnaud.ebalard@ssi.gouv.fr>
 *      Jean-Pierre FLORI <jean-pierre.flori@ssi.gouv.fr>
 *
 *  Contributors:
 *      Nicolas VIVET <nicolas.vivet@ssi.gouv.fr>
 *      Karim KHALFALLAH <karim.khalfallah@ssi.gouv.fr>
 *
 *  This software is licensed under a dual BSD and GPL v2 license.
 *  See LICENSE file at the root folder of the project.
 */
#include <libecc/utils/utils.h>

/*
 * Return 1 in 'check' if first 'len' bytes of both buffers a and b are equal, 0 otherwise.
 * It returns 0 if success, -1 on error. 'check' is only relevant on success.
 *
 * The test is done in constant time.
 */
int are_equal(const void *a, const void *b, u32 len, int *check)
{
        const u8 *la = (const u8*)a, *lb = (const u8*)b;
        int ret;
        u32 i;

        MUST_HAVE((a != NULL) && (b != NULL) && (check != NULL), ret, err);

        *check = 1;
        for (i = 0; i < len; i++) {
                (*check) &= (*la == *lb);
                la++;
                lb++;
        }

        ret = 0;

err:
        return ret;
}

/*
 * This function is a simple (non-optimized) reimplementation of memcpy()
 * Returns 0 on success, -1 on error.
 */
int local_memcpy(void *dst, const void *src, u32 n)
{
        const u8 *lsrc = (const u8*)src;
        u8 *ldst = (u8*)dst;
        u32 i;
        int ret;

        MUST_HAVE((dst != NULL) && (src != NULL), ret, err);

        for (i = 0; i < n; i++) {
                *ldst = *lsrc;
                ldst++;
                lsrc++;
        }

        ret = 0;

err:
        return ret;
}

/*
 * This function is a simple (non-optimized) reimplementation of memset()
 * Returns 0 on success, -1 on error.
 */
int local_memset(void *v, u8 c, u32 n)
{
        volatile u8 *p = (volatile u8*)v;
        u32 i;
        int ret;

        MUST_HAVE((v != NULL), ret, err);

        for (i = 0; i < n; i++) {
                *p = c;
                p++;
        }

        ret = 0;

err:
        return ret;
}

/*
 * Return 1 in 'check' if strings are equal, 0 otherwise.
 * It returns 0 if success, -1 on error. 'check' is only relevant on success.
 *
 */
int are_str_equal(const char *s1, const char *s2, int *check)
{
        const char *ls1 = s1, *ls2 = s2;
        int ret;

        MUST_HAVE((s1 != NULL) && (s2 != NULL) && (check != NULL), ret, err);

        while (*ls1 && (*ls1 == *ls2)) {
                ls1++;
                ls2++;
        }

        (*check) = (*ls1 == *ls2);

        ret = 0;

err:
        return ret;
}

/*
 * Return 1 in 'check' if strings are equal up to maxlen, 0 otherwise.
 * It returns 0 if success, -1 on error. 'check' is only relevant on success.
 *
 */
int are_str_equal_nlen(const char *s1, const char *s2, u32 maxlen, int *check)
{
        const char *ls1 = s1, *ls2 = s2;
        u32 i = 0;
        int ret;

        MUST_HAVE((s1 != NULL) && (s2 != NULL) && (check != NULL), ret, err);

        while (*ls1 && (*ls1 == *ls2) && (i < maxlen)) {
                ls1++;
                ls2++;
                i++;
        }

        (*check) = (*ls1 == *ls2);
        ret = 0;

err:
        return ret;
}



/*
 * This function is a simple (non-optimized) reimplementation of strlen()
 * Returns the lenth in 'len'.
 * It returns 0 if success, -1 on error. 'len' is only relevant on success.
 */
int local_strlen(const char *s, u32 *len)
{
        u32 i = 0;
        int ret;

        MUST_HAVE((s != NULL) && (len != NULL), ret, err);

        while (s[i]) {
                i++;
        }
        (*len) = i;

        ret = 0;

err:
        return ret;
}

/*
 * This function is a simple (non-optimized) reimplementation of strnlen()
 * Returns the lenth in 'len'.
 * It returns 0 if success, -1 on error. 'len' is only relevant on success.
 */
int local_strnlen(const char *s, u32 maxlen, u32 *len)
{
        u32 i = 0;
        int ret;

        MUST_HAVE((s != NULL) && (len != NULL), ret, err);

        while ((i < maxlen) && s[i]) {
                i++;
        }
        (*len) = i;

        ret = 0;

err:
        return ret;
}

/*
 * This functin is a simple (non-optimized) reimplementation of strncpy()
 */
int local_strncpy(char *dst, const char *src, u32 n)
{
        u32 i;
        int ret;

        MUST_HAVE((dst != NULL) && (src != NULL), ret, err);

        for (i = 0; (i < n) && src[i]; i++) {
                dst[i] = src[i];
        }
        for (; i < n; i++) {
                dst[i] = 0;
        }

        ret = 0;
err:
        return ret;
}

/*
 * This functin is a simple (non-optimized) reimplementation of strncat()
 */
int local_strncat(char *dst, const char *src, u32 n)
{
        u32 dst_len, i;
        int ret;

        MUST_HAVE((dst != NULL) && (src != NULL), ret, err);

        ret = local_strlen(dst, &dst_len); EG(ret, err);
        for (i = 0; (i < n) && src[i]; i++) {
                dst[dst_len + i] = src[i];
        }
        dst[dst_len + i] = 0;

        ret = 0;
err:
        return ret;
}