podman
1/**
2* Copyright 2023 ByteDance Inc.
3*
4* Licensed under the Apache License, Version 2.0 (the "License");
5* you may not use this file except in compliance with the License.
6* You may obtain a copy of the License at
7*
8* http://www.apache.org/licenses/LICENSE-2.0
9*
10* Unless required by applicable law or agreed to in writing, software
11* distributed under the License is distributed on an "AS IS" BASIS,
12* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13* See the License for the specific language governing permissions and
14* limitations under the License.
15*/
16
17package rt18
19import (20`fmt`21`strings`22`unsafe`23
24)
25
26type Bitmap struct {27N int28B []byte29}
30
31func (self *Bitmap) grow() {32if self.N >= len(self.B) * 8 {33self.B = append(self.B, 0)34}35}
36
37func (self *Bitmap) mark(i int, bv int) {38if bv != 0 {39self.B[i / 8] |= 1 << (i % 8)40} else {41self.B[i / 8] &^= 1 << (i % 8)42}43}
44
45func (self *Bitmap) Set(i int, bv int) {46if i >= self.N {47panic("bitmap: invalid bit position")48} else {49self.mark(i, bv)50}51}
52
53func (self *Bitmap) Append(bv int) {54self.grow()55self.mark(self.N, bv)56self.N++57}
58
59func (self *Bitmap) AppendMany(n int, bv int) {60for i := 0; i < n; i++ {61self.Append(bv)62}63}
64
65// var (
66// _stackMapLock = sync.Mutex{}
67// _stackMapCache = make(map[*StackMap]struct{})
68// )
69
70type BitVec struct {71N uintptr72B unsafe.Pointer73}
74
75func (self BitVec) Bit(i uintptr) byte {76return (*(*byte)(unsafe.Pointer(uintptr(self.B) + i / 8)) >> (i % 8)) & 177}
78
79func (self BitVec) String() string {80var i uintptr81var v []string82
83/* add each bit */84for i = 0; i < self.N; i++ {85v = append(v, fmt.Sprintf("%d", self.Bit(i)))86}87
88/* join them together */89return fmt.Sprintf(90"BitVec { %s }",91strings.Join(v, ", "),92)93}
94
95type StackMap struct {96N int3297L int3298B [1]byte99}
100
101// func (self *StackMap) add() {
102// _stackMapLock.Lock()
103// _stackMapCache[self] = struct{}{}
104// _stackMapLock.Unlock()
105// }
106
107func (self *StackMap) Pin() uintptr {108// self.add()109return uintptr(unsafe.Pointer(self))110}
111
112func (self *StackMap) Get(i int32) BitVec {113return BitVec {114N: uintptr(self.L),115B: unsafe.Pointer(uintptr(unsafe.Pointer(&self.B)) + uintptr(i * ((self.L + 7) >> 3))),116}117}
118
119func (self *StackMap) String() string {120sb := strings.Builder{}121sb.WriteString("StackMap {")122
123/* dump every stack map */124for i := int32(0); i < self.N; i++ {125sb.WriteRune('\n')126sb.WriteString(" " + self.Get(i).String())127}128
129/* close the stackmap */130sb.WriteString("\n}")131return sb.String()132}
133
134func (self *StackMap) MarshalBinary() ([]byte, error) {135size := int(self.N) * int(self.L) + int(unsafe.Sizeof(self.L)) + int(unsafe.Sizeof(self.N))136return BytesFrom(unsafe.Pointer(self), size, size), nil137}
138
139var (140byteType = UnpackEface(byte(0)).Type141)
142
143const (144_StackMapSize = unsafe.Sizeof(StackMap{})145)
146
147//go:linkname mallocgc runtime.mallocgc
148//goland:noinspection GoUnusedParameter
149func mallocgc(nb uintptr, vt *GoType, zero bool) unsafe.Pointer150
151type StackMapBuilder struct {152b Bitmap
153}
154
155//go:nocheckptr
156func (self *StackMapBuilder) Build() (p *StackMap) {157nb := len(self.b.B)158bm := mallocgc(_StackMapSize + uintptr(nb) - 1, byteType, false)159
160/* initialize as 1 bitmap of N bits */161p = (*StackMap)(bm)162p.N, p.L = 1, int32(self.b.N)163copy(BytesFrom(unsafe.Pointer(&p.B), nb, nb), self.b.B)164return165}
166
167func (self *StackMapBuilder) AddField(ptr bool) {168if ptr {169self.b.Append(1)170} else {171self.b.Append(0)172}173}
174
175func (self *StackMapBuilder) AddFields(n int, ptr bool) {176if ptr {177self.b.AppendMany(n, 1)178} else {179self.b.AppendMany(n, 0)180}181}