aboutsummaryrefslogtreecommitdiff
path: root/libgo/runtime/go-memmove.c
blob: 1ca3f4822b736fadbe0db35a5784f730b0f5f5b2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
/* 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
}