mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-26 02:42:02 +01:00
parent
bd43b42ee5
commit
35f597e682
@ -345,8 +345,6 @@ func (c *drawTrianglesCommand) String() string {
|
||||
filter = "nearest"
|
||||
case graphicsdriver.FilterLinear:
|
||||
filter = "linear"
|
||||
case graphicsdriver.FilterScreen:
|
||||
filter = "screen"
|
||||
default:
|
||||
panic(fmt.Sprintf("graphicscommand: invalid filter: %d", c.filter))
|
||||
}
|
||||
|
@ -1254,10 +1254,6 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcs [graphics.Sh
|
||||
var esBody [16]float32
|
||||
var esTranslate [4]float32
|
||||
colorM.Elements(&esBody, &esTranslate)
|
||||
scale := float32(0)
|
||||
if filter == graphicsdriver.FilterScreen {
|
||||
scale = float32(dst.width) / float32(srcImages[0].width)
|
||||
}
|
||||
|
||||
flattenUniforms = []float32{
|
||||
float32(screenWidth),
|
||||
@ -1288,7 +1284,6 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcs [graphics.Sh
|
||||
srcRegion.Y,
|
||||
srcRegion.X + srcRegion.Width,
|
||||
srcRegion.Y + srcRegion.Height,
|
||||
scale,
|
||||
}
|
||||
} else {
|
||||
// TODO: This logic is very similar to Metal's. Let's unify them.
|
||||
|
@ -73,9 +73,6 @@ func (k *builtinPipelineStatesKey) defs() ([]_D3D_SHADER_MACRO, error) {
|
||||
case graphicsdriver.FilterLinear:
|
||||
name := []byte("FILTER_LINEAR\x00")
|
||||
defs = append(defs, _D3D_SHADER_MACRO{&name[0], &defval[0]})
|
||||
case graphicsdriver.FilterScreen:
|
||||
name := []byte("FILTER_SCREEN\x00")
|
||||
defs = append(defs, _D3D_SHADER_MACRO{&name[0], &defval[0]})
|
||||
default:
|
||||
return nil, fmt.Errorf("directx: invalid filter: %d", k.filter)
|
||||
}
|
||||
@ -113,10 +110,6 @@ cbuffer ShaderParameter : register(b0) {
|
||||
float4x4 color_matrix_body;
|
||||
float4 color_matrix_translation;
|
||||
float4 source_region;
|
||||
|
||||
// This member should be the last not to create a new sector.
|
||||
// https://docs.microsoft.com/en-us/windows/win32/direct3dhlsl/dx-graphics-hlsl-packing-rules
|
||||
float scale;
|
||||
}
|
||||
|
||||
PSInput VSMain(float2 position : POSITION, float2 tex : TEXCOORD, float4 color : COLOR) {
|
||||
@ -220,25 +213,6 @@ float4 PSMain(PSInput input) : SV_TARGET {
|
||||
float4 color = lerp(lerp(c0, c1, rate.x), lerp(c2, c3, rate.x), rate.y);
|
||||
#endif // defined(FILTER_LINEAR)
|
||||
|
||||
#if defined(FILTER_SCREEN)
|
||||
float2 pos = input.texcoord;
|
||||
float2 texel_size = 1.0 / source_size;
|
||||
float2 half_scaled_texel_size = texel_size / 2.0 / scale;
|
||||
|
||||
float2 p0 = pos - half_scaled_texel_size + (texel_size / 512.0);
|
||||
float2 p1 = pos + half_scaled_texel_size + (texel_size / 512.0);
|
||||
|
||||
float4 c0 = tex.Sample(samp, p0);
|
||||
float4 c1 = tex.Sample(samp, float2(p1.x, p0.y));
|
||||
float4 c2 = tex.Sample(samp, float2(p0.x, p1.y));
|
||||
float4 c3 = tex.Sample(samp, p1);
|
||||
// Texels must be in the source rect, so it is not necessary to check that like linear filter.
|
||||
|
||||
float2 rate_center = float2(1.0, 1.0) - half_scaled_texel_size;
|
||||
float2 rate = clamp(((frac(p0 * source_size) - rate_center) * scale) + rate_center, 0.0, 1.0);
|
||||
float4 color = lerp(lerp(c0, c1, rate.x), lerp(c2, c3, rate.x), rate.y);
|
||||
#endif // defined(FILTER_SCREEN)
|
||||
|
||||
#if defined(USE_COLOR_MATRIX)
|
||||
// Un-premultiply alpha.
|
||||
// When the alpha is 0, 1.0 - sign(alpha) is 1.0, which means division does nothing.
|
||||
@ -252,8 +226,6 @@ float4 PSMain(PSInput input) : SV_TARGET {
|
||||
// Clamp the output.
|
||||
color.rgb = min(color.rgb, color.a);
|
||||
return color;
|
||||
#elif defined(FILTER_SCREEN)
|
||||
return color;
|
||||
#else
|
||||
return input.color * color;
|
||||
#endif // defined(USE_COLOR_MATRIX)
|
||||
|
@ -19,7 +19,6 @@ type Filter int
|
||||
const (
|
||||
FilterNearest Filter = iota
|
||||
FilterLinear
|
||||
FilterScreen
|
||||
)
|
||||
|
||||
type Address int
|
||||
|
@ -33,7 +33,6 @@ const source = `#include <metal_stdlib>
|
||||
|
||||
#define FILTER_NEAREST {{.FilterNearest}}
|
||||
#define FILTER_LINEAR {{.FilterLinear}}
|
||||
#define FILTER_SCREEN {{.FilterScreen}}
|
||||
|
||||
#define ADDRESS_CLAMP_TO_ZERO {{.AddressClampToZero}}
|
||||
#define ADDRESS_REPEAT {{.AddressRepeat}}
|
||||
@ -105,7 +104,7 @@ constexpr sampler texture_sampler{filter::nearest};
|
||||
|
||||
template<>
|
||||
struct ColorFromTexel<FILTER_NEAREST, ADDRESS_UNSAFE> {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region, float scale) {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region) {
|
||||
float2 p = v.tex;
|
||||
return texture.sample(texture_sampler, p);
|
||||
}
|
||||
@ -113,7 +112,7 @@ struct ColorFromTexel<FILTER_NEAREST, ADDRESS_UNSAFE> {
|
||||
|
||||
template<uint8_t address>
|
||||
struct ColorFromTexel<FILTER_NEAREST, address> {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region, float scale) {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region) {
|
||||
float2 p = AdjustTexelByAddress<address>(v.tex, source_region);
|
||||
if (source_region[0] <= p.x &&
|
||||
source_region[1] <= p.y &&
|
||||
@ -127,7 +126,7 @@ struct ColorFromTexel<FILTER_NEAREST, address> {
|
||||
|
||||
template<>
|
||||
struct ColorFromTexel<FILTER_LINEAR, ADDRESS_UNSAFE> {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region, float scale) {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region) {
|
||||
const float2 texel_size = 1 / source_size;
|
||||
|
||||
// Shift 1/512 [texel] to avoid the tie-breaking issue.
|
||||
@ -147,7 +146,7 @@ struct ColorFromTexel<FILTER_LINEAR, ADDRESS_UNSAFE> {
|
||||
|
||||
template<uint8_t address>
|
||||
struct ColorFromTexel<FILTER_LINEAR, address> {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region, float scale) {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region) {
|
||||
const float2 texel_size = 1 / source_size;
|
||||
|
||||
// Shift 1/512 [texel] to avoid the tie-breaking issue.
|
||||
@ -184,25 +183,6 @@ struct ColorFromTexel<FILTER_LINEAR, address> {
|
||||
}
|
||||
};
|
||||
|
||||
template<uint8_t address>
|
||||
struct ColorFromTexel<FILTER_SCREEN, address> {
|
||||
inline float4 Do(VertexOut v, texture2d<float> texture, constant float2& source_size, constant float4& source_region, float scale) {
|
||||
const float2 texel_size = 1 / source_size;
|
||||
|
||||
float2 p0 = v.tex - texel_size / 2.0 / scale + (texel_size / 512.0);
|
||||
float2 p1 = v.tex + texel_size / 2.0 / scale + (texel_size / 512.0);
|
||||
|
||||
float4 c0 = texture.sample(texture_sampler, p0);
|
||||
float4 c1 = texture.sample(texture_sampler, float2(p1.x, p0.y));
|
||||
float4 c2 = texture.sample(texture_sampler, float2(p0.x, p1.y));
|
||||
float4 c3 = texture.sample(texture_sampler, p1);
|
||||
|
||||
float2 rate_center = float2(1.0, 1.0) - texel_size / 2.0 / scale;
|
||||
float2 rate = clamp(((fract(p0 * source_size) - rate_center) * scale) + rate_center, 0.0, 1.0);
|
||||
return mix(mix(c0, c1, rate.x), mix(c2, c3, rate.x), rate.y);
|
||||
}
|
||||
};
|
||||
|
||||
template<bool useColorM, uint8_t filter, uint8_t address>
|
||||
struct FragmentShaderImpl {
|
||||
inline float4 Do(
|
||||
@ -211,9 +191,8 @@ struct FragmentShaderImpl {
|
||||
constant float2& source_size,
|
||||
constant float4x4& color_matrix_body,
|
||||
constant float4& color_matrix_translation,
|
||||
constant float4& source_region,
|
||||
constant float& scale) {
|
||||
float4 c = ColorFromTexel<filter, address>().Do(v, texture, source_size, source_region, scale);
|
||||
constant float4& source_region) {
|
||||
float4 c = ColorFromTexel<filter, address>().Do(v, texture, source_size, source_region);
|
||||
if (useColorM) {
|
||||
c.rgb /= c.a + (1.0 - sign(c.a));
|
||||
c = (color_matrix_body * c) + color_matrix_translation;
|
||||
@ -227,20 +206,6 @@ struct FragmentShaderImpl {
|
||||
}
|
||||
};
|
||||
|
||||
template<bool useColorM, uint8_t address>
|
||||
struct FragmentShaderImpl<useColorM, FILTER_SCREEN, address> {
|
||||
inline float4 Do(
|
||||
VertexOut v,
|
||||
texture2d<float> texture,
|
||||
constant float2& source_size,
|
||||
constant float4x4& color_matrix_body,
|
||||
constant float4& color_matrix_translation,
|
||||
constant float4& source_region,
|
||||
constant float& scale) {
|
||||
return ColorFromTexel<FILTER_SCREEN, address>().Do(v, texture, source_size, source_region, scale);
|
||||
}
|
||||
};
|
||||
|
||||
// Define Foo and FooCp macros to force macro replacement.
|
||||
// See "6.10.3.1 Argument substitution" in ISO/IEC 9899.
|
||||
|
||||
@ -254,10 +219,9 @@ struct FragmentShaderImpl<useColorM, FILTER_SCREEN, address> {
|
||||
constant float2& source_size [[buffer(2)]], \
|
||||
constant float4x4& color_matrix_body [[buffer(3)]], \
|
||||
constant float4& color_matrix_translation [[buffer(4)]], \
|
||||
constant float4& source_region [[buffer(5)]], \
|
||||
constant float& scale [[buffer(6)]]) { \
|
||||
constant float4& source_region [[buffer(5)]]) { \
|
||||
return FragmentShaderImpl<useColorM, filter, address>().Do( \
|
||||
v, texture, source_size, color_matrix_body, color_matrix_translation, source_region, scale); \
|
||||
v, texture, source_size, color_matrix_body, color_matrix_translation, source_region); \
|
||||
}
|
||||
|
||||
FragmentShaderFunc(0, FILTER_NEAREST, ADDRESS_CLAMP_TO_ZERO)
|
||||
@ -273,8 +237,6 @@ FragmentShaderFunc(1, FILTER_LINEAR, ADDRESS_REPEAT)
|
||||
FragmentShaderFunc(1, FILTER_NEAREST, ADDRESS_UNSAFE)
|
||||
FragmentShaderFunc(1, FILTER_LINEAR, ADDRESS_UNSAFE)
|
||||
|
||||
FragmentShaderFunc(0, FILTER_SCREEN, ADDRESS_UNSAFE)
|
||||
|
||||
#undef FragmentShaderFuncName
|
||||
`
|
||||
|
||||
@ -290,12 +252,11 @@ type rpsKey struct {
|
||||
type Graphics struct {
|
||||
view view
|
||||
|
||||
screenRPS mtl.RenderPipelineState
|
||||
rpss map[rpsKey]mtl.RenderPipelineState
|
||||
cq mtl.CommandQueue
|
||||
cb mtl.CommandBuffer
|
||||
rce mtl.RenderCommandEncoder
|
||||
dsss map[stencilMode]mtl.DepthStencilState
|
||||
rpss map[rpsKey]mtl.RenderPipelineState
|
||||
cq mtl.CommandQueue
|
||||
cb mtl.CommandBuffer
|
||||
rce mtl.RenderCommandEncoder
|
||||
dsss map[stencilMode]mtl.DepthStencilState
|
||||
|
||||
screenDrawable ca.MetalDrawable
|
||||
|
||||
@ -608,7 +569,6 @@ func (g *Graphics) Initialize() error {
|
||||
replaces := map[string]string{
|
||||
"{{.FilterNearest}}": fmt.Sprintf("%d", graphicsdriver.FilterNearest),
|
||||
"{{.FilterLinear}}": fmt.Sprintf("%d", graphicsdriver.FilterLinear),
|
||||
"{{.FilterScreen}}": fmt.Sprintf("%d", graphicsdriver.FilterScreen),
|
||||
"{{.AddressClampToZero}}": fmt.Sprintf("%d", graphicsdriver.AddressClampToZero),
|
||||
"{{.AddressRepeat}}": fmt.Sprintf("%d", graphicsdriver.AddressRepeat),
|
||||
"{{.AddressUnsafe}}": fmt.Sprintf("%d", graphicsdriver.AddressUnsafe),
|
||||
@ -626,27 +586,6 @@ func (g *Graphics) Initialize() error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
fs, err := lib.MakeFunction(
|
||||
fmt.Sprintf("FragmentShader_%d_%d_%d", 0, graphicsdriver.FilterScreen, graphicsdriver.AddressUnsafe))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
rpld := mtl.RenderPipelineDescriptor{
|
||||
VertexFunction: vs,
|
||||
FragmentFunction: fs,
|
||||
}
|
||||
rpld.ColorAttachments[0].PixelFormat = g.view.colorPixelFormat()
|
||||
rpld.ColorAttachments[0].BlendingEnabled = true
|
||||
rpld.ColorAttachments[0].DestinationAlphaBlendFactor = mtl.BlendFactorZero
|
||||
rpld.ColorAttachments[0].DestinationRGBBlendFactor = mtl.BlendFactorZero
|
||||
rpld.ColorAttachments[0].SourceAlphaBlendFactor = mtl.BlendFactorOne
|
||||
rpld.ColorAttachments[0].SourceRGBBlendFactor = mtl.BlendFactorOne
|
||||
rpld.ColorAttachments[0].WriteMask = mtl.ColorWriteMaskAll
|
||||
rps, err := g.view.getMTLDevice().MakeRenderPipelineState(rpld)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
g.screenRPS = rps
|
||||
|
||||
for _, screen := range []bool{false, true} {
|
||||
for _, cm := range []bool{false, true} {
|
||||
@ -871,23 +810,19 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcIDs [graphics.
|
||||
rpss := map[stencilMode]mtl.RenderPipelineState{}
|
||||
var uniformVars [][]float32
|
||||
if shaderID == graphicsdriver.InvalidShaderID {
|
||||
if dst.screen && filter == graphicsdriver.FilterScreen {
|
||||
rpss[noStencil] = g.screenRPS
|
||||
} else {
|
||||
for _, stencil := range []stencilMode{
|
||||
prepareStencil,
|
||||
drawWithStencil,
|
||||
noStencil,
|
||||
} {
|
||||
rpss[stencil] = g.rpss[rpsKey{
|
||||
screen: dst.screen,
|
||||
useColorM: !colorM.IsIdentity(),
|
||||
filter: filter,
|
||||
address: address,
|
||||
compositeMode: mode,
|
||||
stencilMode: stencil,
|
||||
}]
|
||||
}
|
||||
for _, stencil := range []stencilMode{
|
||||
prepareStencil,
|
||||
drawWithStencil,
|
||||
noStencil,
|
||||
} {
|
||||
rpss[stencil] = g.rpss[rpsKey{
|
||||
screen: dst.screen,
|
||||
useColorM: !colorM.IsIdentity(),
|
||||
filter: filter,
|
||||
address: address,
|
||||
compositeMode: mode,
|
||||
stencilMode: stencil,
|
||||
}]
|
||||
}
|
||||
|
||||
w, h := dst.internalSize()
|
||||
@ -900,10 +835,6 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcIDs [graphics.
|
||||
var esBody [16]float32
|
||||
var esTranslate [4]float32
|
||||
colorM.Elements(&esBody, &esTranslate)
|
||||
scale := float32(0)
|
||||
if filter == graphicsdriver.FilterScreen {
|
||||
scale = float32(dst.width) / float32(srcs[0].width)
|
||||
}
|
||||
uniformVars = [][]float32{
|
||||
{float32(w), float32(h)},
|
||||
sourceSize,
|
||||
@ -915,7 +846,6 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcIDs [graphics.
|
||||
srcRegion.X + srcRegion.Width,
|
||||
srcRegion.Y + srcRegion.Height,
|
||||
},
|
||||
{scale},
|
||||
}
|
||||
} else {
|
||||
for _, stencil := range []stencilMode{
|
||||
|
@ -83,8 +83,6 @@ func fragmentShaderStr(useColorM bool, filter graphicsdriver.Filter, address gra
|
||||
defs = append(defs, "#define FILTER_NEAREST")
|
||||
case graphicsdriver.FilterLinear:
|
||||
defs = append(defs, "#define FILTER_LINEAR")
|
||||
case graphicsdriver.FilterScreen:
|
||||
defs = append(defs, "#define FILTER_SCREEN")
|
||||
default:
|
||||
panic(fmt.Sprintf("opengl: invalid filter: %d", filter))
|
||||
}
|
||||
@ -151,10 +149,6 @@ uniform vec4 color_matrix_translation;
|
||||
|
||||
uniform highp vec2 source_size;
|
||||
|
||||
#if defined(FILTER_SCREEN)
|
||||
uniform highp float scale;
|
||||
#endif
|
||||
|
||||
varying highp vec2 varying_tex;
|
||||
varying highp vec4 varying_color_scale;
|
||||
|
||||
@ -235,27 +229,6 @@ void main(void) {
|
||||
color = mix(mix(c0, c1, rate.x), mix(c2, c3, rate.x), rate.y);
|
||||
#endif // defined(FILTER_LINEAR)
|
||||
|
||||
#if defined(FILTER_SCREEN)
|
||||
highp vec2 texel_size = 1.0 / source_size;
|
||||
highp vec2 half_scaled_texel_size = texel_size / 2.0 / scale;
|
||||
|
||||
highp vec2 p0 = pos - half_scaled_texel_size + (texel_size / 512.0);
|
||||
highp vec2 p1 = pos + half_scaled_texel_size + (texel_size / 512.0);
|
||||
|
||||
vec4 c0 = texture2D(T0, p0);
|
||||
vec4 c1 = texture2D(T0, vec2(p1.x, p0.y));
|
||||
vec4 c2 = texture2D(T0, vec2(p0.x, p1.y));
|
||||
vec4 c3 = texture2D(T0, p1);
|
||||
// Texels must be in the source rect, so it is not necessary to check that like linear filter.
|
||||
|
||||
vec2 rate_center = vec2(1.0, 1.0) - half_scaled_texel_size;
|
||||
vec2 rate = clamp(((fract(p0 * source_size) - rate_center) * scale) + rate_center, 0.0, 1.0);
|
||||
gl_FragColor = mix(mix(c0, c1, rate.x), mix(c2, c3, rate.x), rate.y);
|
||||
|
||||
// Assume that a color matrix and color vector values are not used with FILTER_SCREEN.
|
||||
|
||||
#else
|
||||
|
||||
# if defined(USE_COLOR_MATRIX)
|
||||
// Un-premultiply alpha.
|
||||
// When the alpha is 0, 1.0 - sign(alpha) is 1.0, which means division does nothing.
|
||||
@ -275,9 +248,6 @@ void main(void) {
|
||||
# endif // defined(USE_COLOR_MATRIX)
|
||||
|
||||
gl_FragColor = color;
|
||||
|
||||
#endif // defined(FILTER_SCREEN)
|
||||
|
||||
}
|
||||
`
|
||||
)
|
||||
|
@ -244,15 +244,6 @@ func (g *Graphics) DrawTriangles(dstID graphicsdriver.ImageID, srcIDs [graphics.
|
||||
typ: shaderir.Type{Main: shaderir.Vec2},
|
||||
})
|
||||
}
|
||||
|
||||
if filter == graphicsdriver.FilterScreen {
|
||||
scale := float32(destination.width) / float32(g.images[srcIDs[0]].width)
|
||||
g.uniformVars = append(g.uniformVars, uniformVariable{
|
||||
name: "scale",
|
||||
value: []float32{scale},
|
||||
typ: shaderir.Type{Main: shaderir.Float},
|
||||
})
|
||||
}
|
||||
} else {
|
||||
shader := g.shaders[shaderID]
|
||||
program = shader.p
|
||||
|
@ -191,7 +191,6 @@ func (s *openGLState) reset(context *context) error {
|
||||
for _, f := range []graphicsdriver.Filter{
|
||||
graphicsdriver.FilterNearest,
|
||||
graphicsdriver.FilterLinear,
|
||||
graphicsdriver.FilterScreen,
|
||||
} {
|
||||
shaderFragmentColorMatrixNative, err := context.newFragmentShader(fragmentShaderStr(c, f, a))
|
||||
if err != nil {
|
||||
|
@ -65,7 +65,7 @@ func (m *Mipmap) DrawTriangles(srcs [graphics.ShaderImageCount]*Mipmap, vertices
|
||||
|
||||
level := 0
|
||||
// TODO: Do we need to check all the sources' states of being volatile?
|
||||
if !canSkipMipmap && srcs[0] != nil && !srcs[0].volatile && filter != graphicsdriver.FilterScreen {
|
||||
if !canSkipMipmap && srcs[0] != nil && !srcs[0].volatile {
|
||||
level = math.MaxInt32
|
||||
for i := 0; i < len(indices)/3; i++ {
|
||||
const n = graphics.VertexFloatCount
|
||||
@ -226,10 +226,6 @@ func (m *Mipmap) disposeMipmaps() {
|
||||
func mipmapLevelFromDistance(dx0, dy0, dx1, dy1, sx0, sy0, sx1, sy1 float32, filter graphicsdriver.Filter) int {
|
||||
const maxLevel = 6
|
||||
|
||||
if filter == graphicsdriver.FilterScreen {
|
||||
return 0
|
||||
}
|
||||
|
||||
d := (dx1-dx0)*(dx1-dx0) + (dy1-dy0)*(dy1-dy0)
|
||||
s := (sx1-sx0)*(sx1-sx0) + (sy1-sy0)*(sy1-sy0)
|
||||
if s == 0 {
|
||||
|
Loading…
Reference in New Issue
Block a user