liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
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
// Copyright 2016 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.
 
package reflect
 
import "unsafe"
 
// Swapper returns a function that swaps the elements in the provided
// slice.
//
// Swapper panics if the provided interface is not a slice.
func Swapper(slice interface{}) func(i, j int) {
   v := ValueOf(slice)
   if v.Kind() != Slice {
       panic(&ValueError{Method: "Swapper", Kind: v.Kind()})
   }
   // Fast path for slices of size 0 and 1. Nothing to swap.
   switch v.Len() {
   case 0:
       return func(i, j int) { panic("reflect: slice index out of range") }
   case 1:
       return func(i, j int) {
           if i != 0 || j != 0 {
               panic("reflect: slice index out of range")
           }
       }
   }
 
   typ := v.Type().Elem().(*rtype)
   size := typ.Size()
   hasPtr := typ.kind&kindNoPointers == 0
 
   // Some common & small cases, without using memmove:
   if hasPtr {
       if size == ptrSize {
           ps := *(*[]unsafe.Pointer)(v.ptr)
           return func(i, j int) { ps[i], ps[j] = ps[j], ps[i] }
       }
       if typ.Kind() == String {
           ss := *(*[]string)(v.ptr)
           return func(i, j int) { ss[i], ss[j] = ss[j], ss[i] }
       }
   } else {
       switch size {
       case 8:
           is := *(*[]int64)(v.ptr)
           return func(i, j int) { is[i], is[j] = is[j], is[i] }
       case 4:
           is := *(*[]int32)(v.ptr)
           return func(i, j int) { is[i], is[j] = is[j], is[i] }
       case 2:
           is := *(*[]int16)(v.ptr)
           return func(i, j int) { is[i], is[j] = is[j], is[i] }
       case 1:
           is := *(*[]int8)(v.ptr)
           return func(i, j int) { is[i], is[j] = is[j], is[i] }
       }
   }
 
   s := (*sliceHeader)(v.ptr)
   tmp := unsafe_New(typ) // swap scratch space
 
   return func(i, j int) {
       if uint(i) >= uint(s.Len) || uint(j) >= uint(s.Len) {
           panic("reflect: slice index out of range")
       }
       val1 := arrayAt(s.Data, i, size, "i < s.Len")
       val2 := arrayAt(s.Data, j, size, "j < s.Len")
       typedmemmove(typ, tmp, val1)
       typedmemmove(typ, val1, val2)
       typedmemmove(typ, val2, tmp)
   }
}