-
Notifications
You must be signed in to change notification settings - Fork 1.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Change the memory queue implementation to not pre-allocate capacity o…
…bjects. (#12070) Change the sized channel to use a "fifo" queue without pre-allocated capacity: * Allows us to soon allow different sizers like bytes, which otherwise would pre-allocate way too much memory. E.g. if configure with a limit of 1GB would pre-allocate 16GB for the buffered channel only. * Does not pre-allocate memory when not needed. **Before:** ``` cpu: Apple M2 Max BenchmarkOffer BenchmarkOffer/requests_based BenchmarkOffer/requests_based-12 5559658 207.3 ns/op 0 B/op 0 allocs/op BenchmarkOffer/items_based BenchmarkOffer/items_based-12 6025567 209.4 ns/op 0 B/op 0 allocs/op ``` **After:** ``` cpu: Apple M2 Max BenchmarkOffer BenchmarkOffer/requests_based BenchmarkOffer/requests_based-12 4904857 224.3 ns/op 48 B/op 1 allocs/op BenchmarkOffer/items_based BenchmarkOffer/items_based-12 5470143 229.9 ns/op 48 B/op 1 allocs/op ``` Overall there is a ~20ns performance downgrade but this is expected since no memory pre-allocation. Signed-off-by: Bogdan Drutu <[email protected]>
- Loading branch information
1 parent
e6df9d8
commit 7f4664e
Showing
6 changed files
with
251 additions
and
152 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
# Use this changelog template to create an entry for release notes. | ||
|
||
# One of 'breaking', 'deprecation', 'new_component', 'enhancement', 'bug_fix' | ||
change_type: enhancement | ||
|
||
# The name of the component, or a single word describing the area of concern, (e.g. otlpreceiver) | ||
component: exporterhelper | ||
|
||
# A brief description of the change. Surround your text with quotes ("") if it needs to start with a backtick (`). | ||
note: Change the memory queue implementation to not pre-allocate capacity objects. | ||
|
||
# One or more tracking issues or pull requests related to the change | ||
issues: [12070] | ||
|
||
# (Optional) One or more lines of additional information to render under the primary note. | ||
# These lines will be padded with 2 spaces and then inserted directly into the document. | ||
# Use pipe (|) for multiline entries. | ||
subtext: This change improves memory usage of the collector under low utilization and is a prerequisite for supporting different other size limitations (number of items, bytes). | ||
|
||
# Optional: The change log or logs in which this entry should be included. | ||
# e.g. '[user]' or '[user, api]' | ||
# Include 'user' if the change is relevant to end users. | ||
# Include 'api' if there is a change to a library API. | ||
# Default: '[user]' | ||
change_logs: [user] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
// Copyright The OpenTelemetry Authors | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
package exporterqueue // import "go.opentelemetry.io/collector/exporter/exporterqueue" | ||
|
||
import ( | ||
"context" | ||
"errors" | ||
"sync" | ||
) | ||
|
||
var errInvalidSize = errors.New("invalid element size") | ||
|
||
type node[T any] struct { | ||
ctx context.Context | ||
data T | ||
size int64 | ||
next *node[T] | ||
} | ||
|
||
type linkedQueue[T any] struct { | ||
head *node[T] | ||
tail *node[T] | ||
} | ||
|
||
func (l *linkedQueue[T]) push(ctx context.Context, data T, size int64) { | ||
n := &node[T]{ctx: ctx, data: data, size: size} | ||
if l.tail == nil { | ||
l.head = n | ||
l.tail = n | ||
return | ||
} | ||
l.tail.next = n | ||
l.tail = n | ||
} | ||
|
||
func (l *linkedQueue[T]) pop() (context.Context, T, int64) { | ||
n := l.head | ||
l.head = n.next | ||
if l.head == nil { | ||
l.tail = nil | ||
} | ||
n.next = nil | ||
return n.ctx, n.data, n.size | ||
} | ||
|
||
// sizedQueue is a channel wrapper for sized elements with a capacity set to a total size of all the elements. | ||
// The channel will accept elements until the total size of the elements reaches the capacity. | ||
type sizedQueue[T any] struct { | ||
sizer sizer[T] | ||
cap int64 | ||
|
||
mu sync.Mutex | ||
hasElements *sync.Cond | ||
items *linkedQueue[T] | ||
size int64 | ||
stopped bool | ||
} | ||
|
||
// newSizedQueue creates a sized elements channel. Each element is assigned a size by the provided sizer. | ||
// capacity is the capacity of the queue. | ||
func newSizedQueue[T any](capacity int64, sizer sizer[T]) *sizedQueue[T] { | ||
sq := &sizedQueue[T]{ | ||
sizer: sizer, | ||
cap: capacity, | ||
items: &linkedQueue[T]{}, | ||
} | ||
sq.hasElements = sync.NewCond(&sq.mu) | ||
return sq | ||
} | ||
|
||
// Offer puts the element into the queue with the given sized if there is enough capacity. | ||
// Returns an error if the queue is full. | ||
func (sq *sizedQueue[T]) Offer(ctx context.Context, el T) error { | ||
elSize := sq.sizer.Sizeof(el) | ||
if elSize == 0 { | ||
return nil | ||
} | ||
|
||
if elSize <= 0 { | ||
return errInvalidSize | ||
} | ||
|
||
sq.mu.Lock() | ||
defer sq.mu.Unlock() | ||
|
||
if sq.size+elSize > sq.cap { | ||
return ErrQueueIsFull | ||
} | ||
|
||
sq.size += elSize | ||
sq.items.push(ctx, el, elSize) | ||
// Signal one consumer if any. | ||
sq.hasElements.Signal() | ||
return nil | ||
} | ||
|
||
// pop removes the element from the queue and returns it. | ||
// The call blocks until there is an item available or the queue is stopped. | ||
// The function returns true when an item is consumed or false if the queue is stopped and emptied. | ||
func (sq *sizedQueue[T]) pop() (context.Context, T, bool) { | ||
sq.mu.Lock() | ||
defer sq.mu.Unlock() | ||
|
||
for { | ||
if sq.size > 0 { | ||
ctx, el, elSize := sq.items.pop() | ||
sq.size -= elSize | ||
return ctx, el, true | ||
} | ||
|
||
if sq.stopped { | ||
var el T | ||
return context.Background(), el, false | ||
} | ||
|
||
sq.hasElements.Wait() | ||
} | ||
} | ||
|
||
// Shutdown closes the queue channel to initiate draining of the queue. | ||
func (sq *sizedQueue[T]) Shutdown(context.Context) error { | ||
sq.mu.Lock() | ||
defer sq.mu.Unlock() | ||
sq.stopped = true | ||
sq.hasElements.Broadcast() | ||
return nil | ||
} | ||
|
||
func (sq *sizedQueue[T]) Size() int { | ||
sq.mu.Lock() | ||
defer sq.mu.Unlock() | ||
return int(sq.size) | ||
} | ||
|
||
func (sq *sizedQueue[T]) Capacity() int { | ||
return int(sq.cap) | ||
} |
Oops, something went wrong.