mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-20 09:05:26 +00:00
scheduler: add FIFO queue
This is a basic implementation of a first-in-first-out queue with unbounded size. It's useful for cases where a channel with fixed size might deadlock. The caller is responsible for locking.
This commit is contained in:
parent
bae83009d3
commit
639f86915b
110
pkg/scheduler/util/queue/fifo.go
Normal file
110
pkg/scheduler/util/queue/fifo.go
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2024 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package queue
|
||||||
|
|
||||||
|
const (
|
||||||
|
// normalSize limits the size of the buffer that is kept
|
||||||
|
// for reuse.
|
||||||
|
normalSize = 4
|
||||||
|
)
|
||||||
|
|
||||||
|
// FIFO implements a first-in-first-out queue with unbounded size.
|
||||||
|
// The null FIFO is a valid empty queue.
|
||||||
|
//
|
||||||
|
// Access must be protected by the caller when used concurrently by
|
||||||
|
// different goroutines, the queue itself implements no locking.
|
||||||
|
type FIFO[T any] struct {
|
||||||
|
// elements contains a buffer for elements which have been
|
||||||
|
// pushed and not popped yet. Two scenarios are possible:
|
||||||
|
// - one chunk in the middle (start <= end)
|
||||||
|
// - one chunk at the end, followed by one chunk at the
|
||||||
|
// beginning (end <= start)
|
||||||
|
//
|
||||||
|
// start == end can be either an empty queue or a completely
|
||||||
|
// full one (with two chunks).
|
||||||
|
elements []T
|
||||||
|
|
||||||
|
// len counts the number of elements which have been pushed and
|
||||||
|
// not popped yet.
|
||||||
|
len int
|
||||||
|
|
||||||
|
// start is the index of the first valid element.
|
||||||
|
start int
|
||||||
|
|
||||||
|
// end is the index after the last valid element.
|
||||||
|
end int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *FIFO[T]) Len() int {
|
||||||
|
return q.len
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *FIFO[T]) Push(element T) {
|
||||||
|
size := len(q.elements)
|
||||||
|
if q.len == size {
|
||||||
|
// Need larger buffer.
|
||||||
|
newSize := size * 2
|
||||||
|
if newSize == 0 {
|
||||||
|
newSize = normalSize
|
||||||
|
}
|
||||||
|
elements := make([]T, newSize)
|
||||||
|
if q.start == 0 {
|
||||||
|
copy(elements, q.elements)
|
||||||
|
} else {
|
||||||
|
copy(elements, q.elements[q.start:])
|
||||||
|
copy(elements[len(q.elements)-q.start:], q.elements[0:q.end])
|
||||||
|
}
|
||||||
|
q.start = 0
|
||||||
|
q.end = q.len
|
||||||
|
q.elements = elements
|
||||||
|
size = newSize
|
||||||
|
}
|
||||||
|
if q.end == size {
|
||||||
|
// Wrap around.
|
||||||
|
q.elements[0] = element
|
||||||
|
q.end = 1
|
||||||
|
q.len++
|
||||||
|
return
|
||||||
|
}
|
||||||
|
q.elements[q.end] = element
|
||||||
|
q.end++
|
||||||
|
q.len++
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *FIFO[T]) Pop() (element T, ok bool) {
|
||||||
|
if q.len == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
element = q.elements[q.start]
|
||||||
|
q.start++
|
||||||
|
if q.start == len(q.elements) {
|
||||||
|
// Wrap around.
|
||||||
|
q.start = 0
|
||||||
|
}
|
||||||
|
q.len--
|
||||||
|
|
||||||
|
// Once it is empty, shrink down to avoid hanging onto
|
||||||
|
// a large buffer forever.
|
||||||
|
if q.len == 0 && len(q.elements) > normalSize {
|
||||||
|
q.elements = make([]T, normalSize)
|
||||||
|
q.start = 0
|
||||||
|
q.end = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
ok = true
|
||||||
|
return
|
||||||
|
}
|
117
pkg/scheduler/util/queue/fifo_test.go
Normal file
117
pkg/scheduler/util/queue/fifo_test.go
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2024 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package queue
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func verifyPop(t *testing.T, expectedValue int, expectedOk bool, queue *FIFO[int]) {
|
||||||
|
t.Helper()
|
||||||
|
actual, ok := queue.Pop()
|
||||||
|
require.Equal(t, expectedOk, ok)
|
||||||
|
require.Equal(t, expectedValue, actual)
|
||||||
|
}
|
||||||
|
|
||||||
|
func verifyEmpty(t *testing.T, queue *FIFO[int]) {
|
||||||
|
t.Helper()
|
||||||
|
require.Equal(t, 0, queue.Len())
|
||||||
|
verifyPop(t, 0, false, queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNull(t *testing.T) {
|
||||||
|
var queue FIFO[int]
|
||||||
|
verifyEmpty(t, &queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestOnePushPop(t *testing.T) {
|
||||||
|
var queue FIFO[int]
|
||||||
|
|
||||||
|
expected := 10
|
||||||
|
queue.Push(10)
|
||||||
|
require.Equal(t, 1, queue.Len())
|
||||||
|
verifyPop(t, expected, true, &queue)
|
||||||
|
verifyEmpty(t, &queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pushes some elements, pops all of them, then the same again.
|
||||||
|
func TestWrapAroundEmpty(t *testing.T) {
|
||||||
|
var queue FIFO[int]
|
||||||
|
|
||||||
|
for i := 0; i < 5; i++ {
|
||||||
|
queue.Push(i)
|
||||||
|
}
|
||||||
|
require.Equal(t, 5, queue.Len())
|
||||||
|
for i := 0; i < 5; i++ {
|
||||||
|
verifyPop(t, i, true, &queue)
|
||||||
|
}
|
||||||
|
verifyEmpty(t, &queue)
|
||||||
|
|
||||||
|
for i := 5; i < 10; i++ {
|
||||||
|
queue.Push(i)
|
||||||
|
}
|
||||||
|
for i := 5; i < 10; i++ {
|
||||||
|
verifyPop(t, i, true, &queue)
|
||||||
|
}
|
||||||
|
verifyEmpty(t, &queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pushes some elements, pops one, adds more, then pops all.
|
||||||
|
func TestWrapAroundPartial(t *testing.T) {
|
||||||
|
var queue FIFO[int]
|
||||||
|
|
||||||
|
for i := 0; i < 5; i++ {
|
||||||
|
queue.Push(i)
|
||||||
|
}
|
||||||
|
require.Equal(t, 5, queue.Len())
|
||||||
|
verifyPop(t, 0, true, &queue)
|
||||||
|
|
||||||
|
for i := 5; i < 10; i++ {
|
||||||
|
queue.Push(i)
|
||||||
|
}
|
||||||
|
for i := 1; i < 10; i++ {
|
||||||
|
verifyPop(t, i, true, &queue)
|
||||||
|
}
|
||||||
|
verifyEmpty(t, &queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Push an unusual amount of elements, pop all, and verify that
|
||||||
|
// the FIFO shrinks back again.
|
||||||
|
func TestShrink(t *testing.T) {
|
||||||
|
var queue FIFO[int]
|
||||||
|
|
||||||
|
for i := 0; i < normalSize*2; i++ {
|
||||||
|
queue.Push(i)
|
||||||
|
}
|
||||||
|
require.Equal(t, normalSize*2, queue.Len())
|
||||||
|
require.LessOrEqual(t, 2*normalSize, len(queue.elements))
|
||||||
|
|
||||||
|
// Pop all, should be shrunken when done.
|
||||||
|
for i := 0; i < normalSize*2; i++ {
|
||||||
|
verifyPop(t, i, true, &queue)
|
||||||
|
}
|
||||||
|
require.Equal(t, 0, queue.Len())
|
||||||
|
require.Equal(t, normalSize, len(queue.elements))
|
||||||
|
|
||||||
|
// Still usable after shrinking?
|
||||||
|
queue.Push(42)
|
||||||
|
verifyPop(t, 42, true, &queue)
|
||||||
|
require.Equal(t, 0, queue.Len())
|
||||||
|
require.Equal(t, normalSize, len(queue.elements))
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user