150 lines
4.2 KiB
Go
150 lines
4.2 KiB
Go
// Copyright 2024 PingCAP, Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package internal
|
|
|
|
import (
|
|
"reflect"
|
|
"unsafe"
|
|
)
|
|
|
|
// Sizeof returns the size of 'v' in bytes.
|
|
// If there is an error during calculation, Of returns -1.
|
|
func Sizeof(v any) int {
|
|
// Cache with every visited pointer so we don't count two pointers
|
|
// to the same memory twice.
|
|
cache := make(map[uintptr]bool)
|
|
return sizeOf(reflect.Indirect(reflect.ValueOf(v)), cache)
|
|
}
|
|
|
|
// sizeOf returns the number of bytes the actual data represented by v occupies in memory.
|
|
// If there is an error, sizeOf returns -1.
|
|
func sizeOf(v reflect.Value, cache map[uintptr]bool) int {
|
|
switch v.Kind() {
|
|
case reflect.Array:
|
|
sum := 0
|
|
for i := 0; i < v.Len(); i++ {
|
|
s := sizeOf(v.Index(i), cache)
|
|
if s < 0 {
|
|
return -1
|
|
}
|
|
sum += s
|
|
}
|
|
return sum + (v.Cap()-v.Len())*int(v.Type().Elem().Size())
|
|
case reflect.Slice:
|
|
// return 0 if this node has been visited already
|
|
if cache[v.Pointer()] {
|
|
return 0
|
|
}
|
|
cache[v.Pointer()] = true
|
|
sum := 0
|
|
for i := 0; i < v.Len(); i++ {
|
|
s := sizeOf(v.Index(i), cache)
|
|
if s < 0 {
|
|
return -1
|
|
}
|
|
sum += s
|
|
}
|
|
sum += (v.Cap() - v.Len()) * int(v.Type().Elem().Size())
|
|
return sum + int(v.Type().Size())
|
|
case reflect.Struct:
|
|
sum := 0
|
|
for i, n := 0, v.NumField(); i < n; i++ {
|
|
s := sizeOf(v.Field(i), cache)
|
|
if s < 0 {
|
|
return -1
|
|
}
|
|
sum += s
|
|
}
|
|
// Look for struct padding.
|
|
padding := int(v.Type().Size())
|
|
for i, n := 0, v.NumField(); i < n; i++ {
|
|
padding -= int(v.Field(i).Type().Size())
|
|
}
|
|
return sum + padding
|
|
case reflect.String:
|
|
s := v.String()
|
|
hdr := (*reflect.StringHeader)(unsafe.Pointer(&s))
|
|
if cache[hdr.Data] {
|
|
return int(v.Type().Size())
|
|
}
|
|
cache[hdr.Data] = true
|
|
return len(s) + int(v.Type().Size())
|
|
case reflect.Ptr:
|
|
// return Ptr size if this node has been visited already (infinite recursion)
|
|
if cache[v.Pointer()] {
|
|
return int(v.Type().Size())
|
|
}
|
|
cache[v.Pointer()] = true
|
|
if v.IsNil() {
|
|
return int(reflect.New(v.Type()).Type().Size())
|
|
}
|
|
s := sizeOf(reflect.Indirect(v), cache)
|
|
if s < 0 {
|
|
return -1
|
|
}
|
|
return s + int(v.Type().Size())
|
|
case reflect.Bool,
|
|
reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64,
|
|
reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
|
reflect.Int, reflect.Uint,
|
|
reflect.Chan,
|
|
reflect.Uintptr,
|
|
reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128,
|
|
reflect.Func:
|
|
return int(v.Type().Size())
|
|
case reflect.Map:
|
|
// return 0 if this node has been visited already (infinite recursion)
|
|
if cache[v.Pointer()] {
|
|
return 0
|
|
}
|
|
cache[v.Pointer()] = true
|
|
sum := 0
|
|
keys := v.MapKeys()
|
|
for i := range keys {
|
|
val := v.MapIndex(keys[i])
|
|
// calculate size of key and value separately
|
|
sv := sizeOf(val, cache)
|
|
if sv < 0 {
|
|
return -1
|
|
}
|
|
sum += sv
|
|
sk := sizeOf(keys[i], cache)
|
|
if sk < 0 {
|
|
return -1
|
|
}
|
|
sum += sk
|
|
}
|
|
// Include overhead due to unused map buckets. 10.79 comes
|
|
// from https://golang.org/src/runtime/map.go.
|
|
return sum + int(v.Type().Size()) + int(float64(len(keys))*10.79)
|
|
case reflect.Interface:
|
|
// A hack here.
|
|
// When we calculate the size of a table.PartitionTable, it somehow reference the kv.Storage object.
|
|
// For mockstore.Storage instance (unistore), it's too large to follow the references.
|
|
// So just take it as a pointer.
|
|
switch v.Type().Name() {
|
|
case "Storage", "Client", "Allocator":
|
|
return 8
|
|
}
|
|
return sizeOf(v.Elem(), cache) + int(v.Type().Size())
|
|
case reflect.UnsafePointer:
|
|
// It's difficult to get the size of an unsafe pointer.
|
|
// Just treat the size as machine word and ignore its reference.
|
|
s := int(v.Type().Size())
|
|
return s
|
|
}
|
|
return -1
|
|
}
|