This commit is contained in:
Aliaksandr Valialkin 2024-05-22 11:38:10 +02:00
parent df6110bf06
commit ddc3914fa7
No known key found for this signature in database
GPG key ID: 52C003EE2BCDB9EB
2 changed files with 21 additions and 103 deletions

View file

@ -2,7 +2,6 @@ package logstorage
import ( import (
"fmt" "fmt"
"unsafe"
) )
// pipeExtract processes '| extract from <field> <pattern>' pipe. // pipeExtract processes '| extract from <field> <pattern>' pipe.
@ -69,109 +68,20 @@ func (pe *pipeExtract) updateNeededFields(neededFields, unneededFields fieldsSet
} }
func (pe *pipeExtract) newPipeProcessor(workersCount int, _ <-chan struct{}, _ func(), ppBase pipeProcessor) pipeProcessor { func (pe *pipeExtract) newPipeProcessor(workersCount int, _ <-chan struct{}, _ func(), ppBase pipeProcessor) pipeProcessor {
shards := make([]pipeExtractProcessorShard, workersCount) patterns := make([]*pattern, workersCount)
for i := range shards { for i := range patterns {
shards[i] = pipeExtractProcessorShard{ patterns[i] = newPattern(pe.steps)
pipeExtractProcessorShardNopad: pipeExtractProcessorShardNopad{ }
ef: newPattern(pe.steps),
}, unpackFunc := func(uctx *fieldsUnpackerContext, s, fieldPrefix string) {
ptn := patterns[uctx.workerID]
ptn.apply(s)
for _, f := range ptn.fields {
uctx.addField(f.name, *f.value, fieldPrefix)
} }
} }
pep := &pipeExtractProcessor{ return newPipeUnpackProcessor(workersCount, unpackFunc, ppBase, pe.fromField, "", pe.iff)
pe: pe,
ppBase: ppBase,
shards: shards,
}
return pep
}
type pipeExtractProcessor struct {
pe *pipeExtract
ppBase pipeProcessor
shards []pipeExtractProcessorShard
}
type pipeExtractProcessorShard struct {
pipeExtractProcessorShardNopad
// The padding prevents false sharing on widespread platforms with 128 mod (cache line size) = 0 .
_ [128 - unsafe.Sizeof(pipeExtractProcessorShardNopad{})%128]byte
}
type pipeExtractProcessorShardNopad struct {
ef *pattern
bm bitmap
uctx fieldsUnpackerContext
wctx pipeUnpackWriteContext
}
func (pep *pipeExtractProcessor) writeBlock(workerID uint, br *blockResult) {
if len(br.timestamps) == 0 {
return
}
shard := &pep.shards[workerID]
shard.wctx.init(workerID, br, pep.ppBase)
ef := shard.ef
bm := &shard.bm
bm.init(len(br.timestamps))
bm.setBits()
if iff := pep.pe.iff; iff != nil {
iff.f.applyToBlockResult(br, bm)
if bm.isZero() {
// Fast path - nothing to extract.
pep.ppBase.writeBlock(workerID, br)
return
}
}
c := br.getColumnByName(pep.pe.fromField)
if c.isConst {
v := c.valuesEncoded[0]
ef.apply(v)
for _, f := range ef.fields {
shard.uctx.addField(f.name, *f.value, "")
}
for i := range br.timestamps {
if bm.isSetBit(i) {
shard.wctx.writeRow(i, shard.uctx.fields)
} else {
shard.wctx.writeRow(i, nil)
}
}
} else {
values := c.getValues(br)
vPrevApplied := ""
for i, v := range values {
if bm.isSetBit(i) {
if vPrevApplied != v {
ef.apply(v)
shard.uctx.resetFields()
for _, f := range ef.fields {
shard.uctx.addField(f.name, *f.value, "")
}
vPrevApplied = v
}
shard.wctx.writeRow(i, shard.uctx.fields)
} else {
shard.wctx.writeRow(i, nil)
}
}
}
shard.wctx.flush()
shard.uctx.reset()
}
func (pep *pipeExtractProcessor) flush() error {
return nil
} }
func parsePipeExtract(lex *lexer) (*pipeExtract, error) { func parsePipeExtract(lex *lexer) (*pipeExtract, error) {

View file

@ -7,11 +7,13 @@ import (
) )
type fieldsUnpackerContext struct { type fieldsUnpackerContext struct {
fields []Field workerID uint
a arena fields []Field
a arena
} }
func (uctx *fieldsUnpackerContext) reset() { func (uctx *fieldsUnpackerContext) reset() {
uctx.workerID = 0
uctx.resetFields() uctx.resetFields()
uctx.a.reset() uctx.a.reset()
} }
@ -42,6 +44,12 @@ func (uctx *fieldsUnpackerContext) addField(name, value, fieldPrefix string) {
func newPipeUnpackProcessor(workersCount int, unpackFunc func(uctx *fieldsUnpackerContext, s, fieldPrefix string), ppBase pipeProcessor, func newPipeUnpackProcessor(workersCount int, unpackFunc func(uctx *fieldsUnpackerContext, s, fieldPrefix string), ppBase pipeProcessor,
fromField, fieldPrefix string, iff *ifFilter) *pipeUnpackProcessor { fromField, fieldPrefix string, iff *ifFilter) *pipeUnpackProcessor {
shards := make([]pipeUnpackProcessorShard, workersCount)
for i := range shards {
shards[i].wctx.workerID = uint(i)
}
return &pipeUnpackProcessor{ return &pipeUnpackProcessor{
unpackFunc: unpackFunc, unpackFunc: unpackFunc,
ppBase: ppBase, ppBase: ppBase,