summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorYu-Ping Wu <yupingso@chromium.org>2024-08-05 03:58:56 +0000
committerYu-Ping Wu <yupingso@google.com>2024-08-14 03:09:20 +0000
commitaff734bc42b4e132abb5de0e44b0464dd4688590 (patch)
tree5ed1178030ddd3d660cd5d5304fe95dc009432ff
parent0dcdc0347c8d0405c1c6b444d23483dd9bf31d34 (diff)
commonlib/bsd: Add strcat() and strncat() functions
An upcoming vboot feature [1] will need strcat() to be defined in string.h. Therefore, add strcat() and strncat() to commonlib/bsd. Remove those functions from libpayload. [1] https://chromium-review.googlesource.com/c/chromiumos/platform/vboot_reference/+/5650810 Change-Id: If02fce0eafb4f6fa01d8bab17d87a32360f4ac83 Signed-off-by: Yu-Ping Wu <yupingso@chromium.org> Reviewed-on: https://review.coreboot.org/c/coreboot/+/83765 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Julius Werner <jwerner@chromium.org>
-rw-r--r--payloads/libpayload/include/string.h2
-rw-r--r--payloads/libpayload/libc/string.c40
-rw-r--r--src/commonlib/bsd/include/commonlib/bsd/string.h2
-rw-r--r--src/commonlib/bsd/string.c23
-rw-r--r--tests/commonlib/bsd/string-test.c74
5 files changed, 99 insertions, 42 deletions
diff --git a/payloads/libpayload/include/string.h b/payloads/libpayload/include/string.h
index 7762c36b67..b0028228eb 100644
--- a/payloads/libpayload/include/string.h
+++ b/payloads/libpayload/include/string.h
@@ -53,8 +53,6 @@ int strcasecmp(const char *s1, const char *s2);
int strncasecmp(const char *s1, const char *s2, size_t maxlen);
char *strncpy(char *d, const char *s, size_t n);
char *strcpy(char *d, const char *s);
-char *strncat(char *d, const char *s, size_t n);
-char *strcat(char *d, const char *s);
char *strchr(const char *s, int c);
char *strrchr(const char *s, int c);
char *strdup(const char *s);
diff --git a/payloads/libpayload/libc/string.c b/payloads/libpayload/libc/string.c
index f9805e39bf..a1b7d4d59d 100644
--- a/payloads/libpayload/libc/string.c
+++ b/payloads/libpayload/libc/string.c
@@ -153,46 +153,6 @@ char *strcpy(char *d, const char *s)
}
/**
- * Concatenates two strings
- *
- * @param d The destination string.
- * @param s The source string.
- * @return A pointer to the destination string.
- */
-char *strcat(char *d, const char *s)
-{
- char *p = d + strlen(d);
- size_t sl = strlen(s);
-
- for (size_t i = 0; i < sl; i++)
- p[i] = s[i];
-
- p[sl] = '\0';
- return d;
-}
-
-/**
- * Concatenates two strings with a maximum length.
- *
- * @param d The destination string.
- * @param s The source string.
- * @param n Not more than n characters from s will be appended to d.
- * @return A pointer to the destination string.
- */
-char *strncat(char *d, const char *s, size_t n)
-{
- char *p = d + strlen(d);
- size_t sl = strlen(s);
- size_t max = n > sl ? sl : n;
-
- for (size_t i = 0; i < max; i++)
- p[i] = s[i];
-
- p[max] = '\0';
- return d;
-}
-
-/**
* Concatenates two strings with a maximum length.
*
* @param d The destination string.
diff --git a/src/commonlib/bsd/include/commonlib/bsd/string.h b/src/commonlib/bsd/include/commonlib/bsd/string.h
index 8e8b896e7e..5c97c65cae 100644
--- a/src/commonlib/bsd/include/commonlib/bsd/string.h
+++ b/src/commonlib/bsd/include/commonlib/bsd/string.h
@@ -8,6 +8,8 @@
size_t strlen(const char *src);
size_t strnlen(const char *str, size_t maxlen);
+char *strcat(char *dst, const char *src);
+char *strncat(char *dst, const char *src, size_t n);
unsigned int skip_atoi(char **ptr);
diff --git a/src/commonlib/bsd/string.c b/src/commonlib/bsd/string.c
index a9dce39e8d..16cd4b5e1d 100644
--- a/src/commonlib/bsd/string.c
+++ b/src/commonlib/bsd/string.c
@@ -23,6 +23,29 @@ size_t strnlen(const char *str, size_t maxlen)
return ptr - str - 1;
}
+char *strcat(char *dst, const char *src)
+{
+ char *ptr = dst + strlen(dst);
+
+ while (*src)
+ *ptr++ = *src++;
+
+ *ptr = '\0';
+ return dst;
+}
+
+char *strncat(char *dst, const char *src, size_t n)
+{
+ char *ptr = dst + strlen(dst);
+
+ /* Not using strncpy() because '\0' may not be appended. */
+ while (n-- > 0 && *src)
+ *ptr++ = *src++;
+
+ *ptr = '\0';
+ return dst;
+}
+
unsigned int skip_atoi(char **ptr)
{
unsigned int result = 0;
diff --git a/tests/commonlib/bsd/string-test.c b/tests/commonlib/bsd/string-test.c
index 29e3977675..37419f049d 100644
--- a/tests/commonlib/bsd/string-test.c
+++ b/tests/commonlib/bsd/string-test.c
@@ -2,6 +2,7 @@
#include <commonlib/bsd/string.h>
#include <stddef.h>
+#include <string.h>
#include <tests/test.h>
static void test_strlen(void **state)
@@ -38,6 +39,77 @@ static void test_strnlen(void **state)
assert_int_equal(0, strnlen("", 3));
}
+static void test_strcat(void **state)
+{
+ static const char str[] = "Hello ";
+ size_t len = __builtin_strlen(str);
+ static const char src[] = "World";
+ static const char expected[] = "Hello World";
+ size_t expected_len = __builtin_strlen(expected);
+ char dst[100];
+ char *ret;
+
+ /* Empty src & dst */
+ dst[0] = '\0';
+ memset(dst + 1, 0xee, sizeof(dst) - 1);
+ ret = strcat(dst, "");
+ assert_ptr_equal(dst, ret);
+ assert_int_equal('\0', dst[0]);
+
+ /* Empty src */
+ memcpy(dst, str, len + 1);
+ memset(dst + len + 1, 0xee, sizeof(dst) - len - 1);
+ ret = strcat(dst, "");
+ assert_memory_equal(str, dst, len + 1);
+
+ /* Empty dst */
+ memset(dst, 0x0, sizeof(dst));
+ memset(dst + 1, 0xee, sizeof(dst) - 1);
+ ret = strcat(dst, src);
+ assert_ptr_equal(dst, ret);
+ assert_memory_equal(src, dst, __builtin_strlen(src) + 1);
+
+ /* Non-empty str & dst */
+ memcpy(dst, str, len + 1);
+ memset(dst + len + 1, 0xee, sizeof(dst) - len - 1);
+ ret = strcat(dst, src);
+ assert_ptr_equal(dst, ret);
+ assert_memory_equal(expected, dst, expected_len + 1);
+}
+
+static void test_strncat(void **state)
+{
+ static const char str[] = "Hello ";
+ size_t len = __builtin_strlen(str);
+ static const char src[] = "World";
+ size_t src_len = __builtin_strlen(src);
+ static const char expected[] = "Hello World";
+ size_t expected_len = __builtin_strlen(expected);
+ char dst[100];
+ char *ret;
+
+ /* n larger than src len */
+ memcpy(dst, str, len + 1);
+ memset(dst + len + 1, 0xee, sizeof(dst) - len - 1);
+ ret = strncat(dst, src, src_len + 5);
+ assert_ptr_equal(dst, ret);
+ assert_memory_equal(expected, dst, expected_len + 1);
+
+ /* n smaller than src len */
+ memcpy(dst, str, len + 1);
+ memset(dst + len + 1, 0xee, sizeof(dst) - len - 1);
+ ret = strncat(dst, src, src_len - 2);
+ assert_ptr_equal(dst, ret);
+ assert_memory_equal("Hello Wor", dst, expected_len - 2 + 1);
+
+ /* n is 0 */
+ memcpy(dst, str, len + 1);
+ memset(dst + len + 1, 0xee, sizeof(dst) - len - 1);
+ ret = strncat(dst, src, 0);
+ assert_ptr_equal(dst, ret);
+ assert_memory_equal(str, dst, len + 1);
+}
+
static void test_skip_atoi(void **state)
{
int i;
@@ -66,6 +138,8 @@ int main(void)
const struct CMUnitTest tests[] = {
cmocka_unit_test(test_strlen),
cmocka_unit_test(test_strnlen),
+ cmocka_unit_test(test_strcat),
+ cmocka_unit_test(test_strncat),
cmocka_unit_test(test_skip_atoi),
};