mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
This change takes the EvEndOfGeneration event and promotes it to a real event that appears in the trace. This allows the trace parser to unambiguously identify truncated traces vs. broken traces. It also makes a lot of the logic around parsing simpler, because there's no more batch spilling necessary. Fixes #73904. Change-Id: I37c359b32b6b5f894825aafc02921adeaacf2595 Reviewed-on: https://go-review.googlesource.com/c/go/+/693398 Reviewed-by: Carlos Amedee <carlos@golang.org> Reviewed-by: Michael Pratt <mpratt@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
88 lines
2.2 KiB
Go
88 lines
2.2 KiB
Go
// Copyright 2025 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package trace
|
|
|
|
import (
|
|
"fmt"
|
|
"internal/trace/tracev2"
|
|
)
|
|
|
|
// timestamp is an unprocessed timestamp.
|
|
type timestamp uint64
|
|
|
|
type batch struct {
|
|
time timestamp
|
|
gen uint64
|
|
data []byte
|
|
}
|
|
|
|
// readBatch copies b and parses the trace batch header inside.
|
|
// Returns the batch, bytes read, and an error.
|
|
func readBatch(b []byte) (batch, uint64, error) {
|
|
if len(b) == 0 {
|
|
return batch{}, 0, fmt.Errorf("batch is empty")
|
|
}
|
|
data := make([]byte, len(b))
|
|
copy(data, b)
|
|
|
|
// Read batch header byte.
|
|
if typ := tracev2.EventType(b[0]); typ == tracev2.EvEndOfGeneration {
|
|
if len(b) != 1 {
|
|
return batch{}, 1, fmt.Errorf("unexpected end of generation in batch of size >1")
|
|
}
|
|
return batch{data: data}, 1, nil
|
|
}
|
|
if typ := tracev2.EventType(b[0]); typ != tracev2.EvEventBatch && typ != tracev2.EvExperimentalBatch {
|
|
return batch{}, 1, fmt.Errorf("expected batch event, got event %d", typ)
|
|
}
|
|
total := 1
|
|
b = b[1:]
|
|
|
|
// Read the generation
|
|
gen, n, err := readUvarint(b)
|
|
if err != nil {
|
|
return batch{}, uint64(total + n), fmt.Errorf("error reading batch gen: %w", err)
|
|
}
|
|
total += n
|
|
b = b[n:]
|
|
|
|
// Read the M (discard it).
|
|
_, n, err = readUvarint(b)
|
|
if err != nil {
|
|
return batch{}, uint64(total + n), fmt.Errorf("error reading batch M ID: %w", err)
|
|
}
|
|
total += n
|
|
b = b[n:]
|
|
|
|
// Read the timestamp.
|
|
ts, n, err := readUvarint(b)
|
|
if err != nil {
|
|
return batch{}, uint64(total + n), fmt.Errorf("error reading batch timestamp: %w", err)
|
|
}
|
|
total += n
|
|
b = b[n:]
|
|
|
|
// Read the size of the batch to follow.
|
|
size, n, err := readUvarint(b)
|
|
if err != nil {
|
|
return batch{}, uint64(total + n), fmt.Errorf("error reading batch size: %w", err)
|
|
}
|
|
if size > tracev2.MaxBatchSize {
|
|
return batch{}, uint64(total + n), fmt.Errorf("invalid batch size %d, maximum is %d", size, tracev2.MaxBatchSize)
|
|
}
|
|
total += n
|
|
total += int(size)
|
|
if total != len(data) {
|
|
return batch{}, uint64(total), fmt.Errorf("expected complete batch")
|
|
}
|
|
data = data[:total]
|
|
|
|
// Return the batch.
|
|
return batch{
|
|
gen: gen,
|
|
time: timestamp(ts),
|
|
data: data,
|
|
}, uint64(total), nil
|
|
}
|