zclwrite: use a single, flat writer token buffer
Previously we were allocating a separate heap object for each token, which creates a lot of small objects for the GC to manage. Since we know that we're always converting from a flat array of native tokens, we can produce a flat array of writer tokens first and _then_ take pointers into that array to achieve our goal of making a slice of pointers. For the use-case of formatting a sequence of tokens by tweaking the "SpacesBefore" value, this means we can get all of our memory allocation done in a single chunk and then just tweak the allocated, contiguous tokens in-place, which should reduce memory pressure for a task which will likely be done frequently by a text editor integration doing "format on save".
This commit is contained in:
parent
3c0dde2ae5
commit
c233270a9b
@ -133,7 +133,10 @@ func parseBody(nativeBody *zclsyntax.Body, from inputTokens) (inputTokens, *Body
|
|||||||
// indices as the input, allowing the two sets of tokens to be correlated
|
// indices as the input, allowing the two sets of tokens to be correlated
|
||||||
// by index.
|
// by index.
|
||||||
func writerTokens(nativeTokens zclsyntax.Tokens) Tokens {
|
func writerTokens(nativeTokens zclsyntax.Tokens) Tokens {
|
||||||
ret := make(Tokens, len(nativeTokens))
|
// Ultimately we want a slice of token _pointers_, but since we can
|
||||||
|
// predict how much memory we're going to devote to tokens we'll allocate
|
||||||
|
// it all as a single flat buffer and thus give the GC less work to do.
|
||||||
|
tokBuf := make([]Token, len(nativeTokens))
|
||||||
var lastByteOffset int
|
var lastByteOffset int
|
||||||
for i, mainToken := range nativeTokens {
|
for i, mainToken := range nativeTokens {
|
||||||
// Create a copy of the bytes so that we can mutate without
|
// Create a copy of the bytes so that we can mutate without
|
||||||
@ -141,7 +144,7 @@ func writerTokens(nativeTokens zclsyntax.Tokens) Tokens {
|
|||||||
bytes := make([]byte, len(mainToken.Bytes))
|
bytes := make([]byte, len(mainToken.Bytes))
|
||||||
copy(bytes, mainToken.Bytes)
|
copy(bytes, mainToken.Bytes)
|
||||||
|
|
||||||
ret[i] = &Token{
|
tokBuf[i] = Token{
|
||||||
Type: mainToken.Type,
|
Type: mainToken.Type,
|
||||||
Bytes: bytes,
|
Bytes: bytes,
|
||||||
|
|
||||||
@ -154,6 +157,12 @@ func writerTokens(nativeTokens zclsyntax.Tokens) Tokens {
|
|||||||
lastByteOffset = mainToken.Range.End.Byte
|
lastByteOffset = mainToken.Range.End.Byte
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Now make a slice of pointers into the previous slice.
|
||||||
|
ret := make(Tokens, len(tokBuf))
|
||||||
|
for i := range ret {
|
||||||
|
ret[i] = &tokBuf[i]
|
||||||
|
}
|
||||||
|
|
||||||
return ret
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user