-
Notifications
You must be signed in to change notification settings - Fork 3.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Robert/addr memory leak #8717
Robert/addr memory leak #8717
Changes from 5 commits
b69e19b
280168c
60e6d71
ecc288e
991169a
fbe6b78
2294920
7e577b9
0a3ff3f
3b1f40b
a02c0d4
a890b1e
4d26f57
cbbf8b3
f71691f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,6 +16,7 @@ import ( | |
"github.com/cosmos/cosmos-sdk/types/address" | ||
"github.com/cosmos/cosmos-sdk/types/bech32" | ||
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors" | ||
"github.com/hashicorp/golang-lru/simplelru" | ||
) | ||
|
||
const ( | ||
|
@@ -72,6 +73,22 @@ const ( | |
Bech32PrefixConsPub = Bech32MainPrefix + PrefixValidator + PrefixConsensus + PrefixPublic | ||
) | ||
|
||
// cache variables | ||
var ( | ||
// AccAddress.String() is expensive and if unoptimized dominantly showed up in profiles, | ||
// yet has no mechanisms to trivially cache the result given that AccAddress is a []byte type. | ||
addrMu sync.Mutex | ||
addrCache *simplelru.LRU | ||
) | ||
|
||
func init() { | ||
var err error | ||
addrCache, err = simplelru.NewLRU(1000, nil) | ||
if err != nil { | ||
panic(err) | ||
} | ||
} | ||
|
||
// Address is a common interface for different types of addresses used by the SDK | ||
type Address interface { | ||
Equals(Address) bool | ||
|
@@ -158,12 +175,7 @@ func (aa AccAddress) Equals(aa2 Address) bool { | |
|
||
// Returns boolean for whether an AccAddress is empty | ||
func (aa AccAddress) Empty() bool { | ||
if aa == nil { | ||
return true | ||
} | ||
|
||
aa2 := AccAddress{} | ||
return bytes.Equal(aa.Bytes(), aa2.Bytes()) | ||
return aa == nil || len(aa) == 0 | ||
} | ||
|
||
// Marshal returns the raw address bytes. It is needed for protobuf | ||
|
@@ -237,39 +249,30 @@ func (aa AccAddress) Bytes() []byte { | |
return aa | ||
} | ||
|
||
// AccAddress.String() is expensive and if unoptimized dominantly showed up in profiles, | ||
// yet has no mechanisms to trivially cache the result given that AccAddress is a []byte type. | ||
// With the string interning below, we are able to achieve zero cost allocations for string->[]byte | ||
// because the Go compiler recognizes the special case of map[string([]byte)] when used exactly | ||
// in that pattern. See https://github.com/cosmos/cosmos-sdk/issues/8693. | ||
odeke-em marked this conversation as resolved.
Show resolved
Hide resolved
|
||
var addMu sync.Mutex | ||
var addrStrMemoize = make(map[string]string) | ||
|
||
// String implements the Stringer interface. | ||
func (aa AccAddress) String() (str string) { | ||
addMu.Lock() | ||
defer addMu.Unlock() | ||
|
||
if str, ok := addrStrMemoize[string(aa)]; ok { | ||
return str | ||
func (aa AccAddress) String() string { | ||
var key = string(aa) | ||
odeke-em marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if str, ok := addrCache.Get(key); ok { | ||
return str.(string) | ||
} | ||
|
||
// Otherwise cache it for later memoization. | ||
defer func() { | ||
addrStrMemoize[string(aa)] = str | ||
}() | ||
|
||
if aa.Empty() { | ||
addrMu.Lock() | ||
addrCache.Add(key, "") | ||
addrMu.Unlock() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There is no need to add this, I'd move all this into the defer as it was before. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Defer is doing extra allocation ;) |
||
return "" | ||
} | ||
|
||
bech32PrefixAccAddr := GetConfig().GetBech32AccountAddrPrefix() | ||
|
||
bech32Addr, err := bech32.ConvertAndEncode(bech32PrefixAccAddr, aa.Bytes()) | ||
bech32Addr, err := bech32.ConvertAndEncode(bech32PrefixAccAddr, aa) | ||
alessio marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if err != nil { | ||
panic(err) | ||
} | ||
|
||
addrMu.Lock() | ||
addrCache.Add(key, bech32Addr) | ||
addrMu.Unlock() | ||
return bech32Addr | ||
} | ||
|
||
|
@@ -332,12 +335,7 @@ func (va ValAddress) Equals(va2 Address) bool { | |
|
||
// Returns boolean for whether an AccAddress is empty | ||
func (va ValAddress) Empty() bool { | ||
if va == nil { | ||
return true | ||
} | ||
|
||
va2 := ValAddress{} | ||
return bytes.Equal(va.Bytes(), va2.Bytes()) | ||
return va == nil || len(va) == 0 | ||
} | ||
|
||
// Marshal returns the raw address bytes. It is needed for protobuf | ||
|
@@ -492,12 +490,7 @@ func (ca ConsAddress) Equals(ca2 Address) bool { | |
|
||
// Returns boolean for whether an ConsAddress is empty | ||
func (ca ConsAddress) Empty() bool { | ||
if ca == nil { | ||
return true | ||
} | ||
|
||
ca2 := ConsAddress{} | ||
return bytes.Equal(ca.Bytes(), ca2.Bytes()) | ||
return ca == nil || len(ca) == 0 | ||
} | ||
|
||
// Marshal returns the raw address bytes. It is needed for protobuf | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,6 +11,18 @@ import ( | |
"github.com/cosmos/cosmos-sdk/types" | ||
) | ||
|
||
func BenchmarkAccAddressString(b *testing.B) { | ||
b.ReportAllocs() | ||
pkBz := make([]byte, ed25519.PubKeySize) | ||
pk := &ed25519.PubKey{Key: pkBz} | ||
aa := pk.Address() | ||
var str string | ||
for i := 0; i < b.N; i++ { | ||
str = aa.String() | ||
} | ||
require.NotEmpty(b, str) | ||
alessio marked this conversation as resolved.
Show resolved
Hide resolved
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. To better get a good benchmark in which code won't be optimized away, use a global variable that's an interface e.g. var sink, revert interface{}
...
for i := 0; i < b.N; i++ {
result := ...
sink = result
}
if sink == nil {
b.Fatal("benchmark never ran")
}
sink = revert There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You mean a variable outside of any function? We are not using it in other benchmarks. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Well just because it wasn't done correctly before doesn't mean we shouldn't do it correctly, I certainly have submitted many other benchmarks where we set to a global sink and check that. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I checked it locally - it doesn't change anything. Are you sure we need to trick the benchmark code? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I am sure about code at times getting optimized away and that being the only way to block that. Humbly, for a bit of clarity, I work on the Go project, across almost all the packages and I am bringing this advice from the many years of working on benchmarking related code :-) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I see. But why to obfuscate more if there is no difference (I verified it)? The |
||
} | ||
|
||
func BenchmarkBech32ifyPubKey(b *testing.B) { | ||
b.ReportAllocs() | ||
pkBz := make([]byte, ed25519.PubKeySize) | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍