2016-05-07 12:42:07 +02:00
|
|
|
// Copyright 2016 Hajime Hoshi
|
|
|
|
//
|
|
|
|
// 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 opengl
|
|
|
|
|
2017-12-02 19:51:42 +01:00
|
|
|
import (
|
2019-02-07 09:19:24 +01:00
|
|
|
"fmt"
|
2019-06-21 21:47:48 +02:00
|
|
|
"sync"
|
2019-02-07 09:19:24 +01:00
|
|
|
|
2019-06-25 17:43:09 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/driver"
|
2019-06-05 17:05:53 +02:00
|
|
|
"github.com/hajimehoshi/ebiten/internal/thread"
|
2017-12-02 19:51:42 +01:00
|
|
|
)
|
|
|
|
|
2019-06-25 17:43:09 +02:00
|
|
|
func convertOperation(op driver.Operation) operation {
|
2018-10-28 12:42:57 +01:00
|
|
|
switch op {
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.Zero:
|
2018-10-28 12:42:57 +01:00
|
|
|
return zero
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.One:
|
2018-10-28 12:42:57 +01:00
|
|
|
return one
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.SrcAlpha:
|
2018-10-28 12:42:57 +01:00
|
|
|
return srcAlpha
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.DstAlpha:
|
2018-10-28 12:42:57 +01:00
|
|
|
return dstAlpha
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.OneMinusSrcAlpha:
|
2018-10-28 12:42:57 +01:00
|
|
|
return oneMinusSrcAlpha
|
2019-06-25 17:43:09 +02:00
|
|
|
case driver.OneMinusDstAlpha:
|
2018-10-28 12:42:57 +01:00
|
|
|
return oneMinusDstAlpha
|
|
|
|
default:
|
2019-02-07 09:19:24 +01:00
|
|
|
panic(fmt.Sprintf("opengl: invalid operation %d at convertOperation", op))
|
2018-10-28 12:42:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
type context struct {
|
2016-05-07 12:42:07 +02:00
|
|
|
locationCache *locationCache
|
2018-11-04 09:43:26 +01:00
|
|
|
screenFramebuffer framebufferNative // This might not be the default frame buffer '0' (e.g. iOS).
|
|
|
|
lastFramebuffer framebufferNative
|
2018-11-04 11:46:20 +01:00
|
|
|
lastTexture textureNative
|
2016-06-05 00:47:11 +02:00
|
|
|
lastViewportWidth int
|
|
|
|
lastViewportHeight int
|
2019-06-25 17:43:09 +02:00
|
|
|
lastCompositeMode driver.CompositeMode
|
2018-03-09 03:03:55 +01:00
|
|
|
maxTextureSize int
|
2019-07-03 18:15:51 +02:00
|
|
|
maxTextureSizeOnce sync.Once
|
2019-06-21 21:47:48 +02:00
|
|
|
highp bool
|
|
|
|
highpOnce sync.Once
|
2019-06-05 17:05:53 +02:00
|
|
|
|
|
|
|
t *thread.Thread
|
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
contextImpl
|
2016-05-07 12:42:07 +02:00
|
|
|
}
|
2016-05-31 19:33:31 +02:00
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
func (c *context) bindTexture(t textureNative) {
|
2018-02-18 18:49:00 +01:00
|
|
|
if c.lastTexture == t {
|
2017-09-24 17:11:19 +02:00
|
|
|
return
|
2016-07-09 12:15:53 +02:00
|
|
|
}
|
2017-09-24 17:11:19 +02:00
|
|
|
c.bindTextureImpl(t)
|
2016-07-09 12:15:53 +02:00
|
|
|
c.lastTexture = t
|
|
|
|
}
|
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
func (c *context) bindFramebuffer(f framebufferNative) {
|
2018-02-18 18:11:16 +01:00
|
|
|
if c.lastFramebuffer == f {
|
2017-09-24 17:14:25 +02:00
|
|
|
return
|
2016-06-03 20:40:56 +02:00
|
|
|
}
|
2017-09-24 17:14:25 +02:00
|
|
|
c.bindFramebufferImpl(f)
|
2016-06-05 00:47:11 +02:00
|
|
|
c.lastFramebuffer = f
|
2016-06-18 15:47:34 +02:00
|
|
|
}
|
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
func (c *context) setViewport(f *framebuffer) {
|
2018-11-04 09:28:33 +01:00
|
|
|
c.bindFramebuffer(f.native)
|
|
|
|
if c.lastViewportWidth != f.width || c.lastViewportHeight != f.height {
|
2018-11-17 11:29:52 +01:00
|
|
|
// On some environments, viewport size must be within the framebuffer size.
|
|
|
|
// e.g. Edge (#71), Chrome on GPD Pocket (#420), macOS Mojave (#691).
|
|
|
|
// Use the same size of the framebuffer here.
|
2018-11-04 09:28:33 +01:00
|
|
|
c.setViewportImpl(f.width, f.height)
|
2018-11-17 11:29:52 +01:00
|
|
|
|
2018-11-01 19:24:35 +01:00
|
|
|
// glViewport must be called at least at every frame on iOS.
|
|
|
|
// As the screen framebuffer is the last render target, next SetViewport should be
|
|
|
|
// the first call at a frame.
|
2018-11-04 09:28:33 +01:00
|
|
|
if f.native == c.screenFramebuffer {
|
2018-11-01 19:24:35 +01:00
|
|
|
c.lastViewportWidth = 0
|
|
|
|
c.lastViewportHeight = 0
|
|
|
|
} else {
|
2018-11-04 09:28:33 +01:00
|
|
|
c.lastViewportWidth = f.width
|
|
|
|
c.lastViewportHeight = f.height
|
2018-11-01 19:24:35 +01:00
|
|
|
}
|
2016-06-18 15:47:34 +02:00
|
|
|
}
|
2016-05-31 19:33:31 +02:00
|
|
|
}
|
2016-06-17 21:46:33 +02:00
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
func (c *context) getScreenFramebuffer() framebufferNative {
|
2016-06-17 21:46:33 +02:00
|
|
|
return c.screenFramebuffer
|
|
|
|
}
|
2016-06-17 23:25:40 +02:00
|
|
|
|
2018-11-10 17:14:26 +01:00
|
|
|
func (c *context) getMaxTextureSize() int {
|
2019-07-03 18:15:51 +02:00
|
|
|
c.maxTextureSizeOnce.Do(func() {
|
2018-03-09 03:03:55 +01:00
|
|
|
c.maxTextureSize = c.maxTextureSizeImpl()
|
2019-07-03 18:15:51 +02:00
|
|
|
})
|
2018-03-09 03:03:55 +01:00
|
|
|
return c.maxTextureSize
|
|
|
|
}
|
2019-06-21 21:47:48 +02:00
|
|
|
|
|
|
|
// highpPrecision represents an enough mantissa of float values in a shader.
|
|
|
|
const highpPrecision = 23
|
|
|
|
|
|
|
|
func (c *context) hasHighPrecisionFloat() bool {
|
|
|
|
c.highpOnce.Do(func() {
|
|
|
|
c.highp = c.getShaderPrecisionFormatPrecision() >= highpPrecision
|
|
|
|
})
|
|
|
|
return c.highp
|
|
|
|
}
|