| // Copyright 2022 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 cmerge |
| |
| // package cmerge provides a few small utility APIs for helping |
| // with merging of counter data for a given function. |
| |
| import ( |
| "fmt" |
| "internal/coverage" |
| "math" |
| ) |
| |
| // Merger provides state and methods to help manage the process of |
| // merging together coverage counter data for a given function, for |
| // tools that need to implicitly merge counter as they read multiple |
| // coverage counter data files. |
| type Merger struct { |
| cmode coverage.CounterMode |
| cgran coverage.CounterGranularity |
| overflow bool |
| } |
| |
| // MergeCounters takes the counter values in 'src' and merges them |
| // into 'dst' according to the correct counter mode. |
| func (m *Merger) MergeCounters(dst, src []uint32) (error, bool) { |
| if len(src) != len(dst) { |
| return fmt.Errorf("merging counters: len(dst)=%d len(src)=%d", len(dst), len(src)), false |
| } |
| if m.cmode == coverage.CtrModeSet { |
| for i := 0; i < len(src); i++ { |
| if src[i] != 0 { |
| dst[i] = 1 |
| } |
| } |
| } else { |
| for i := 0; i < len(src); i++ { |
| dst[i] = m.SaturatingAdd(dst[i], src[i]) |
| } |
| } |
| ovf := m.overflow |
| m.overflow = false |
| return nil, ovf |
| } |
| |
| // Saturating add does a saturating addition of 'dst' and 'src', |
| // returning added value or math.MaxUint32 if there is an overflow. |
| // Overflows are recorded in case the client needs to track them. |
| func (m *Merger) SaturatingAdd(dst, src uint32) uint32 { |
| result, overflow := SaturatingAdd(dst, src) |
| if overflow { |
| m.overflow = true |
| } |
| return result |
| } |
| |
| // Saturating add does a saturing addition of 'dst' and 'src', |
| // returning added value or math.MaxUint32 plus an overflow flag. |
| func SaturatingAdd(dst, src uint32) (uint32, bool) { |
| d, s := uint64(dst), uint64(src) |
| sum := d + s |
| overflow := false |
| if uint64(uint32(sum)) != sum { |
| overflow = true |
| sum = math.MaxUint32 |
| } |
| return uint32(sum), overflow |
| } |
| |
| // SetModeAndGranularity records the counter mode and granularity for |
| // the current merge. In the specific case of merging across coverage |
| // data files from different binaries, where we're combining data from |
| // more than one meta-data file, we need to check for mode/granularity |
| // clashes. |
| func (cm *Merger) SetModeAndGranularity(mdf string, cmode coverage.CounterMode, cgran coverage.CounterGranularity) error { |
| // Collect counter mode and granularity so as to detect clashes. |
| if cm.cmode != coverage.CtrModeInvalid { |
| if cm.cmode != cmode { |
| return fmt.Errorf("counter mode clash while reading meta-data file %s: previous file had %s, new file has %s", mdf, cm.cmode.String(), cmode.String()) |
| } |
| if cm.cgran != cgran { |
| return fmt.Errorf("counter granularity clash while reading meta-data file %s: previous file had %s, new file has %s", mdf, cm.cgran.String(), cgran.String()) |
| } |
| } |
| cm.cmode = cmode |
| cm.cgran = cgran |
| return nil |
| } |
| |
| func (cm *Merger) ResetModeAndGranularity() { |
| cm.cmode = coverage.CtrModeInvalid |
| cm.cgran = coverage.CtrGranularityInvalid |
| cm.overflow = false |
| } |
| |
| func (cm *Merger) Mode() coverage.CounterMode { |
| return cm.cmode |
| } |
| |
| func (cm *Merger) Granularity() coverage.CounterGranularity { |
| return cm.cgran |
| } |