-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpacking.go
118 lines (93 loc) · 3.44 KB
/
packing.go
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
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
package main
import (
"sort"
"github.com/rensr/spack/solidity"
)
func packStructCurrentFieldOrder(fields []solidity.DataDef) []solidity.StorageSlot {
var storageSlots []solidity.StorageSlot
for _, field := range fields {
// If we have no packed fields yet, add the first field as a packed field
if len(storageSlots) == 0 {
storageSlots = []solidity.StorageSlot{{Fields: []solidity.DataDef{field}, Offset: field.Size}}
continue
}
lastSlot := storageSlots[len(storageSlots)-1]
if lastSlot.Offset+field.Size <= 32 {
storageSlots[len(storageSlots)-1].Fields = append(lastSlot.Fields, field)
storageSlots[len(storageSlots)-1].Offset += field.Size
continue
}
storageSlots = append(storageSlots, solidity.StorageSlot{Fields: []solidity.DataDef{field}, Offset: field.Size})
}
return storageSlots
}
func packStructOptimal(fields []solidity.DataDef) []solidity.StorageSlot {
sort.Slice(fields, func(i, j int) bool {
return fields[i].Size > fields[j].Size
})
return binPacking(fields, []solidity.StorageSlot{})
}
func binPacking(fields []solidity.DataDef, existingSlots []solidity.StorageSlot) []solidity.StorageSlot {
if len(fields) == 0 {
return existingSlots
}
currentItem := fields[0]
var packingOptions [][]solidity.StorageSlot
for i, slot := range existingSlots {
// The field doesn't fit into the slot, so skip it
if slot.Offset+currentItem.Size > 32 {
continue
}
// It the field does fit, make a copy of the existing slots and add the field to the slot
slotsCopy := make([]solidity.StorageSlot, len(existingSlots))
copy(slotsCopy, existingSlots)
// Make sure we copy slot.Fields to avoid modifying the original slice
slotsCopy[i].Fields = AddToCopyOfList(slot.Fields, currentItem)
slotsCopy[i].Offset += currentItem.Size
// Recursively call binPacking with the remaining fields and the new slot state
packingOptions = append(packingOptions, binPacking(fields[1:], slotsCopy))
}
// If we can't fit the field into any existing slots, create a new slot and recursively call binPacking
packingOptions = append(packingOptions, binPacking(fields[1:], append(existingSlots, solidity.StorageSlot{
Fields: []solidity.DataDef{currentItem},
Offset: currentItem.Size,
})))
return findOptimalPacking(packingOptions)
}
func findOptimalPacking(options [][]solidity.StorageSlot) []solidity.StorageSlot {
if len(options) == 0 {
return []solidity.StorageSlot{}
}
// First find the options with the least amount of slots
// This gives the first option twice if it happens to be the best
// This doesn't matter much since we only return a single option anyway
leastAmountOfSlots := [][]solidity.StorageSlot{options[0]}
for _, option := range options {
if len(option) < len(leastAmountOfSlots[0]) {
leastAmountOfSlots = [][]solidity.StorageSlot{option}
} else if len(option) == len(leastAmountOfSlots[0]) {
leastAmountOfSlots = append(leastAmountOfSlots, option)
}
}
// Then find the options with the most completely filled slots
bestSlots := leastAmountOfSlots[0]
mostFilledStorageSlots := 0
for _, option := range leastAmountOfSlots {
filledSlots := 0
for _, slot := range option {
if slot.Offset == 32 {
filledSlots++
}
}
if filledSlots > mostFilledStorageSlots {
bestSlots = option
mostFilledStorageSlots = filledSlots
}
}
return bestSlots
}
func AddToCopyOfList[T any](list []T, item T) []T {
listCopy := make([]T, len(list))
copy(listCopy, list)
return append(listCopy, item)
}