Commit 3b06f69
authored
perf(kapp): zero-copy GetAndClearReturnData + pre-size receipt filters (#45)
* perf(kapp): zero-copy GetAndClearReturnData + pre-size receipt filters
Two related allocation reductions in core/kapp/context.go.
1. GetAndClearReturnData: move semantics instead of deep copy
The function deep-copied k.returnData into a fresh slice and then
reset k.returnData to empty. Both call sites
(blockChainHook.ProcessBuiltInFunction and the kvm world wrapper)
assign the result straight into vmOutput.ReturnData and never
touch the context's slice afterward, so the deep copy was guarding
against an aliasing scenario that cannot occur:
- The context's slice is reset on the same call.
- SetReturnData and AddReturnData both allocate fresh storage on
every write, so the context never re-aliases the returned
slice.
Replace the deep copy with ownership transfer. Keep the original
non-nil-empty invariant for both the returned slice and the field:
VMOutputApi carries a json:"returnData" tag (nil renders as null,
[] renders as []), so downstream API consumers can distinguish.
Get cost drops to O(1) regardless of payload shape.
Bench (Apple M4 Max, darwin/arm64), Get cost (combined - Set):
shape before after allocs (before -> after)
1 x 32 B 22.5 ns ~0 2 -> 0
5 x 32 B 75.3 ns ~0 6 -> 0
10 x 64 B 175 ns ~0 11 -> 0
50 x 256 B 1686 ns ~0 51 -> 0
1 x 4096 B 434 ns ~0 2 -> 0
The savings scale with return-data payload size. Hot for SC TX
paths that emit many or large return values (view functions,
batch operations, mint/transfer arrays).
SetReturnData and AddReturnData still deep-copy on input. Their
defensive copies guard against caller-side mutation of the source
slice and were left unchanged: the gain from removing them is
smaller than the risk of a future caller silently aliasing.
2. ReceiptSlice.GetByType / GetPreserved: pre-size filtered slice
Both started with `var filtered []*X`, so the first append
allocated and subsequent grows reallocated (log2(N) backing
arrays). Pre-size capacity to len(*r) so the filter is bounded to
one allocation regardless of match count. Worst case == prior
memory; best case == fewer allocs. No nil-vs-empty caller
sensitivity (all callers use len() or range).
Adds three unit tests (none existed for the returnData lifecycle):
- TestKappContext_ReturnData_RoundTrip exercises Set -> Get ->
empty -> Add -> Get on a single context.
- TestKappContext_ReturnData_GetNeverReturnsNil pins the
JSON-visible invariant that Get always returns a non-nil slice.
- TestKappContext_ReturnData_GetIsolatesFromFutureWrites pins the
invariant that the slice handed to the caller is never mutated
by subsequent context writes (guards the move-semantics
optimization against future regressions).
Adds context_returndata_bench_test.go with both Get and Set isolated
across realistic SC return shapes.
* chore(kapp): remove unused benchmark helper
newCtxWithReturnData was introduced for benchmark setup but ended up
unused once both benchmarks constructed their contexts inline. Flagged
by golangci-lint (unused) in PR #45 review.1 parent d695ec5 commit 3b06f69
3 files changed
Lines changed: 198 additions & 13 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
46 | 46 | | |
47 | 47 | | |
48 | 48 | | |
49 | | - | |
| 49 | + | |
50 | 50 | | |
51 | 51 | | |
52 | 52 | | |
| |||
56 | 56 | | |
57 | 57 | | |
58 | 58 | | |
59 | | - | |
| 59 | + | |
60 | 60 | | |
61 | 61 | | |
62 | 62 | | |
| |||
150 | 150 | | |
151 | 151 | | |
152 | 152 | | |
153 | | - | |
154 | | - | |
155 | | - | |
156 | | - | |
157 | | - | |
158 | | - | |
159 | | - | |
160 | | - | |
161 | | - | |
| 153 | + | |
| 154 | + | |
| 155 | + | |
| 156 | + | |
| 157 | + | |
| 158 | + | |
| 159 | + | |
| 160 | + | |
| 161 | + | |
| 162 | + | |
| 163 | + | |
| 164 | + | |
| 165 | + | |
162 | 166 | | |
163 | | - | |
164 | 167 | | |
165 | | - | |
| 168 | + | |
166 | 169 | | |
167 | 170 | | |
168 | 171 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 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 | + | |
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
1 | 1 | | |
2 | 2 | | |
3 | 3 | | |
| 4 | + | |
4 | 5 | | |
5 | 6 | | |
6 | 7 | | |
| |||
76 | 77 | | |
77 | 78 | | |
78 | 79 | | |
| 80 | + | |
| 81 | + | |
| 82 | + | |
| 83 | + | |
| 84 | + | |
| 85 | + | |
| 86 | + | |
| 87 | + | |
| 88 | + | |
| 89 | + | |
| 90 | + | |
| 91 | + | |
| 92 | + | |
| 93 | + | |
| 94 | + | |
| 95 | + | |
| 96 | + | |
| 97 | + | |
| 98 | + | |
| 99 | + | |
| 100 | + | |
| 101 | + | |
| 102 | + | |
| 103 | + | |
| 104 | + | |
| 105 | + | |
| 106 | + | |
| 107 | + | |
| 108 | + | |
| 109 | + | |
| 110 | + | |
| 111 | + | |
| 112 | + | |
| 113 | + | |
| 114 | + | |
| 115 | + | |
| 116 | + | |
| 117 | + | |
| 118 | + | |
| 119 | + | |
| 120 | + | |
| 121 | + | |
| 122 | + | |
| 123 | + | |
| 124 | + | |
| 125 | + | |
| 126 | + | |
| 127 | + | |
| 128 | + | |
| 129 | + | |
| 130 | + | |
| 131 | + | |
| 132 | + | |
| 133 | + | |
| 134 | + | |
| 135 | + | |
| 136 | + | |
| 137 | + | |
| 138 | + | |
| 139 | + | |
| 140 | + | |
| 141 | + | |
| 142 | + | |
| 143 | + | |
| 144 | + | |
| 145 | + | |
| 146 | + | |
| 147 | + | |
| 148 | + | |
| 149 | + | |
| 150 | + | |
| 151 | + | |
| 152 | + | |
| 153 | + | |
| 154 | + | |
| 155 | + | |
| 156 | + | |
| 157 | + | |
| 158 | + | |
| 159 | + | |
| 160 | + | |
| 161 | + | |
| 162 | + | |
| 163 | + | |
| 164 | + | |
| 165 | + | |
| 166 | + | |
| 167 | + | |
| 168 | + | |
| 169 | + | |
| 170 | + | |
| 171 | + | |
| 172 | + | |
79 | 173 | | |
80 | 174 | | |
81 | 175 | | |
| |||
0 commit comments