-
Notifications
You must be signed in to change notification settings - Fork 96
Expand file tree
/
Copy pathdiff_pointer.go
More file actions
80 lines (66 loc) · 1.94 KB
/
diff_pointer.go
File metadata and controls
80 lines (66 loc) · 1.94 KB
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
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
package diff
import (
"reflect"
"unsafe"
)
var isExportFlag uintptr = (1 << 5) | (1 << 6)
func (d *Differ) diffPtr(path []string, a, b reflect.Value, parent interface{}) error {
if a.Kind() != b.Kind() {
if a.Kind() == reflect.Invalid {
if !b.IsNil() {
return d.diff(path, reflect.ValueOf(nil), reflect.Indirect(b), parent)
}
d.cl.Add(CREATE, path, nil, exportInterface(b), parent)
return nil
}
if b.Kind() == reflect.Invalid {
if !a.IsNil() {
return d.diff(path, reflect.Indirect(a), reflect.ValueOf(nil), parent)
}
d.cl.Add(DELETE, path, exportInterface(a), nil, parent)
return nil
}
return ErrTypeMismatch
}
if a.IsNil() && b.IsNil() {
return nil
}
if a.IsNil() {
d.cl.Add(UPDATE, path, nil, exportInterface(b), parent)
return nil
}
if b.IsNil() {
d.cl.Add(UPDATE, path, exportInterface(a), nil, parent)
return nil
}
// If two pointers have already been compared, assume they have no changes
// This mirrors how reflect.DeepEqual works, and guarantees termination
aSeen, ok := d.pointersSeen[a.Pointer()]
if !ok {
aSeen = make(map[uintptr]struct{})
d.pointersSeen[a.Pointer()] = aSeen
}
bSeen, ok := d.pointersSeen[b.Pointer()]
if !ok {
bSeen = make(map[uintptr]struct{})
d.pointersSeen[b.Pointer()] = bSeen
}
_, aok := aSeen[b.Pointer()]
_, bok := aSeen[b.Pointer()]
if aok || bok {
return nil
}
aSeen[b.Pointer()] = struct{}{}
bSeen[a.Pointer()] = struct{}{}
return d.diff(path, reflect.Indirect(a), reflect.Indirect(b), parent)
}
func exportInterface(v reflect.Value) interface{} {
if !v.CanInterface() {
flagTmp := (*uintptr)(unsafe.Pointer(uintptr(unsafe.Pointer(&v)) + 2*unsafe.Sizeof(uintptr(0))))
*flagTmp = (*flagTmp) & (^isExportFlag)
}
return v.Interface()
}