| /* go-memmove.c -- memmove |
| |
| Copyright 2021 The Go Authors. All rights reserved. |
| Use of this source code is governed by a BSD-style |
| license that can be found in the LICENSE file. */ |
| |
| #include "runtime.h" |
| |
| void gomemmove(void *, void *, uintptr) |
| __asm__ (GOSYM_PREFIX "runtime.memmove") |
| __attribute__ ((no_split_stack)); |
| |
| // This implementation is necessary since |
| // the __builtin_memmove might use __libc_memmove |
| // which doesn't require atomicity of 8 byte |
| // moves. |
| |
| void |
| gomemmove (void *dst, void *src, uintptr len) |
| { |
| #if !defined(__PPC64__) |
| __builtin_memmove(dst, src, len); |
| #else |
| uint64 offset, tail; |
| int64 rem; |
| uint64 dwords; |
| uint64 i; |
| char *bdst,*bsrc; |
| |
| rem = len; |
| |
| if (len == 0) { |
| return; |
| } |
| |
| // If src and dst don't have the same 8 byte alignment then |
| // there is no issue with copying pointer atomicity. Use the |
| // builtin. |
| if (((uint64)dst % 8) != ((uint64)src % 8) || len < 8) { |
| __builtin_memmove(dst, src, len); |
| return; |
| } |
| |
| // Length >= 8 && same ptr alignment |
| offset = (uint64)dst % 8; |
| |
| // If not 8 byte alignment, move the intial bytes. |
| if (offset > 0) { |
| __builtin_memmove(dst, src, 8-offset); |
| dst += (8-offset); |
| src += (8-offset); |
| rem -= (8-offset); |
| } |
| |
| // Move the tail bytes to make the backward move |
| // easier. |
| tail = rem % 8; |
| if (tail > 0) { |
| __builtin_memmove(dst+rem-tail, src+rem-tail, tail); |
| rem -= tail; |
| } |
| |
| if (rem == 0) { |
| return; |
| } |
| |
| // Must now be 8 byte alignment and rem is multiple of 8. |
| dwords = len>>3; |
| |
| // Determine if a backwards move is needed |
| // Forward or backward, move all doublewords |
| |
| if ((uint64)(dst - src) < (uint64)rem) { |
| bdst = dst+rem-8; |
| bsrc = src+rem-8; |
| for (i = 0; i<dwords; i++) { |
| *(uint64*)bdst = *(uint64*)bsrc; |
| bdst -= 8; |
| bsrc -= 8; |
| } |
| } else { |
| for (i = 0; i<dwords; i++) { |
| *(uint64*)dst = *(uint64*)src; |
| dst += 8; |
| src += 8; |
| } |
| } |
| #endif |
| } |