graphicsdriver/metal, graphicsdriver/opengl: Reland: Remove the thread usages for performance

Instead, graphicscommand package has a thread.

Updates #1367
This commit is contained in:
Hajime Hoshi 2020-10-13 00:39:45 +09:00
parent 713eee1117
commit eed619ad0f
12 changed files with 548 additions and 739 deletions

View File

@ -20,7 +20,6 @@ import (
"github.com/hajimehoshi/ebiten/v2/internal/affine" "github.com/hajimehoshi/ebiten/v2/internal/affine"
"github.com/hajimehoshi/ebiten/v2/internal/graphics" "github.com/hajimehoshi/ebiten/v2/internal/graphics"
"github.com/hajimehoshi/ebiten/v2/internal/shaderir" "github.com/hajimehoshi/ebiten/v2/internal/shaderir"
"github.com/hajimehoshi/ebiten/v2/internal/thread"
) )
type Region struct { type Region struct {
@ -31,7 +30,6 @@ type Region struct {
} }
type Graphics interface { type Graphics interface {
SetThread(thread *thread.Thread)
Begin() Begin()
End() End()
SetTransparent(transparent bool) SetTransparent(transparent bool)

View File

@ -200,10 +200,19 @@ func (q *commandQueue) Enqueue(command command) {
// Flush flushes the command queue. // Flush flushes the command queue.
func (q *commandQueue) Flush() error { func (q *commandQueue) Flush() error {
return RunOnMainThread(func() error {
return q.flush()
})
}
// flush must be called the main thread.
func (q *commandQueue) flush() error {
if len(q.commands) == 0 { if len(q.commands) == 0 {
return nil return nil
} }
// TODO: Use thread.Call here!
es := q.indices es := q.indices
vs := q.vertices vs := q.vertices
if recordLog() { if recordLog() {
@ -716,10 +725,17 @@ func (c *newShaderCommand) CanMergeWithDrawTrianglesCommand(dst *Image, src [gra
// ResetGraphicsDriverState resets or initializes the current graphics driver state. // ResetGraphicsDriverState resets or initializes the current graphics driver state.
func ResetGraphicsDriverState() error { func ResetGraphicsDriverState() error {
return theGraphicsDriver.Reset() return RunOnMainThread(func() error {
return theGraphicsDriver.Reset()
})
} }
// MaxImageSize returns the maximum size of an image. // MaxImageSize returns the maximum size of an image.
func MaxImageSize() int { func MaxImageSize() int {
return theGraphicsDriver.MaxImageSize() var size int
_ = RunOnMainThread(func() error {
size = theGraphicsDriver.MaxImageSize()
return nil
})
return size
} }

View File

@ -0,0 +1,35 @@
// Copyright 2020 The Ebiten 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 graphicscommand
import (
"github.com/hajimehoshi/ebiten/v2/internal/thread"
)
var theThread *thread.Thread
func SetMainThread(thread *thread.Thread) {
theThread = thread
}
func RunOnMainThread(f func() error) error {
// The thread is nil when 1) GOOS=js or 2) using golang.org/x/mobile/gl.
// When golang.org/x/mobile/gl is used, all the GL functions are called via Context, which already runs on an
// appropriate thread.
if theThread == nil {
return f()
}
return theThread.Call(f)
}

View File

@ -27,7 +27,6 @@ import (
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/metal/ca" "github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/metal/ca"
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/metal/mtl" "github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/metal/mtl"
"github.com/hajimehoshi/ebiten/v2/internal/shaderir" "github.com/hajimehoshi/ebiten/v2/internal/shaderir"
"github.com/hajimehoshi/ebiten/v2/internal/thread"
) )
// #cgo CFLAGS: -x objective-c // #cgo CFLAGS: -x objective-c
@ -328,8 +327,6 @@ type Graphics struct {
maxImageSize int maxImageSize int
tmpTexture mtl.Texture tmpTexture mtl.Texture
t *thread.Thread
pool unsafe.Pointer pool unsafe.Pointer
} }
@ -339,36 +336,23 @@ func Get() *Graphics {
return &theGraphics return &theGraphics
} }
func (g *Graphics) SetThread(thread *thread.Thread) {
g.t = thread
}
func (g *Graphics) Begin() { func (g *Graphics) Begin() {
g.t.Call(func() error { // NSAutoreleasePool is required to release drawable correctly (#847).
// NSAutoreleasePool is required to release drawable correctly (#847). // https://developer.apple.com/library/archive/documentation/3DDrawing/Conceptual/MTLBestPracticesGuide/Drawables.html
// https://developer.apple.com/library/archive/documentation/3DDrawing/Conceptual/MTLBestPracticesGuide/Drawables.html g.pool = C.allocAutoreleasePool()
g.pool = C.allocAutoreleasePool()
return nil
})
} }
func (g *Graphics) End() { func (g *Graphics) End() {
g.flushIfNeeded(false, true) g.flushIfNeeded(false, true)
g.t.Call(func() error { g.screenDrawable = ca.MetalDrawable{}
g.screenDrawable = ca.MetalDrawable{} C.releaseAutoreleasePool(g.pool)
C.releaseAutoreleasePool(g.pool) g.pool = nil
g.pool = nil
return nil
})
} }
func (g *Graphics) SetWindow(window uintptr) { func (g *Graphics) SetWindow(window uintptr) {
g.t.Call(func() error { // Note that [NSApp mainWindow] returns nil when the window is borderless.
// Note that [NSApp mainWindow] returns nil when the window is borderless. // Then the window is needed to be given explicitly.
// Then the window is needed to be given explicitly. g.view.setWindow(window)
g.view.setWindow(window)
return nil
})
} }
func (g *Graphics) SetUIView(uiview uintptr) { func (g *Graphics) SetUIView(uiview uintptr) {
@ -377,37 +361,30 @@ func (g *Graphics) SetUIView(uiview uintptr) {
} }
func (g *Graphics) SetVertices(vertices []float32, indices []uint16) { func (g *Graphics) SetVertices(vertices []float32, indices []uint16) {
g.t.Call(func() error { if g.vb != (mtl.Buffer{}) {
if g.vb != (mtl.Buffer{}) { g.vb.Release()
g.vb.Release() }
} if g.ib != (mtl.Buffer{}) {
if g.ib != (mtl.Buffer{}) { g.ib.Release()
g.ib.Release() }
} g.vb = g.view.getMTLDevice().MakeBufferWithBytes(unsafe.Pointer(&vertices[0]), unsafe.Sizeof(vertices[0])*uintptr(len(vertices)), resourceStorageMode)
g.vb = g.view.getMTLDevice().MakeBufferWithBytes(unsafe.Pointer(&vertices[0]), unsafe.Sizeof(vertices[0])*uintptr(len(vertices)), resourceStorageMode) g.ib = g.view.getMTLDevice().MakeBufferWithBytes(unsafe.Pointer(&indices[0]), unsafe.Sizeof(indices[0])*uintptr(len(indices)), resourceStorageMode)
g.ib = g.view.getMTLDevice().MakeBufferWithBytes(unsafe.Pointer(&indices[0]), unsafe.Sizeof(indices[0])*uintptr(len(indices)), resourceStorageMode)
return nil
})
} }
func (g *Graphics) flushIfNeeded(wait bool, present bool) { func (g *Graphics) flushIfNeeded(wait bool, present bool) {
g.t.Call(func() error { if g.cb == (mtl.CommandBuffer{}) {
if g.cb == (mtl.CommandBuffer{}) { return
return nil }
}
if present && g.screenDrawable != (ca.MetalDrawable{}) { if present && g.screenDrawable != (ca.MetalDrawable{}) {
g.cb.PresentDrawable(g.screenDrawable) g.cb.PresentDrawable(g.screenDrawable)
} }
g.cb.Commit() g.cb.Commit()
if wait { if wait {
g.cb.WaitUntilCompleted() g.cb.WaitUntilCompleted()
} }
g.cb = mtl.CommandBuffer{} g.cb = mtl.CommandBuffer{}
return nil
})
} }
func (g *Graphics) checkSize(width, height int) { func (g *Graphics) checkSize(width, height int) {
@ -452,11 +429,7 @@ func (g *Graphics) NewImage(width, height int) (driver.Image, error) {
StorageMode: storageMode, StorageMode: storageMode,
Usage: mtl.TextureUsageShaderRead | mtl.TextureUsageRenderTarget, Usage: mtl.TextureUsageShaderRead | mtl.TextureUsageRenderTarget,
} }
var t mtl.Texture t := g.view.getMTLDevice().MakeTexture(td)
g.t.Call(func() error {
t = g.view.getMTLDevice().MakeTexture(td)
return nil
})
i := &Image{ i := &Image{
id: g.genNextImageID(), id: g.genNextImageID(),
graphics: g, graphics: g,
@ -469,10 +442,7 @@ func (g *Graphics) NewImage(width, height int) (driver.Image, error) {
} }
func (g *Graphics) NewScreenFramebufferImage(width, height int) (driver.Image, error) { func (g *Graphics) NewScreenFramebufferImage(width, height int) (driver.Image, error) {
g.t.Call(func() error { g.view.setDrawableSize(width, height)
g.view.setDrawableSize(width, height)
return nil
})
i := &Image{ i := &Image{
id: g.genNextImageID(), id: g.genNextImageID(),
graphics: g, graphics: g,
@ -524,126 +494,120 @@ func operationToBlendFactor(c driver.Operation) mtl.BlendFactor {
} }
func (g *Graphics) Reset() error { func (g *Graphics) Reset() error {
if err := g.t.Call(func() error { if g.cq != (mtl.CommandQueue{}) {
if g.cq != (mtl.CommandQueue{}) { g.cq.Release()
g.cq.Release() g.cq = mtl.CommandQueue{}
g.cq = mtl.CommandQueue{} }
}
// TODO: Release existing rpss // TODO: Release existing rpss
if g.rpss == nil { if g.rpss == nil {
g.rpss = map[rpsKey]mtl.RenderPipelineState{} g.rpss = map[rpsKey]mtl.RenderPipelineState{}
} }
if err := g.view.reset(); err != nil { if err := g.view.reset(); err != nil {
return err return err
} }
if g.transparent { if g.transparent {
g.view.ml.SetOpaque(false) g.view.ml.SetOpaque(false)
} }
replaces := map[string]string{ replaces := map[string]string{
"{{.FilterNearest}}": fmt.Sprintf("%d", driver.FilterNearest), "{{.FilterNearest}}": fmt.Sprintf("%d", driver.FilterNearest),
"{{.FilterLinear}}": fmt.Sprintf("%d", driver.FilterLinear), "{{.FilterLinear}}": fmt.Sprintf("%d", driver.FilterLinear),
"{{.FilterScreen}}": fmt.Sprintf("%d", driver.FilterScreen), "{{.FilterScreen}}": fmt.Sprintf("%d", driver.FilterScreen),
"{{.AddressClampToZero}}": fmt.Sprintf("%d", driver.AddressClampToZero), "{{.AddressClampToZero}}": fmt.Sprintf("%d", driver.AddressClampToZero),
"{{.AddressRepeat}}": fmt.Sprintf("%d", driver.AddressRepeat), "{{.AddressRepeat}}": fmt.Sprintf("%d", driver.AddressRepeat),
"{{.AddressUnsafe}}": fmt.Sprintf("%d", driver.AddressUnsafe), "{{.AddressUnsafe}}": fmt.Sprintf("%d", driver.AddressUnsafe),
} }
src := source src := source
for k, v := range replaces { for k, v := range replaces {
src = strings.Replace(src, k, v, -1) src = strings.Replace(src, k, v, -1)
} }
lib, err := g.view.getMTLDevice().MakeLibrary(src, mtl.CompileOptions{}) lib, err := g.view.getMTLDevice().MakeLibrary(src, mtl.CompileOptions{})
if err != nil { if err != nil {
return err return err
} }
vs, err := lib.MakeFunction("VertexShader") vs, err := lib.MakeFunction("VertexShader")
if err != nil { if err != nil {
return err return err
} }
fs, err := lib.MakeFunction( fs, err := lib.MakeFunction(
fmt.Sprintf("FragmentShader_%d_%d_%d", 0, driver.FilterScreen, driver.AddressUnsafe)) fmt.Sprintf("FragmentShader_%d_%d_%d", 0, driver.FilterScreen, driver.AddressUnsafe))
if err != nil { if err != nil {
return err return err
} }
rpld := mtl.RenderPipelineDescriptor{ rpld := mtl.RenderPipelineDescriptor{
VertexFunction: vs, VertexFunction: vs,
FragmentFunction: fs, FragmentFunction: fs,
} }
rpld.ColorAttachments[0].PixelFormat = g.view.colorPixelFormat() rpld.ColorAttachments[0].PixelFormat = g.view.colorPixelFormat()
rpld.ColorAttachments[0].BlendingEnabled = true rpld.ColorAttachments[0].BlendingEnabled = true
rpld.ColorAttachments[0].DestinationAlphaBlendFactor = mtl.BlendFactorZero rpld.ColorAttachments[0].DestinationAlphaBlendFactor = mtl.BlendFactorZero
rpld.ColorAttachments[0].DestinationRGBBlendFactor = mtl.BlendFactorZero rpld.ColorAttachments[0].DestinationRGBBlendFactor = mtl.BlendFactorZero
rpld.ColorAttachments[0].SourceAlphaBlendFactor = mtl.BlendFactorOne rpld.ColorAttachments[0].SourceAlphaBlendFactor = mtl.BlendFactorOne
rpld.ColorAttachments[0].SourceRGBBlendFactor = mtl.BlendFactorOne rpld.ColorAttachments[0].SourceRGBBlendFactor = mtl.BlendFactorOne
rps, err := g.view.getMTLDevice().MakeRenderPipelineState(rpld) rps, err := g.view.getMTLDevice().MakeRenderPipelineState(rpld)
if err != nil { if err != nil {
return err return err
} }
g.screenRPS = rps g.screenRPS = rps
for _, screen := range []bool{false, true} { for _, screen := range []bool{false, true} {
for _, cm := range []bool{false, true} { for _, cm := range []bool{false, true} {
for _, a := range []driver.Address{ for _, a := range []driver.Address{
driver.AddressClampToZero, driver.AddressClampToZero,
driver.AddressRepeat, driver.AddressRepeat,
driver.AddressUnsafe, driver.AddressUnsafe,
} {
for _, f := range []driver.Filter{
driver.FilterNearest,
driver.FilterLinear,
} { } {
for _, f := range []driver.Filter{ for c := driver.CompositeModeSourceOver; c <= driver.CompositeModeMax; c++ {
driver.FilterNearest, cmi := 0
driver.FilterLinear, if cm {
} { cmi = 1
for c := driver.CompositeModeSourceOver; c <= driver.CompositeModeMax; c++ {
cmi := 0
if cm {
cmi = 1
}
fs, err := lib.MakeFunction(fmt.Sprintf("FragmentShader_%d_%d_%d", cmi, f, a))
if err != nil {
return err
}
rpld := mtl.RenderPipelineDescriptor{
VertexFunction: vs,
FragmentFunction: fs,
}
pix := mtl.PixelFormatRGBA8UNorm
if screen {
pix = g.view.colorPixelFormat()
}
rpld.ColorAttachments[0].PixelFormat = pix
rpld.ColorAttachments[0].BlendingEnabled = true
src, dst := c.Operations()
rpld.ColorAttachments[0].DestinationAlphaBlendFactor = operationToBlendFactor(dst)
rpld.ColorAttachments[0].DestinationRGBBlendFactor = operationToBlendFactor(dst)
rpld.ColorAttachments[0].SourceAlphaBlendFactor = operationToBlendFactor(src)
rpld.ColorAttachments[0].SourceRGBBlendFactor = operationToBlendFactor(src)
rps, err := g.view.getMTLDevice().MakeRenderPipelineState(rpld)
if err != nil {
return err
}
g.rpss[rpsKey{
screen: screen,
useColorM: cm,
filter: f,
address: a,
compositeMode: c,
}] = rps
} }
fs, err := lib.MakeFunction(fmt.Sprintf("FragmentShader_%d_%d_%d", cmi, f, a))
if err != nil {
return err
}
rpld := mtl.RenderPipelineDescriptor{
VertexFunction: vs,
FragmentFunction: fs,
}
pix := mtl.PixelFormatRGBA8UNorm
if screen {
pix = g.view.colorPixelFormat()
}
rpld.ColorAttachments[0].PixelFormat = pix
rpld.ColorAttachments[0].BlendingEnabled = true
src, dst := c.Operations()
rpld.ColorAttachments[0].DestinationAlphaBlendFactor = operationToBlendFactor(dst)
rpld.ColorAttachments[0].DestinationRGBBlendFactor = operationToBlendFactor(dst)
rpld.ColorAttachments[0].SourceAlphaBlendFactor = operationToBlendFactor(src)
rpld.ColorAttachments[0].SourceRGBBlendFactor = operationToBlendFactor(src)
rps, err := g.view.getMTLDevice().MakeRenderPipelineState(rpld)
if err != nil {
return err
}
g.rpss[rpsKey{
screen: screen,
useColorM: cm,
filter: f,
address: a,
compositeMode: c,
}] = rps
} }
} }
} }
} }
g.cq = g.view.getMTLDevice().MakeCommandQueue()
return nil
}); err != nil {
return err
} }
g.cq = g.view.getMTLDevice().MakeCommandQueue()
return nil return nil
} }
@ -733,37 +697,32 @@ func (g *Graphics) Draw(dstID, srcID driver.ImageID, indexLen int, indexOffset i
}] }]
} }
if err := g.t.Call(func() error { w, h := dst.internalSize()
w, h := dst.internalSize() sourceSize := []float32{0, 0}
sourceSize := []float32{0, 0} if filter != driver.FilterNearest {
if filter != driver.FilterNearest { w, h := srcs[0].internalSize()
w, h := srcs[0].internalSize() sourceSize[0] = float32(w)
sourceSize[0] = float32(w) sourceSize[1] = float32(h)
sourceSize[1] = float32(h) }
} esBody, esTranslate := colorM.UnsafeElements()
esBody, esTranslate := colorM.UnsafeElements() scale := float32(0)
scale := float32(0) if filter == driver.FilterScreen {
if filter == driver.FilterScreen { scale = float32(dst.width) / float32(srcs[0].width)
scale = float32(dst.width) / float32(srcs[0].width) }
} uniforms := []interface{}{
uniforms := []interface{}{ []float32{float32(w), float32(h)},
[]float32{float32(w), float32(h)}, sourceSize,
sourceSize, esBody,
esBody, esTranslate,
esTranslate, scale,
scale, []float32{
[]float32{ sourceRegion.X,
sourceRegion.X, sourceRegion.Y,
sourceRegion.Y, sourceRegion.X + sourceRegion.Width,
sourceRegion.X + sourceRegion.Width, sourceRegion.Y + sourceRegion.Height,
sourceRegion.Y + sourceRegion.Height, },
}, }
} if err := g.draw(rps, dst, srcs, indexLen, indexOffset, uniforms); err != nil {
if err := g.draw(rps, dst, srcs, indexLen, indexOffset, uniforms); err != nil {
return err
}
return nil
}); err != nil {
return err return err
} }
return nil return nil
@ -790,52 +749,40 @@ func (g *Graphics) HasHighPrecisionFloat() bool {
} }
func (g *Graphics) MaxImageSize() int { func (g *Graphics) MaxImageSize() int {
m := 0 if g.maxImageSize == 0 {
g.t.Call(func() error { g.maxImageSize = 4096
if g.maxImageSize == 0 { // https://developer.apple.com/metal/Metal-Feature-Set-Tables.pdf
switch {
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily5_v1):
g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily4_v1):
g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily3_v1):
g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily2_v2):
g.maxImageSize = 8192
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily2_v1):
g.maxImageSize = 4096 g.maxImageSize = 4096
// https://developer.apple.com/metal/Metal-Feature-Set-Tables.pdf case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily1_v2):
switch { g.maxImageSize = 8192
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily5_v1): case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily1_v1):
g.maxImageSize = 16384 g.maxImageSize = 4096
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily4_v1): case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_tvOS_GPUFamily2_v1):
g.maxImageSize = 16384 g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily3_v1): case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_tvOS_GPUFamily1_v1):
g.maxImageSize = 16384 g.maxImageSize = 8192
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily2_v2): case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_macOS_GPUFamily1_v1):
g.maxImageSize = 8192 g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily2_v1): default:
g.maxImageSize = 4096 panic("metal: there is no supported feature set")
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily1_v2):
g.maxImageSize = 8192
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_iOS_GPUFamily1_v1):
g.maxImageSize = 4096
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_tvOS_GPUFamily2_v1):
g.maxImageSize = 16384
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_tvOS_GPUFamily1_v1):
g.maxImageSize = 8192
case g.view.getMTLDevice().SupportsFeatureSet(mtl.FeatureSet_macOS_GPUFamily1_v1):
g.maxImageSize = 16384
default:
panic("metal: there is no supported feature set")
}
} }
m = g.maxImageSize }
return nil return g.maxImageSize
})
return m
} }
func (g *Graphics) NewShader(program *shaderir.Program) (driver.Shader, error) { func (g *Graphics) NewShader(program *shaderir.Program) (driver.Shader, error) {
var s *Shader s, err := newShader(g.view.getMTLDevice(), g.genNextShaderID(), program)
if err := g.t.Call(func() error { if err != nil {
var err error
s, err = newShader(g.view.getMTLDevice(), g.genNextShaderID(), program)
if err != nil {
return err
}
return nil
}); err != nil {
return nil, err return nil, err
} }
g.addShader(s) g.addShader(s)
@ -877,13 +824,10 @@ func (i *Image) internalSize() (int, int) {
} }
func (i *Image) Dispose() { func (i *Image) Dispose() {
i.graphics.t.Call(func() error { if i.texture != (mtl.Texture{}) {
if i.texture != (mtl.Texture{}) { i.texture.Release()
i.texture.Release() i.texture = mtl.Texture{}
i.texture = mtl.Texture{} }
}
return nil
})
i.graphics.removeImage(i) i.graphics.removeImage(i)
} }
@ -897,19 +841,16 @@ func (i *Image) IsInvalidated() bool {
func (i *Image) syncTexture() { func (i *Image) syncTexture() {
// Calling SynchronizeTexture is ignored on iOS (see mtl.m), but it looks like committing BliCommandEncoder // Calling SynchronizeTexture is ignored on iOS (see mtl.m), but it looks like committing BliCommandEncoder
// is necessary (#1337). // is necessary (#1337).
i.graphics.t.Call(func() error { if i.graphics.cb != (mtl.CommandBuffer{}) {
if i.graphics.cb != (mtl.CommandBuffer{}) { panic("metal: command buffer must be empty at syncTexture: flushIfNeeded is not called yet?")
panic("metal: command buffer must be empty at syncTexture: flushIfNeeded is not called yet?") }
}
cb := i.graphics.cq.MakeCommandBuffer() cb := i.graphics.cq.MakeCommandBuffer()
bce := cb.MakeBlitCommandEncoder() bce := cb.MakeBlitCommandEncoder()
bce.SynchronizeTexture(i.texture, 0, 0) bce.SynchronizeTexture(i.texture, 0, 0)
bce.EndEncoding() bce.EndEncoding()
cb.Commit() cb.Commit()
cb.WaitUntilCompleted() cb.WaitUntilCompleted()
return nil
})
} }
func (i *Image) Pixels() ([]byte, error) { func (i *Image) Pixels() ([]byte, error) {
@ -917,12 +858,9 @@ func (i *Image) Pixels() ([]byte, error) {
i.syncTexture() i.syncTexture()
b := make([]byte, 4*i.width*i.height) b := make([]byte, 4*i.width*i.height)
i.graphics.t.Call(func() error { i.texture.GetBytes(&b[0], uintptr(4*i.width), mtl.Region{
i.texture.GetBytes(&b[0], uintptr(4*i.width), mtl.Region{ Size: mtl.Size{Width: i.width, Height: i.height, Depth: 1},
Size: mtl.Size{Width: i.width, Height: i.height, Depth: 1}, }, 0)
}, 0)
return nil
})
return b, nil return b, nil
} }
@ -932,58 +870,50 @@ func (i *Image) ReplacePixels(args []*driver.ReplacePixelsArgs) {
// If the memory is shared (e.g., iOS), texture data doen't have to be synced. Send the data directly. // If the memory is shared (e.g., iOS), texture data doen't have to be synced. Send the data directly.
if storageMode == mtl.StorageModeShared { if storageMode == mtl.StorageModeShared {
g.t.Call(func() error { for _, a := range args {
for _, a := range args { i.texture.ReplaceRegion(mtl.Region{
i.texture.ReplaceRegion(mtl.Region{ Origin: mtl.Origin{X: a.X, Y: a.Y, Z: 0},
Origin: mtl.Origin{X: a.X, Y: a.Y, Z: 0}, Size: mtl.Size{Width: a.Width, Height: a.Height, Depth: 1},
Size: mtl.Size{Width: a.Width, Height: a.Height, Depth: 1}, }, 0, unsafe.Pointer(&a.Pixels[0]), 4*a.Width)
}, 0, unsafe.Pointer(&a.Pixels[0]), 4*a.Width) }
}
return nil
})
return
} }
// If the memory is managed (e.g., macOS), texture data cannot be sent to the destination directly because // If the memory is managed (e.g., macOS), texture data cannot be sent to the destination directly because
// this requires synchronizing data between CPU and GPU. As synchronizing is inefficient, let's send the // this requires synchronizing data between CPU and GPU. As synchronizing is inefficient, let's send the
// data to a temporary texture once, and then copy it in GPU. // data to a temporary texture once, and then copy it in GPU.
g.t.Call(func() error { w, h := i.texture.Width(), i.texture.Height()
w, h := i.texture.Width(), i.texture.Height() if g.tmpTexture == (mtl.Texture{}) || w > g.tmpTexture.Width() || h > g.tmpTexture.Height() {
if g.tmpTexture == (mtl.Texture{}) || w > g.tmpTexture.Width() || h > g.tmpTexture.Height() { if g.tmpTexture != (mtl.Texture{}) {
if g.tmpTexture != (mtl.Texture{}) { g.tmpTexture.Release()
g.tmpTexture.Release()
}
td := mtl.TextureDescriptor{
TextureType: mtl.TextureType2D,
PixelFormat: mtl.PixelFormatRGBA8UNorm,
Width: w,
Height: h,
StorageMode: storageMode,
Usage: mtl.TextureUsageShaderRead | mtl.TextureUsageRenderTarget,
}
g.tmpTexture = g.view.getMTLDevice().MakeTexture(td)
} }
td := mtl.TextureDescriptor{
TextureType: mtl.TextureType2D,
PixelFormat: mtl.PixelFormatRGBA8UNorm,
Width: w,
Height: h,
StorageMode: storageMode,
Usage: mtl.TextureUsageShaderRead | mtl.TextureUsageRenderTarget,
}
g.tmpTexture = g.view.getMTLDevice().MakeTexture(td)
}
for _, a := range args { for _, a := range args {
g.tmpTexture.ReplaceRegion(mtl.Region{ g.tmpTexture.ReplaceRegion(mtl.Region{
Origin: mtl.Origin{X: a.X, Y: a.Y, Z: 0}, Origin: mtl.Origin{X: a.X, Y: a.Y, Z: 0},
Size: mtl.Size{Width: a.Width, Height: a.Height, Depth: 1}, Size: mtl.Size{Width: a.Width, Height: a.Height, Depth: 1},
}, 0, unsafe.Pointer(&a.Pixels[0]), 4*a.Width) }, 0, unsafe.Pointer(&a.Pixels[0]), 4*a.Width)
} }
if g.cb == (mtl.CommandBuffer{}) { if g.cb == (mtl.CommandBuffer{}) {
g.cb = i.graphics.cq.MakeCommandBuffer() g.cb = i.graphics.cq.MakeCommandBuffer()
} }
bce := g.cb.MakeBlitCommandEncoder() bce := g.cb.MakeBlitCommandEncoder()
for _, a := range args { for _, a := range args {
o := mtl.Origin{X: a.X, Y: a.Y, Z: 0} o := mtl.Origin{X: a.X, Y: a.Y, Z: 0}
s := mtl.Size{Width: a.Width, Height: a.Height, Depth: 1} s := mtl.Size{Width: a.Width, Height: a.Height, Depth: 1}
bce.CopyFromTexture(g.tmpTexture, 0, 0, o, s, i.texture, 0, 0, o) bce.CopyFromTexture(g.tmpTexture, 0, 0, o, s, i.texture, 0, 0, o)
} }
bce.EndEncoding() bce.EndEncoding()
return nil
})
} }
func (g *Graphics) DrawShader(dstID driver.ImageID, srcIDs [graphics.ShaderImageNum]driver.ImageID, offsets [graphics.ShaderImageNum - 1][2]float32, shader driver.ShaderID, indexLen int, indexOffset int, sourceRegion driver.Region, mode driver.CompositeMode, uniforms []interface{}) error { func (g *Graphics) DrawShader(dstID driver.ImageID, srcIDs [graphics.ShaderImageNum]driver.ImageID, offsets [graphics.ShaderImageNum - 1][2]float32, shader driver.ShaderID, indexLen int, indexOffset int, sourceRegion driver.Region, mode driver.CompositeMode, uniforms []interface{}) error {
@ -993,56 +923,51 @@ func (g *Graphics) DrawShader(dstID driver.ImageID, srcIDs [graphics.ShaderImage
srcs[i] = g.images[srcID] srcs[i] = g.images[srcID]
} }
if err := g.t.Call(func() error { rps, err := g.shaders[shader].RenderPipelineState(g.view.getMTLDevice(), mode)
rps, err := g.shaders[shader].RenderPipelineState(g.view.getMTLDevice(), mode) if err != nil {
if err != nil { return err
return err }
us := make([]interface{}, graphics.PreservedUniformVariablesNum+len(uniforms))
// Set the destination texture size.
dw, dh := dst.internalSize()
us[graphics.DestinationTextureSizeUniformVariableIndex] = []float32{float32(dw), float32(dh)}
// Set the source texture sizes.
usizes := make([]float32, 2*len(srcs))
for i, src := range srcs {
if src != nil {
w, h := src.internalSize()
usizes[2*i] = float32(w)
usizes[2*i+1] = float32(h)
} }
}
us[graphics.TextureSizesUniformVariableIndex] = usizes
us := make([]interface{}, graphics.PreservedUniformVariablesNum+len(uniforms)) // Set the source offsets.
uoffsets := make([]float32, 2*len(offsets))
for i, offset := range offsets {
uoffsets[2*i] = offset[0]
uoffsets[2*i+1] = offset[1]
}
us[graphics.TextureSourceOffsetsUniformVariableIndex] = uoffsets
// Set the destination texture size. // Set the source region's origin of texture0.
dw, dh := dst.internalSize() uorigin := []float32{float32(sourceRegion.X), float32(sourceRegion.Y)}
us[graphics.DestinationTextureSizeUniformVariableIndex] = []float32{float32(dw), float32(dh)} us[graphics.TextureSourceRegionOriginUniformVariableIndex] = uorigin
// Set the source texture sizes. // Set the source region's size of texture0.
usizes := make([]float32, 2*len(srcs)) ussize := []float32{float32(sourceRegion.Width), float32(sourceRegion.Height)}
for i, src := range srcs { us[graphics.TextureSourceRegionSizeUniformVariableIndex] = ussize
if src != nil {
w, h := src.internalSize()
usizes[2*i] = float32(w)
usizes[2*i+1] = float32(h)
}
}
us[graphics.TextureSizesUniformVariableIndex] = usizes
// Set the source offsets. // Set the additional uniform variables.
uoffsets := make([]float32, 2*len(offsets)) for i, v := range uniforms {
for i, offset := range offsets { const offset = graphics.PreservedUniformVariablesNum
uoffsets[2*i] = offset[0] us[offset+i] = v
uoffsets[2*i+1] = offset[1] }
}
us[graphics.TextureSourceOffsetsUniformVariableIndex] = uoffsets
// Set the source region's origin of texture0. if err := g.draw(rps, dst, srcs, indexLen, indexOffset, us); err != nil {
uorigin := []float32{float32(sourceRegion.X), float32(sourceRegion.Y)}
us[graphics.TextureSourceRegionOriginUniformVariableIndex] = uorigin
// Set the source region's size of texture0.
ussize := []float32{float32(sourceRegion.Width), float32(sourceRegion.Height)}
us[graphics.TextureSourceRegionSizeUniformVariableIndex] = ussize
// Set the additional uniform variables.
for i, v := range uniforms {
const offset = graphics.PreservedUniformVariablesNum
us[offset+i] = v
}
if err := g.draw(rps, dst, srcs, indexLen, indexOffset, us); err != nil {
return err
}
return nil
}); err != nil {
return err return err
} }
return nil return nil

View File

@ -19,7 +19,6 @@ import (
"sync" "sync"
"github.com/hajimehoshi/ebiten/v2/internal/driver" "github.com/hajimehoshi/ebiten/v2/internal/driver"
"github.com/hajimehoshi/ebiten/v2/internal/thread"
) )
func convertOperation(op driver.Operation) operation { func convertOperation(op driver.Operation) operation {
@ -56,8 +55,6 @@ type context struct {
highp bool highp bool
highpOnce sync.Once highpOnce sync.Once
t *thread.Thread
contextImpl contextImpl
} }

View File

@ -101,131 +101,90 @@ type contextImpl struct {
} }
func (c *context) reset() error { func (c *context) reset() error {
if err := c.t.Call(func() error { if !c.init {
if c.init {
return nil
}
// Note that this initialization must be done after Loop is called. // Note that this initialization must be done after Loop is called.
if err := gl.Init(); err != nil { if err := gl.Init(); err != nil {
return fmt.Errorf("opengl: initializing error %v", err) return fmt.Errorf("opengl: initializing error %v", err)
} }
c.init = true c.init = true
return nil
}); err != nil {
return err
} }
c.locationCache = newLocationCache() c.locationCache = newLocationCache()
c.lastTexture = invalidTexture c.lastTexture = invalidTexture
c.lastFramebuffer = invalidFramebuffer c.lastFramebuffer = invalidFramebuffer
c.lastViewportWidth = 0 c.lastViewportWidth = 0
c.lastViewportHeight = 0 c.lastViewportHeight = 0
c.lastCompositeMode = driver.CompositeModeUnknown c.lastCompositeMode = driver.CompositeModeUnknown
_ = c.t.Call(func() error { gl.Enable(gl.BLEND)
gl.Enable(gl.BLEND)
return nil
})
c.blendFunc(driver.CompositeModeSourceOver) c.blendFunc(driver.CompositeModeSourceOver)
_ = c.t.Call(func() error {
f := int32(0) f := int32(0)
gl.GetIntegerv(gl.FRAMEBUFFER_BINDING, &f) gl.GetIntegerv(gl.FRAMEBUFFER_BINDING, &f)
c.screenFramebuffer = framebufferNative(f) c.screenFramebuffer = framebufferNative(f)
return nil
})
return nil return nil
} }
func (c *context) blendFunc(mode driver.CompositeMode) { func (c *context) blendFunc(mode driver.CompositeMode) {
_ = c.t.Call(func() error { if c.lastCompositeMode == mode {
if c.lastCompositeMode == mode { return
return nil }
} c.lastCompositeMode = mode
c.lastCompositeMode = mode s, d := mode.Operations()
s, d := mode.Operations() s2, d2 := convertOperation(s), convertOperation(d)
s2, d2 := convertOperation(s), convertOperation(d) gl.BlendFunc(uint32(s2), uint32(d2))
gl.BlendFunc(uint32(s2), uint32(d2))
return nil
})
} }
func (c *context) newTexture(width, height int) (textureNative, error) { func (c *context) newTexture(width, height int) (textureNative, error) {
var texture textureNative var t uint32
if err := c.t.Call(func() error { gl.GenTextures(1, &t)
var t uint32 // TODO: Use gl.IsTexture
gl.GenTextures(1, &t) if t <= 0 {
// TODO: Use gl.IsTexture return 0, errors.New("opengl: creating texture failed")
if t <= 0 {
return errors.New("opengl: creating texture failed")
}
gl.PixelStorei(gl.UNPACK_ALIGNMENT, 4)
texture = textureNative(t)
return nil
}); err != nil {
return 0, err
} }
gl.PixelStorei(gl.UNPACK_ALIGNMENT, 4)
texture := textureNative(t)
c.bindTexture(texture) c.bindTexture(texture)
_ = c.t.Call(func() error { gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST)
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST) gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST)
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST) gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE)
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE) gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE)
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE) // If data is nil, this just allocates memory and the content is undefined.
// If data is nil, this just allocates memory and the content is undefined. // https://www.khronos.org/registry/OpenGL-Refpages/gl4/html/glTexImage2D.xhtml
// https://www.khronos.org/registry/OpenGL-Refpages/gl4/html/glTexImage2D.xhtml gl.TexImage2D(gl.TEXTURE_2D, 0, gl.RGBA, int32(width), int32(height), 0, gl.RGBA, gl.UNSIGNED_BYTE, nil)
gl.TexImage2D(gl.TEXTURE_2D, 0, gl.RGBA, int32(width), int32(height), 0, gl.RGBA, gl.UNSIGNED_BYTE, nil)
return nil
})
return texture, nil return texture, nil
} }
func (c *context) bindFramebufferImpl(f framebufferNative) { func (c *context) bindFramebufferImpl(f framebufferNative) {
_ = c.t.Call(func() error { gl.BindFramebufferEXT(gl.FRAMEBUFFER, uint32(f))
gl.BindFramebufferEXT(gl.FRAMEBUFFER, uint32(f))
return nil
})
} }
func (c *context) framebufferPixels(f *framebuffer, width, height int) ([]byte, error) { func (c *context) framebufferPixels(f *framebuffer, width, height int) []byte {
var pixels []byte gl.Flush()
_ = c.t.Call(func() error {
gl.Flush()
return nil
})
c.bindFramebuffer(f.native) c.bindFramebuffer(f.native)
if err := c.t.Call(func() error { pixels := make([]byte, 4*width*height)
pixels = make([]byte, 4*width*height) gl.ReadPixels(0, 0, int32(width), int32(height), gl.RGBA, gl.UNSIGNED_BYTE, gl.Ptr(pixels))
gl.ReadPixels(0, 0, int32(width), int32(height), gl.RGBA, gl.UNSIGNED_BYTE, gl.Ptr(pixels)) return pixels
return nil
}); err != nil {
return nil, err
}
return pixels, nil
} }
func (c *context) activeTexture(idx int) { func (c *context) activeTexture(idx int) {
_ = c.t.Call(func() error { gl.ActiveTexture(gl.TEXTURE0 + uint32(idx))
gl.ActiveTexture(gl.TEXTURE0 + uint32(idx))
return nil
})
} }
func (c *context) bindTextureImpl(t textureNative) { func (c *context) bindTextureImpl(t textureNative) {
_ = c.t.Call(func() error { gl.BindTexture(gl.TEXTURE_2D, uint32(t))
gl.BindTexture(gl.TEXTURE_2D, uint32(t))
return nil
})
} }
func (c *context) deleteTexture(t textureNative) { func (c *context) deleteTexture(t textureNative) {
_ = c.t.Call(func() error { tt := uint32(t)
tt := uint32(t) if !gl.IsTexture(tt) {
if !gl.IsTexture(tt) { return
return nil }
} if c.lastTexture == t {
if c.lastTexture == t { c.lastTexture = invalidTexture
c.lastTexture = invalidTexture }
} gl.DeleteTextures(1, &tt)
gl.DeleteTextures(1, &tt)
return nil
})
} }
func (c *context) isTexture(t textureNative) bool { func (c *context) isTexture(t textureNative) bool {
@ -233,155 +192,114 @@ func (c *context) isTexture(t textureNative) bool {
} }
func (c *context) newFramebuffer(texture textureNative) (framebufferNative, error) { func (c *context) newFramebuffer(texture textureNative) (framebufferNative, error) {
var framebuffer framebufferNative
var f uint32 var f uint32
if err := c.t.Call(func() error { gl.GenFramebuffersEXT(1, &f)
gl.GenFramebuffersEXT(1, &f) // TODO: Use gl.IsFramebuffer
// TODO: Use gl.IsFramebuffer if f <= 0 {
if f <= 0 { return 0, errors.New("opengl: creating framebuffer failed: gl.IsFramebuffer returns false")
return errors.New("opengl: creating framebuffer failed: gl.IsFramebuffer returns false")
}
return nil
}); err != nil {
return 0, err
} }
c.bindFramebuffer(framebufferNative(f)) c.bindFramebuffer(framebufferNative(f))
if err := c.t.Call(func() error { gl.FramebufferTexture2DEXT(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, uint32(texture), 0)
gl.FramebufferTexture2DEXT(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, uint32(texture), 0) s := gl.CheckFramebufferStatusEXT(gl.FRAMEBUFFER)
s := gl.CheckFramebufferStatusEXT(gl.FRAMEBUFFER) if s != gl.FRAMEBUFFER_COMPLETE {
if s != gl.FRAMEBUFFER_COMPLETE { if s != 0 {
if s != 0 { return 0, fmt.Errorf("opengl: creating framebuffer failed: %v", s)
return fmt.Errorf("opengl: creating framebuffer failed: %v", s)
}
if e := gl.GetError(); e != gl.NO_ERROR {
return fmt.Errorf("opengl: creating framebuffer failed: (glGetError) %d", e)
}
return fmt.Errorf("opengl: creating framebuffer failed: unknown error")
} }
framebuffer = framebufferNative(f) if e := gl.GetError(); e != gl.NO_ERROR {
return nil return 0, fmt.Errorf("opengl: creating framebuffer failed: (glGetError) %d", e)
}); err != nil { }
return 0, err return 0, fmt.Errorf("opengl: creating framebuffer failed: unknown error")
} }
return framebuffer, nil return framebufferNative(f), nil
} }
func (c *context) setViewportImpl(width, height int) { func (c *context) setViewportImpl(width, height int) {
_ = c.t.Call(func() error { gl.Viewport(0, 0, int32(width), int32(height))
gl.Viewport(0, 0, int32(width), int32(height))
return nil
})
} }
func (c *context) deleteFramebuffer(f framebufferNative) { func (c *context) deleteFramebuffer(f framebufferNative) {
_ = c.t.Call(func() error { ff := uint32(f)
ff := uint32(f) if !gl.IsFramebufferEXT(ff) {
if !gl.IsFramebufferEXT(ff) { return
return nil }
} if c.lastFramebuffer == f {
if c.lastFramebuffer == f { c.lastFramebuffer = invalidFramebuffer
c.lastFramebuffer = invalidFramebuffer c.lastViewportWidth = 0
c.lastViewportWidth = 0 c.lastViewportHeight = 0
c.lastViewportHeight = 0 }
} gl.DeleteFramebuffersEXT(1, &ff)
gl.DeleteFramebuffersEXT(1, &ff)
return nil
})
} }
func (c *context) newShader(shaderType shaderType, source string) (shader, error) { func (c *context) newShader(shaderType shaderType, source string) (shader, error) {
var sh shader s := gl.CreateShader(uint32(shaderType))
if err := c.t.Call(func() error { if s == 0 {
s := gl.CreateShader(uint32(shaderType)) return 0, fmt.Errorf("opengl: glCreateShader failed: shader type: %d", shaderType)
if s == 0 {
return fmt.Errorf("opengl: glCreateShader failed: shader type: %d", shaderType)
}
cSources, free := gl.Strs(source + "\x00")
gl.ShaderSource(uint32(s), 1, cSources, nil)
free()
gl.CompileShader(s)
var v int32
gl.GetShaderiv(s, gl.COMPILE_STATUS, &v)
if v == gl.FALSE {
var l int32
var log []byte
gl.GetShaderiv(uint32(s), gl.INFO_LOG_LENGTH, &l)
if l != 0 {
log = make([]byte, l)
gl.GetShaderInfoLog(s, l, nil, (*uint8)(gl.Ptr(log)))
}
return fmt.Errorf("opengl: shader compile failed: %s", log)
}
sh = shader(s)
return nil
}); err != nil {
return 0, err
} }
return sh, nil cSources, free := gl.Strs(source + "\x00")
gl.ShaderSource(uint32(s), 1, cSources, nil)
free()
gl.CompileShader(s)
var v int32
gl.GetShaderiv(s, gl.COMPILE_STATUS, &v)
if v == gl.FALSE {
var l int32
var log []byte
gl.GetShaderiv(uint32(s), gl.INFO_LOG_LENGTH, &l)
if l != 0 {
log = make([]byte, l)
gl.GetShaderInfoLog(s, l, nil, (*uint8)(gl.Ptr(log)))
}
return 0, fmt.Errorf("opengl: shader compile failed: %s", log)
}
return shader(s), nil
} }
func (c *context) deleteShader(s shader) { func (c *context) deleteShader(s shader) {
_ = c.t.Call(func() error { gl.DeleteShader(uint32(s))
gl.DeleteShader(uint32(s))
return nil
})
} }
func (c *context) newProgram(shaders []shader, attributes []string) (program, error) { func (c *context) newProgram(shaders []shader, attributes []string) (program, error) {
var pr program p := gl.CreateProgram()
if err := c.t.Call(func() error { if p == 0 {
p := gl.CreateProgram() return 0, errors.New("opengl: glCreateProgram failed")
if p == 0 {
return errors.New("opengl: glCreateProgram failed")
}
for _, shader := range shaders {
gl.AttachShader(p, uint32(shader))
}
for i, name := range attributes {
l, free := gl.Strs(name + "\x00")
gl.BindAttribLocation(p, uint32(i), *l)
free()
}
gl.LinkProgram(p)
var v int32
gl.GetProgramiv(p, gl.LINK_STATUS, &v)
if v == gl.FALSE {
var l int32
var log []byte
gl.GetProgramiv(p, gl.INFO_LOG_LENGTH, &l)
if l != 0 {
log = make([]byte, l)
gl.GetProgramInfoLog(p, l, nil, (*uint8)(gl.Ptr(log)))
}
return fmt.Errorf("opengl: program error: %s", log)
}
pr = program(p)
return nil
}); err != nil {
return 0, err
} }
return pr, nil
for _, shader := range shaders {
gl.AttachShader(p, uint32(shader))
}
for i, name := range attributes {
l, free := gl.Strs(name + "\x00")
gl.BindAttribLocation(p, uint32(i), *l)
free()
}
gl.LinkProgram(p)
var v int32
gl.GetProgramiv(p, gl.LINK_STATUS, &v)
if v == gl.FALSE {
var l int32
var log []byte
gl.GetProgramiv(p, gl.INFO_LOG_LENGTH, &l)
if l != 0 {
log = make([]byte, l)
gl.GetProgramInfoLog(p, l, nil, (*uint8)(gl.Ptr(log)))
}
return 0, fmt.Errorf("opengl: program error: %s", log)
}
return program(p), nil
} }
func (c *context) useProgram(p program) { func (c *context) useProgram(p program) {
_ = c.t.Call(func() error { gl.UseProgram(uint32(p))
gl.UseProgram(uint32(p))
return nil
})
} }
func (c *context) deleteProgram(p program) { func (c *context) deleteProgram(p program) {
_ = c.t.Call(func() error { if !gl.IsProgram(uint32(p)) {
if !gl.IsProgram(uint32(p)) { return
return nil }
} gl.DeleteProgram(uint32(p))
gl.DeleteProgram(uint32(p))
return nil
})
} }
func (c *context) getUniformLocationImpl(p program, location string) uniformLocation { func (c *context) getUniformLocationImpl(p program, location string) uniformLocation {
@ -392,164 +310,110 @@ func (c *context) getUniformLocationImpl(p program, location string) uniformLoca
} }
func (c *context) uniformInt(p program, location string, v int) bool { func (c *context) uniformInt(p program, location string, v int) bool {
var r bool l := int32(c.locationCache.GetUniformLocation(c, p, location))
_ = c.t.Call(func() error { if l == invalidUniform {
l := int32(c.locationCache.GetUniformLocation(c, p, location)) return false
if l == invalidUniform { }
return nil gl.Uniform1i(l, int32(v))
} return true
r = true
gl.Uniform1i(l, int32(v))
return nil
})
return r
} }
func (c *context) uniformFloat(p program, location string, v float32) bool { func (c *context) uniformFloat(p program, location string, v float32) bool {
var r bool l := int32(c.locationCache.GetUniformLocation(c, p, location))
_ = c.t.Call(func() error { if l == invalidUniform {
l := int32(c.locationCache.GetUniformLocation(c, p, location)) return false
if l == invalidUniform { }
return nil gl.Uniform1f(l, v)
} return true
r = true
gl.Uniform1f(l, v)
return nil
})
return r
} }
func (c *context) uniformFloats(p program, location string, v []float32, typ shaderir.Type) bool { func (c *context) uniformFloats(p program, location string, v []float32, typ shaderir.Type) bool {
var r bool l := int32(c.locationCache.GetUniformLocation(c, p, location))
_ = c.t.Call(func() error { if l == invalidUniform {
l := int32(c.locationCache.GetUniformLocation(c, p, location)) return false
if l == invalidUniform { }
return nil
}
r = true
base := typ.Main base := typ.Main
len := int32(1) len := int32(1)
if base == shaderir.Array { if base == shaderir.Array {
base = typ.Sub[0].Main base = typ.Sub[0].Main
len = int32(typ.Length) len = int32(typ.Length)
} }
switch base { switch base {
case shaderir.Float: case shaderir.Float:
gl.Uniform1fv(l, len, (*float32)(gl.Ptr(v))) gl.Uniform1fv(l, len, (*float32)(gl.Ptr(v)))
case shaderir.Vec2: case shaderir.Vec2:
gl.Uniform2fv(l, len, (*float32)(gl.Ptr(v))) gl.Uniform2fv(l, len, (*float32)(gl.Ptr(v)))
case shaderir.Vec3: case shaderir.Vec3:
gl.Uniform3fv(l, len, (*float32)(gl.Ptr(v))) gl.Uniform3fv(l, len, (*float32)(gl.Ptr(v)))
case shaderir.Vec4: case shaderir.Vec4:
gl.Uniform4fv(l, len, (*float32)(gl.Ptr(v))) gl.Uniform4fv(l, len, (*float32)(gl.Ptr(v)))
case shaderir.Mat2: case shaderir.Mat2:
gl.UniformMatrix2fv(l, len, false, (*float32)(gl.Ptr(v))) gl.UniformMatrix2fv(l, len, false, (*float32)(gl.Ptr(v)))
case shaderir.Mat3: case shaderir.Mat3:
gl.UniformMatrix3fv(l, len, false, (*float32)(gl.Ptr(v))) gl.UniformMatrix3fv(l, len, false, (*float32)(gl.Ptr(v)))
case shaderir.Mat4: case shaderir.Mat4:
gl.UniformMatrix4fv(l, len, false, (*float32)(gl.Ptr(v))) gl.UniformMatrix4fv(l, len, false, (*float32)(gl.Ptr(v)))
default: default:
panic(fmt.Sprintf("opengl: unexpected type: %s", typ.String())) panic(fmt.Sprintf("opengl: unexpected type: %s", typ.String()))
} }
return nil return true
})
return r
} }
func (c *context) vertexAttribPointer(p program, index int, size int, dataType dataType, stride int, offset int) { func (c *context) vertexAttribPointer(p program, index int, size int, dataType dataType, stride int, offset int) {
_ = c.t.Call(func() error { gl.VertexAttribPointer(uint32(index), int32(size), uint32(dataType), false, int32(stride), uintptr(offset))
gl.VertexAttribPointer(uint32(index), int32(size), uint32(dataType), false, int32(stride), uintptr(offset))
return nil
})
} }
func (c *context) enableVertexAttribArray(p program, index int) { func (c *context) enableVertexAttribArray(p program, index int) {
_ = c.t.Call(func() error { gl.EnableVertexAttribArray(uint32(index))
gl.EnableVertexAttribArray(uint32(index))
return nil
})
} }
func (c *context) disableVertexAttribArray(p program, index int) { func (c *context) disableVertexAttribArray(p program, index int) {
_ = c.t.Call(func() error { gl.DisableVertexAttribArray(uint32(index))
gl.DisableVertexAttribArray(uint32(index))
return nil
})
} }
func (c *context) newArrayBuffer(size int) buffer { func (c *context) newArrayBuffer(size int) buffer {
var bf buffer var b uint32
_ = c.t.Call(func() error { gl.GenBuffers(1, &b)
var b uint32 gl.BindBuffer(uint32(arrayBuffer), b)
gl.GenBuffers(1, &b) gl.BufferData(uint32(arrayBuffer), size, nil, uint32(dynamicDraw))
gl.BindBuffer(uint32(arrayBuffer), b) return buffer(b)
gl.BufferData(uint32(arrayBuffer), size, nil, uint32(dynamicDraw))
bf = buffer(b)
return nil
})
return bf
} }
func (c *context) newElementArrayBuffer(size int) buffer { func (c *context) newElementArrayBuffer(size int) buffer {
var bf buffer var b uint32
_ = c.t.Call(func() error { gl.GenBuffers(1, &b)
var b uint32 gl.BindBuffer(uint32(elementArrayBuffer), b)
gl.GenBuffers(1, &b) gl.BufferData(uint32(elementArrayBuffer), size, nil, uint32(dynamicDraw))
gl.BindBuffer(uint32(elementArrayBuffer), b) return buffer(b)
gl.BufferData(uint32(elementArrayBuffer), size, nil, uint32(dynamicDraw))
bf = buffer(b)
return nil
})
return bf
} }
func (c *context) bindBuffer(bufferType bufferType, b buffer) { func (c *context) bindBuffer(bufferType bufferType, b buffer) {
_ = c.t.Call(func() error { gl.BindBuffer(uint32(bufferType), uint32(b))
gl.BindBuffer(uint32(bufferType), uint32(b))
return nil
})
} }
func (c *context) arrayBufferSubData(data []float32) { func (c *context) arrayBufferSubData(data []float32) {
_ = c.t.Call(func() error { gl.BufferSubData(uint32(arrayBuffer), 0, len(data)*4, gl.Ptr(data))
gl.BufferSubData(uint32(arrayBuffer), 0, len(data)*4, gl.Ptr(data))
return nil
})
} }
func (c *context) elementArrayBufferSubData(data []uint16) { func (c *context) elementArrayBufferSubData(data []uint16) {
_ = c.t.Call(func() error { gl.BufferSubData(uint32(elementArrayBuffer), 0, len(data)*2, gl.Ptr(data))
gl.BufferSubData(uint32(elementArrayBuffer), 0, len(data)*2, gl.Ptr(data))
return nil
})
} }
func (c *context) deleteBuffer(b buffer) { func (c *context) deleteBuffer(b buffer) {
_ = c.t.Call(func() error { bb := uint32(b)
bb := uint32(b) gl.DeleteBuffers(1, &bb)
gl.DeleteBuffers(1, &bb)
return nil
})
} }
func (c *context) drawElements(len int, offsetInBytes int) { func (c *context) drawElements(len int, offsetInBytes int) {
_ = c.t.Call(func() error { gl.DrawElements(gl.TRIANGLES, int32(len), gl.UNSIGNED_SHORT, uintptr(offsetInBytes))
gl.DrawElements(gl.TRIANGLES, int32(len), gl.UNSIGNED_SHORT, uintptr(offsetInBytes))
return nil
})
} }
func (c *context) maxTextureSizeImpl() int { func (c *context) maxTextureSizeImpl() int {
size := 0 s := int32(0)
_ = c.t.Call(func() error { gl.GetIntegerv(gl.MAX_TEXTURE_SIZE, &s)
s := int32(0) return int(s)
gl.GetIntegerv(gl.MAX_TEXTURE_SIZE, &s)
size = int(s)
return nil
})
return size
} }
func (c *context) getShaderPrecisionFormatPrecision() int { func (c *context) getShaderPrecisionFormatPrecision() int {
@ -559,10 +423,7 @@ func (c *context) getShaderPrecisionFormatPrecision() int {
} }
func (c *context) flush() { func (c *context) flush() {
_ = c.t.Call(func() error { gl.Flush()
gl.Flush()
return nil
})
} }
func (c *context) needsRestoring() bool { func (c *context) needsRestoring() bool {
@ -570,54 +431,37 @@ func (c *context) needsRestoring() bool {
} }
func (c *context) canUsePBO() bool { func (c *context) canUsePBO() bool {
var available bool return isPBOAvailable()
_ = c.t.Call(func() error {
available = isPBOAvailable()
return nil
})
return available
} }
func (c *context) texSubImage2D(t textureNative, width, height int, args []*driver.ReplacePixelsArgs) { func (c *context) texSubImage2D(t textureNative, width, height int, args []*driver.ReplacePixelsArgs) {
c.bindTexture(t) c.bindTexture(t)
_ = c.t.Call(func() error { for _, a := range args {
for _, a := range args { gl.TexSubImage2D(gl.TEXTURE_2D, 0, int32(a.X), int32(a.Y), int32(a.Width), int32(a.Height), gl.RGBA, gl.UNSIGNED_BYTE, gl.Ptr(a.Pixels))
gl.TexSubImage2D(gl.TEXTURE_2D, 0, int32(a.X), int32(a.Y), int32(a.Width), int32(a.Height), gl.RGBA, gl.UNSIGNED_BYTE, gl.Ptr(a.Pixels)) }
}
return nil
})
} }
func (c *context) newPixelBufferObject(width, height int) buffer { func (c *context) newPixelBufferObject(width, height int) buffer {
var bf buffer var b uint32
_ = c.t.Call(func() error { gl.GenBuffers(1, &b)
var b uint32 gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, b)
gl.GenBuffers(1, &b) gl.BufferData(gl.PIXEL_UNPACK_BUFFER, 4*width*height, nil, gl.STREAM_DRAW)
gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, b) gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, 0)
gl.BufferData(gl.PIXEL_UNPACK_BUFFER, 4*width*height, nil, gl.STREAM_DRAW) return buffer(b)
gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, 0)
bf = buffer(b)
return nil
})
return bf
} }
func (c *context) replacePixelsWithPBO(buffer buffer, t textureNative, width, height int, args []*driver.ReplacePixelsArgs) { func (c *context) replacePixelsWithPBO(buffer buffer, t textureNative, width, height int, args []*driver.ReplacePixelsArgs) {
c.bindTexture(t) c.bindTexture(t)
_ = c.t.Call(func() error { gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, uint32(buffer))
gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, uint32(buffer))
stride := 4 * width stride := 4 * width
for _, a := range args { for _, a := range args {
offset := 4 * (a.Y*width + a.X) offset := 4 * (a.Y*width + a.X)
for j := 0; j < a.Height; j++ { for j := 0; j < a.Height; j++ {
gl.BufferSubData(gl.PIXEL_UNPACK_BUFFER, offset+stride*j, 4*a.Width, gl.Ptr(a.Pixels[4*a.Width*j:4*a.Width*(j+1)])) gl.BufferSubData(gl.PIXEL_UNPACK_BUFFER, offset+stride*j, 4*a.Width, gl.Ptr(a.Pixels[4*a.Width*j:4*a.Width*(j+1)]))
}
} }
}
gl.TexSubImage2D(gl.TEXTURE_2D, 0, 0, 0, int32(width), int32(height), gl.RGBA, gl.UNSIGNED_BYTE, nil) gl.TexSubImage2D(gl.TEXTURE_2D, 0, 0, 0, int32(width), int32(height), gl.RGBA, gl.UNSIGNED_BYTE, nil)
gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, 0) gl.BindBuffer(gl.PIXEL_UNPACK_BUFFER, 0)
return nil
})
} }

View File

@ -273,7 +273,7 @@ func (c *context) bindFramebufferImpl(f framebufferNative) {
gl.Call("bindFramebuffer", framebuffer_, js.Value(f)) gl.Call("bindFramebuffer", framebuffer_, js.Value(f))
} }
func (c *context) framebufferPixels(f *framebuffer, width, height int) ([]byte, error) { func (c *context) framebufferPixels(f *framebuffer, width, height int) []byte {
c.ensureGL() c.ensureGL()
gl := c.gl gl := c.gl
@ -282,7 +282,7 @@ func (c *context) framebufferPixels(f *framebuffer, width, height int) ([]byte,
p := jsutil.TemporaryUint8Array(4 * width * height) p := jsutil.TemporaryUint8Array(4 * width * height)
gl.Call("readPixels", 0, 0, width, height, rgba, unsignedByte, p) gl.Call("readPixels", 0, 0, width, height, rgba, unsignedByte, p)
return jsutil.Uint8ArrayToSlice(p), nil return jsutil.Uint8ArrayToSlice(p)
} }
func (c *context) activeTexture(idx int) { func (c *context) activeTexture(idx int) {

View File

@ -148,7 +148,7 @@ func (c *context) bindFramebufferImpl(f framebufferNative) {
gl.BindFramebuffer(mgl.FRAMEBUFFER, mgl.Framebuffer(f)) gl.BindFramebuffer(mgl.FRAMEBUFFER, mgl.Framebuffer(f))
} }
func (c *context) framebufferPixels(f *framebuffer, width, height int) ([]byte, error) { func (c *context) framebufferPixels(f *framebuffer, width, height int) []byte {
gl := c.gl gl := c.gl
gl.Flush() gl.Flush()
@ -156,7 +156,7 @@ func (c *context) framebufferPixels(f *framebuffer, width, height int) ([]byte,
pixels := make([]byte, 4*width*height) pixels := make([]byte, 4*width*height)
gl.ReadPixels(pixels, 0, 0, width, height, mgl.RGBA, mgl.UNSIGNED_BYTE) gl.ReadPixels(pixels, 0, 0, width, height, mgl.RGBA, mgl.UNSIGNED_BYTE)
return pixels, nil return pixels
} }
func (c *context) activeTexture(idx int) { func (c *context) activeTexture(idx int) {

View File

@ -21,7 +21,6 @@ import (
"github.com/hajimehoshi/ebiten/v2/internal/driver" "github.com/hajimehoshi/ebiten/v2/internal/driver"
"github.com/hajimehoshi/ebiten/v2/internal/graphics" "github.com/hajimehoshi/ebiten/v2/internal/graphics"
"github.com/hajimehoshi/ebiten/v2/internal/shaderir" "github.com/hajimehoshi/ebiten/v2/internal/shaderir"
"github.com/hajimehoshi/ebiten/v2/internal/thread"
) )
var theGraphics Graphics var theGraphics Graphics
@ -44,10 +43,6 @@ type Graphics struct {
drawCalled bool drawCalled bool
} }
func (g *Graphics) SetThread(thread *thread.Thread) {
g.context.t = thread
}
func (g *Graphics) Begin() { func (g *Graphics) Begin() {
// Do nothing. // Do nothing.
} }

View File

@ -64,10 +64,7 @@ func (i *Image) Pixels() ([]byte, error) {
if err := i.ensureFramebuffer(); err != nil { if err := i.ensureFramebuffer(); err != nil {
return nil, err return nil, err
} }
p, err := i.graphics.context.framebufferPixels(i.framebuffer, i.width, i.height) p := i.graphics.context.framebufferPixels(i.framebuffer, i.width, i.height)
if err != nil {
return nil, err
}
return p, nil return p, nil
} }

View File

@ -29,6 +29,7 @@ import (
"github.com/hajimehoshi/ebiten/v2/internal/devicescale" "github.com/hajimehoshi/ebiten/v2/internal/devicescale"
"github.com/hajimehoshi/ebiten/v2/internal/driver" "github.com/hajimehoshi/ebiten/v2/internal/driver"
"github.com/hajimehoshi/ebiten/v2/internal/glfw" "github.com/hajimehoshi/ebiten/v2/internal/glfw"
"github.com/hajimehoshi/ebiten/v2/internal/graphicscommand"
"github.com/hajimehoshi/ebiten/v2/internal/hooks" "github.com/hajimehoshi/ebiten/v2/internal/hooks"
"github.com/hajimehoshi/ebiten/v2/internal/thread" "github.com/hajimehoshi/ebiten/v2/internal/thread"
) )
@ -588,7 +589,7 @@ func (u *UserInterface) Run(uicontext driver.UIContext) error {
// Initialize the main thread first so the thread is available at u.run (#809). // Initialize the main thread first so the thread is available at u.run (#809).
u.t = thread.New() u.t = thread.New()
u.Graphics().SetThread(u.t) graphicscommand.SetMainThread(u.t)
ch := make(chan error, 1) ch := make(chan error, 1)
go func() { go func() {

View File

@ -34,6 +34,7 @@ import (
"github.com/hajimehoshi/ebiten/v2/internal/devicescale" "github.com/hajimehoshi/ebiten/v2/internal/devicescale"
"github.com/hajimehoshi/ebiten/v2/internal/driver" "github.com/hajimehoshi/ebiten/v2/internal/driver"
"github.com/hajimehoshi/ebiten/v2/internal/graphicscommand"
"github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/opengl" "github.com/hajimehoshi/ebiten/v2/internal/graphicsdriver/opengl"
"github.com/hajimehoshi/ebiten/v2/internal/hooks" "github.com/hajimehoshi/ebiten/v2/internal/hooks"
"github.com/hajimehoshi/ebiten/v2/internal/restorable" "github.com/hajimehoshi/ebiten/v2/internal/restorable"
@ -315,7 +316,7 @@ func (u *UserInterface) run(context driver.UIContext, mainloop bool) (err error)
u.Graphics().(*opengl.Graphics).SetMobileGLContext(ctx) u.Graphics().(*opengl.Graphics).SetMobileGLContext(ctx)
} else { } else {
u.t = thread.New() u.t = thread.New()
u.Graphics().SetThread(u.t) graphicscommand.SetMainThread(u.t)
} }
// If gomobile-build is used, wait for the outside size fixed. // If gomobile-build is used, wait for the outside size fixed.