internal/packing: enable non-square page size

Closes #2327
This commit is contained in:
Hajime Hoshi 2022-11-11 22:08:29 +09:00
parent fde964312c
commit c72fd7ffcf
2 changed files with 86 additions and 63 deletions

View File

@ -17,6 +17,7 @@ package packing
import ( import (
"errors" "errors"
"fmt"
) )
const ( const (
@ -25,15 +26,15 @@ const (
type Page struct { type Page struct {
root *Node root *Node
size int width int
height int
maxSize int maxSize int
rollbackExtension func()
} }
func NewPage(initSize int, maxSize int) *Page { func NewPage(initSize int, maxSize int) *Page {
return &Page{ return &Page{
size: initSize, width: initSize,
height: initSize,
maxSize: maxSize, maxSize: maxSize,
} }
} }
@ -86,7 +87,7 @@ func square(width, height int) float64 {
func (p *Page) canAlloc(n *Node, width, height int) bool { func (p *Page) canAlloc(n *Node, width, height int) bool {
if p.root == nil { if p.root == nil {
return p.size >= width && p.size >= height return p.width >= width && p.height >= height
} }
return canAlloc(p.root, width, height) return canAlloc(p.root, width, height)
} }
@ -170,7 +171,7 @@ func alloc(n *Node, width, height int) *Node {
} }
func (p *Page) Size() (int, int) { func (p *Page) Size() (int, int) {
return p.size, p.size return p.width, p.height
} }
func (p *Page) SetMaxSize(size int) { func (p *Page) SetMaxSize(size int) {
@ -191,8 +192,8 @@ func (p *Page) Alloc(width, height int) *Node {
if p.root == nil { if p.root == nil {
p.root = &Node{ p.root = &Node{
width: p.size, width: p.width,
height: p.size, height: p.height,
} }
} }
if width < minSize { if width < minSize {
@ -244,47 +245,50 @@ func (p *Page) extendFor(width, height int) bool {
return true return true
} }
nExtended := 1 if p.width >= p.maxSize && p.height >= p.maxSize {
for {
if !p.extend(nExtended) {
// The page can't be extended any more. Return as failure.
return false return false
} }
nExtended++
// (1, 0), (0, 1), (2, 0), (1, 1), (0, 2), (3, 0), (2, 1), (1, 2), (0, 3), ...
for i := 1; ; i++ {
for j := 0; j <= i; j++ {
newWidth := p.width
for k := 0; k < i-j; k++ {
newWidth *= 2
}
newHeight := p.height
for k := 0; k < j; k++ {
newHeight *= 2
}
if newWidth > p.maxSize || newHeight > p.maxSize {
if newWidth > p.maxSize && newHeight > p.maxSize {
panic(fmt.Sprintf("packing: too big extension: (%d, %d)", newWidth, newHeight))
}
continue
}
rollback := p.extend(newWidth, newHeight)
if p.canAlloc(p.root, width, height) { if p.canAlloc(p.root, width, height) {
p.rollbackExtension = nil
break
}
p.rollbackExtension()
p.rollbackExtension = nil
}
return true return true
}
rollback()
// If the allocation failed even with a maximized page, give up the allocation.
if newWidth >= p.maxSize && newHeight >= p.maxSize {
return false
}
}
}
} }
func (p *Page) extend(count int) bool { func (p *Page) extend(newWidth int, newHeight int) func() {
if p.rollbackExtension != nil {
panic("packing: Extend cannot be called without rolling back or committing")
}
if p.size >= p.maxSize {
return false
}
newSize := p.size
for i := 0; i < count; i++ {
newSize *= 2
}
if newSize > p.maxSize {
return false
}
edgeNodes := []*Node{} edgeNodes := []*Node{}
abort := errors.New("abort") abort := errors.New("abort")
aborted := false aborted := false
if p.root != nil { if p.root != nil {
_ = walk(p.root, func(n *Node) error { _ = walk(p.root, func(n *Node) error {
if n.x+n.width < p.size && n.y+n.height < p.size { if n.x+n.width < p.width && n.y+n.height < p.height {
return nil return nil
} }
if n.used { if n.used {
@ -296,21 +300,23 @@ func (p *Page) extend(count int) bool {
}) })
} }
var rollback func()
if aborted { if aborted {
origRoot := *p.root origRoot := *p.root
leftUpper := p.root leftUpper := p.root
leftLower := &Node{ leftLower := &Node{
x: 0, x: 0,
y: p.size, y: p.height,
width: p.size, width: p.width,
height: newSize - p.size, height: newHeight - p.height,
} }
left := &Node{ left := &Node{
x: 0, x: 0,
y: 0, y: 0,
width: p.size, width: p.width,
height: p.size, height: p.height,
child0: leftUpper, child0: leftUpper,
child1: leftLower, child1: leftLower,
} }
@ -318,45 +324,47 @@ func (p *Page) extend(count int) bool {
leftLower.parent = left leftLower.parent = left
right := &Node{ right := &Node{
x: p.size, x: p.width,
y: 0, y: 0,
width: newSize - p.size, width: newWidth - p.width,
height: newSize, height: newHeight,
} }
p.root = &Node{ p.root = &Node{
x: 0, x: 0,
y: 0, y: 0,
width: newSize, width: newWidth,
height: newSize, height: newHeight,
child0: left, child0: left,
child1: right, child1: right,
} }
left.parent = p.root left.parent = p.root
right.parent = p.root right.parent = p.root
origSize := p.size origWidth, origHeight := p.width, p.height
p.rollbackExtension = func() { rollback = func() {
p.size = origSize p.width = origWidth
p.height = origHeight
p.root = &origRoot p.root = &origRoot
} }
} else { } else {
origSize := p.size origWidth, origHeight := p.width, p.height
origWidths := map[*Node]int{} origWidths := map[*Node]int{}
origHeights := map[*Node]int{} origHeights := map[*Node]int{}
for _, n := range edgeNodes { for _, n := range edgeNodes {
if n.x+n.width == p.size { if n.x+n.width == p.width {
origWidths[n] = n.width origWidths[n] = n.width
n.width += newSize - p.size n.width += newWidth - p.width
} }
if n.y+n.height == p.size { if n.y+n.height == p.height {
origHeights[n] = n.height origHeights[n] = n.height
n.height += newSize - p.size n.height += newHeight - p.height
} }
} }
p.rollbackExtension = func() { rollback = func() {
p.size = origSize p.width = origWidth
p.height = origHeight
for n, w := range origWidths { for n, w := range origWidths {
n.width = w n.width = w
} }
@ -366,7 +374,8 @@ func (p *Page) extend(count int) bool {
} }
} }
p.size = newSize p.width = newWidth
p.height = newHeight
return true return rollback
} }

View File

@ -313,3 +313,17 @@ func TestAllocTooMuch(t *testing.T) {
t.Errorf("got: non-nil, want: nil") t.Errorf("got: non-nil, want: nil")
} }
} }
func TestNonSquareAlloc(t *testing.T) {
p := packing.NewPage(1024, 16384)
n0 := p.Alloc(16384, 1)
if _, h := p.Size(); h != 1024 {
t.Errorf("got: %d, want: 1024", h)
}
n1 := p.Alloc(16384, 1)
if _, h := p.Size(); h != 1024 {
t.Errorf("got: %d, want: 1024", h)
}
p.Free(n0)
p.Free(n1)
}