// 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 testing import ( "go/constant" "github.com/hajimehoshi/ebiten/internal/shaderir" ) var ( projectionMatrix = shaderir.Expr{ Type: shaderir.Call, Exprs: []shaderir.Expr{ { Type: shaderir.BuiltinFuncExpr, BuiltinFunc: shaderir.Mat4F, }, { Type: shaderir.Binary, Op: shaderir.Div, Exprs: []shaderir.Expr{ { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(2), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.FieldSelector, Exprs: []shaderir.Expr{ { Type: shaderir.UniformVariable, Index: 0, }, { Type: shaderir.SwizzlingExpr, Swizzling: "x", }, }, }, }, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.Binary, Op: shaderir.Div, Exprs: []shaderir.Expr{ { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(2), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.FieldSelector, Exprs: []shaderir.Expr{ { Type: shaderir.UniformVariable, Index: 0, }, { Type: shaderir.SwizzlingExpr, Swizzling: "y", }, }, }, }, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(1), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(-1), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(-1), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(1), ConstType: shaderir.ConstTypeFloat, }, }, } vertexPosition = shaderir.Expr{ Type: shaderir.Call, Exprs: []shaderir.Expr{ { Type: shaderir.BuiltinFuncExpr, BuiltinFunc: shaderir.Vec4F, }, { Type: shaderir.LocalVariable, Index: 0, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(0), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(1), ConstType: shaderir.ConstTypeFloat, }, }, } defaultVertexFunc = shaderir.VertexFunc{ Block: shaderir.Block{ Stmts: []shaderir.Stmt{ { Type: shaderir.Assign, Exprs: []shaderir.Expr{ { Type: shaderir.LocalVariable, Index: 5, // the varying variable }, { Type: shaderir.LocalVariable, Index: 1, // the 2nd attribute variable }, }, }, { Type: shaderir.Assign, Exprs: []shaderir.Expr{ { Type: shaderir.LocalVariable, Index: 4, // gl_Position in GLSL }, { Type: shaderir.Binary, Op: shaderir.Mul, Exprs: []shaderir.Expr{ projectionMatrix, vertexPosition, }, }, }, }, }, }, } ) func defaultProgram() shaderir.Program { return shaderir.Program{ Uniforms: []shaderir.Type{ {Main: shaderir.Vec2}, // Viewport size. This must be the first uniform variable, and the values are set at graphicscommand driver. }, Attributes: []shaderir.Type{ {Main: shaderir.Vec2}, // Local var (0) in the vertex shader {Main: shaderir.Vec2}, // Local var (1) in the vertex shader {Main: shaderir.Vec4}, // Local var (2) in the vertex shader {Main: shaderir.Vec4}, // Local var (3) in the vertex shader }, Varyings: []shaderir.Type{ {Main: shaderir.Vec2}, // Local var (4) in the vertex shader, (0) in the fragment shader }, VertexFunc: defaultVertexFunc, } } // ShaderProgramFill returns a shader intermediate representation to fill the frambuffer. // // Uniform variable's index and its value are: // // 0: the framebuffer size (Vec2) func ShaderProgramFill(r, g, b, a byte) shaderir.Program { clr := shaderir.Expr{ Type: shaderir.Call, Exprs: []shaderir.Expr{ { Type: shaderir.BuiltinFuncExpr, BuiltinFunc: shaderir.Vec4F, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(float64(r) / 0xff), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(float64(g) / 0xff), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(float64(b) / 0xff), ConstType: shaderir.ConstTypeFloat, }, { Type: shaderir.NumberExpr, Const: constant.MakeFloat64(float64(a) / 0xff), ConstType: shaderir.ConstTypeFloat, }, }, } p := defaultProgram() p.FragmentFunc = shaderir.FragmentFunc{ Block: shaderir.Block{ Stmts: []shaderir.Stmt{ { Type: shaderir.Assign, Exprs: []shaderir.Expr{ { Type: shaderir.LocalVariable, Index: 2, }, clr, }, }, }, }, } return p } // ShaderProgramImages returns a shader intermediate representation to render the frambuffer with the given images. // // Uniform variables's indices and their values are: // // 0: the framebuffer size (Vec2) // 1-: the images (Texture2D) // // The first image's size and region are represented in attribute variables. // // The size and region values are actually not used in this shader so far. func ShaderProgramImages(imageNum int) shaderir.Program { if imageNum <= 0 { panic("testing: imageNum must be >= 1") } p := defaultProgram() for i := 0; i < imageNum; i++ { p.Uniforms = append(p.Uniforms, shaderir.Type{Main: shaderir.Texture2D}) } // In the fragment shader, local variables are: // // 0: gl_FragCoord // 1: Varying variables (vec2) // 2: gl_FragColor // 3: Actual local variables in the main function local := shaderir.Expr{ Type: shaderir.LocalVariable, Index: 3, } fragColor := shaderir.Expr{ Type: shaderir.LocalVariable, Index: 2, } texPos := shaderir.Expr{ Type: shaderir.LocalVariable, Index: 1, } var stmts []shaderir.Stmt for i := 0; i < imageNum; i++ { var rhs shaderir.Expr if i == 0 { rhs = shaderir.Expr{ Type: shaderir.Call, Exprs: []shaderir.Expr{ { Type: shaderir.BuiltinFuncExpr, BuiltinFunc: shaderir.Texture2DF, }, { Type: shaderir.UniformVariable, Index: 1, }, texPos, }, } } else { rhs = shaderir.Expr{ Type: shaderir.Binary, Op: shaderir.Add, Exprs: []shaderir.Expr{ local, { Type: shaderir.Call, Exprs: []shaderir.Expr{ { Type: shaderir.BuiltinFuncExpr, BuiltinFunc: shaderir.Texture2DF, }, { Type: shaderir.UniformVariable, Index: i + 1, }, texPos, }, }, }, } } stmts = append(stmts, shaderir.Stmt{ Type: shaderir.Assign, Exprs: []shaderir.Expr{ local, rhs, }, }) } stmts = append(stmts, shaderir.Stmt{ Type: shaderir.Assign, Exprs: []shaderir.Expr{ fragColor, local, }, }) p.FragmentFunc = shaderir.FragmentFunc{ Block: shaderir.Block{ LocalVars: []shaderir.Type{ {Main: shaderir.Vec4}, }, Stmts: stmts, }, } return p }