summaryrefslogtreecommitdiff
path: root/lib/libc/memmove.c
diff options
context:
space:
mode:
authorDimitris Papastamos <dimitris.papastamos@arm.com>2018-08-22 14:40:50 +0100
committerGitHub <noreply@github.com>2018-08-22 14:40:50 +0100
commit6d4f6aea2cd96a4a57ffa2d88b9230e2cab88f28 (patch)
tree251ed2d6ebe98576eaa9e1871f3aef753552125b /lib/libc/memmove.c
parent11dfe0b49ac8fcb5d1b516a3f52b06bc433ff4d9 (diff)
parent8422a8406b7d2d8e01c113e24eca167854981dfe (diff)
Merge pull request #1528 from antonio-nino-diaz-arm/an/libc
libc: Cleanup library
Diffstat (limited to 'lib/libc/memmove.c')
-rw-r--r--lib/libc/memmove.c31
1 files changed, 31 insertions, 0 deletions
diff --git a/lib/libc/memmove.c b/lib/libc/memmove.c
new file mode 100644
index 00000000..63acf267
--- /dev/null
+++ b/lib/libc/memmove.c
@@ -0,0 +1,31 @@
+/*
+ * Copyright (c) 2013-2018, ARM Limited and Contributors. All rights reserved.
+ *
+ * SPDX-License-Identifier: BSD-3-Clause
+ */
+
+#include <string.h>
+
+void *memmove(void *dst, const void *src, size_t len)
+{
+ /*
+ * The following test makes use of unsigned arithmetic overflow to
+ * more efficiently test the condition !(src <= dst && dst < str+len).
+ * It also avoids the situation where the more explicit test would give
+ * incorrect results were the calculation str+len to overflow (though
+ * that issue is probably moot as such usage is probably undefined
+ * behaviour and a bug anyway.
+ */
+ if ((size_t)dst - (size_t)src >= len) {
+ /* destination not in source data, so can safely use memcpy */
+ return memcpy(dst, src, len);
+ } else {
+ /* copy backwards... */
+ const char *end = dst;
+ const char *s = (const char *)src + len;
+ char *d = (char *)dst + len;
+ while (d != end)
+ *--d = *--s;
+ }
+ return dst;
+}