mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
84 lines
2.2 KiB
Go
84 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
|
||
|
|
|
||
|
|
// batch represents a batch of trace events.
|
||
|
|
// It is unparsed except for its header.
|
||
|
|
type batch struct {
|
||
|
|
m threadID
|
||
|
|
time timestamp
|
||
|
|
data []byte
|
||
|
|
}
|
||
|
|
|
||
|
|
// threadID is the runtime-internal M structure's ID. This is unique
|
||
|
|
// for each OS thread.
|
||
|
|
type threadID int64
|
||
|
|
|
||
|
|
// readBatch copies b and parses the trace batch header inside.
|
||
|
|
// Returns the batch, the generation, bytes read, and an error.
|
||
|
|
func readBatch(b []byte) (batch, uint64, uint64, error) {
|
||
|
|
if len(b) == 0 {
|
||
|
|
return batch{}, 0, 0, fmt.Errorf("batch is empty")
|
||
|
|
}
|
||
|
|
data := make([]byte, len(b))
|
||
|
|
if nw := copy(data, b); nw != len(b) {
|
||
|
|
return batch{}, 0, 0, fmt.Errorf("unexpected error copying batch")
|
||
|
|
}
|
||
|
|
// Read batch header byte.
|
||
|
|
if typ := tracev2.EventType(b[0]); typ != tracev2.EvEventBatch && typ != tracev2.EvExperimentalBatch {
|
||
|
|
return batch{}, 0, 1, fmt.Errorf("expected batch event, got event %d", typ)
|
||
|
|
}
|
||
|
|
|
||
|
|
// Read the batch header: gen (generation), thread (M) ID, base timestamp
|
||
|
|
// for the batch.
|
||
|
|
total := 1
|
||
|
|
b = b[1:]
|
||
|
|
gen, n, err := readUvarint(b)
|
||
|
|
if err != nil {
|
||
|
|
return batch{}, gen, uint64(total + n), fmt.Errorf("error reading batch gen: %w", err)
|
||
|
|
}
|
||
|
|
total += n
|
||
|
|
b = b[n:]
|
||
|
|
m, n, err := readUvarint(b)
|
||
|
|
if err != nil {
|
||
|
|
return batch{}, gen, uint64(total + n), fmt.Errorf("error reading batch M ID: %w", err)
|
||
|
|
}
|
||
|
|
total += n
|
||
|
|
b = b[n:]
|
||
|
|
ts, n, err := readUvarint(b)
|
||
|
|
if err != nil {
|
||
|
|
return batch{}, gen, uint64(total + n), fmt.Errorf("error reading batch timestamp: %w", err)
|
||
|
|
}
|
||
|
|
total += n
|
||
|
|
b = b[n:]
|
||
|
|
|
||
|
|
// Read in the size of the batch to follow.
|
||
|
|
size, n, err := readUvarint(b)
|
||
|
|
if err != nil {
|
||
|
|
return batch{}, gen, uint64(total + n), fmt.Errorf("error reading batch size: %w", err)
|
||
|
|
}
|
||
|
|
if size > tracev2.MaxBatchSize {
|
||
|
|
return batch{}, gen, uint64(total + n), fmt.Errorf("invalid batch size %d, maximum is %d", size, tracev2.MaxBatchSize)
|
||
|
|
}
|
||
|
|
total += n
|
||
|
|
total += int(size)
|
||
|
|
data = data[:total]
|
||
|
|
|
||
|
|
// Return the batch.
|
||
|
|
return batch{
|
||
|
|
m: threadID(m),
|
||
|
|
time: timestamp(ts),
|
||
|
|
data: data,
|
||
|
|
}, gen, uint64(total), nil
|
||
|
|
}
|