summaryrefslogtreecommitdiff
path: root/src/shared/mem.c
diff options
context:
space:
mode:
authordzwdz2022-08-26 12:19:13 +0200
committerdzwdz2022-08-26 12:19:13 +0200
commiteba8e12e8f0d113056264f71b8cbcda730e93506 (patch)
treeb9d5efa10cfa2d308a7df23a91cf3ea2199950be /src/shared/mem.c
parentdc351c97829ef8d15219d90e32101768bd0481be (diff)
shared: memmove
Diffstat (limited to 'src/shared/mem.c')
-rw-r--r--src/shared/mem.c18
1 files changed, 15 insertions, 3 deletions
diff --git a/src/shared/mem.c b/src/shared/mem.c
index e3279f6..14dd6bd 100644
--- a/src/shared/mem.c
+++ b/src/shared/mem.c
@@ -1,3 +1,4 @@
+#include <assert.h>
#include <shared/mem.h>
#include <stdint.h>
@@ -28,16 +29,27 @@ void *memcpy(void *dest, const void *src, size_t n) {
// TODO erms, rep movsb
union dualptr_const s = {.c = src};
union dualptr d = {.c = dest};
+ if (dest == src) return dest;
+ // assert(src >= dest || src + n < dest);
- for (; (d.u & 7) && n != 0; n--)
+ for (; (d.u & 7) && n != 0; n--) {
*(d.c)++ = *(s.c)++;
-
+ }
while (n >= sizeof(uintptr_t)) {
*(d.w)++ = *(s.w)++;
n -= sizeof(uintptr_t);
}
- while (n-- != 0)
+ while (n-- != 0) {
*(d.c)++ = *(s.c)++;
+ }
+ return dest;
+}
+
+void *memmove(void *dest, const void *src, size_t n) {
+ if (src >= dest || src + n < dest)
+ return memcpy(dest, src, n);
+ for (; n; n--) /* naive reverse copy */
+ ((uint8_t*)dest)[n-1] = ((uint8_t*)src)[n-1];
return dest;
}