Add internal/shaderir/glsl

This commit is contained in:
Hajime Hoshi 2020-08-03 23:21:09 +09:00
parent 28af45bf0f
commit 8bac08cbd9
7 changed files with 209 additions and 185 deletions

View File

@ -19,6 +19,7 @@ import (
"github.com/hajimehoshi/ebiten/internal/driver" "github.com/hajimehoshi/ebiten/internal/driver"
"github.com/hajimehoshi/ebiten/internal/shaderir" "github.com/hajimehoshi/ebiten/internal/shaderir"
"github.com/hajimehoshi/ebiten/internal/shaderir/glsl"
) )
type Shader struct { type Shader struct {
@ -51,7 +52,7 @@ func (s *Shader) Dispose() {
} }
func (s *Shader) compile() error { func (s *Shader) compile() error {
vssrc, fssrc := s.ir.Glsl() vssrc, fssrc := glsl.Compile(s.ir)
vs, err := s.graphics.context.newShader(vertexShader, vssrc) vs, err := s.graphics.context.newShader(vertexShader, vssrc)
if err != nil { if err != nil {

View File

@ -24,12 +24,12 @@ import (
"testing" "testing"
. "github.com/hajimehoshi/ebiten/internal/shader" . "github.com/hajimehoshi/ebiten/internal/shader"
"github.com/hajimehoshi/ebiten/internal/shaderir" "github.com/hajimehoshi/ebiten/internal/shaderir/glsl"
) )
func normalize(str string) string { func normalize(str string) string {
if strings.HasPrefix(str, shaderir.GlslFragmentPrelude) { if strings.HasPrefix(str, glsl.FragmentPrelude) {
str = str[len(shaderir.GlslFragmentPrelude):] str = str[len(glsl.FragmentPrelude):]
} }
return strings.TrimSpace(str) return strings.TrimSpace(str)
} }
@ -114,7 +114,7 @@ func TestCompile(t *testing.T) {
t.Error(err) t.Error(err)
return return
} }
vs, fs := s.Glsl() vs, fs := glsl.Compile(s)
if got, want := normalize(vs), normalize(string(tc.VS)); got != want { if got, want := normalize(vs), normalize(string(tc.VS)); got != want {
t.Errorf("got: %v, want: %v", got, want) t.Errorf("got: %v, want: %v", got, want)
} }

View File

@ -12,16 +12,18 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
package shaderir package glsl
import ( import (
"fmt" "fmt"
"go/constant" "go/constant"
"go/token" "go/token"
"strings" "strings"
"github.com/hajimehoshi/ebiten/internal/shaderir"
) )
const GlslFragmentPrelude = `#if defined(GL_ES) const FragmentPrelude = `#if defined(GL_ES)
precision highp float; precision highp float;
#else #else
#define lowp #define lowp
@ -66,23 +68,29 @@ func isValidSwizzling(s string) bool {
return false return false
} }
func (p *Program) structName(t *Type) string { type compileContext struct {
if t.Main != Struct { structNames map[string]string
structTypes []shaderir.Type
}
func (c *compileContext) structName(p *shaderir.Program, t *shaderir.Type) string {
if t.Main != shaderir.Struct {
panic("shaderir: the given type at structName must be a struct") panic("shaderir: the given type at structName must be a struct")
} }
s := t.serialize() s := t.String()
if n, ok := p.structNames[s]; ok { if n, ok := c.structNames[s]; ok {
return n return n
} }
n := fmt.Sprintf("S%d", len(p.structNames)) n := fmt.Sprintf("S%d", len(c.structNames))
p.structNames[s] = n c.structNames[s] = n
p.structTypes = append(p.structTypes, *t) c.structTypes = append(c.structTypes, *t)
return n return n
} }
func (p *Program) Glsl() (vertexShader, fragmentShader string) { func Compile(p *shaderir.Program) (vertexShader, fragmentShader string) {
p.structNames = map[string]string{} c := &compileContext{
p.structTypes = nil structNames: map[string]string{},
}
// Vertex func // Vertex func
var vslines []string var vslines []string
@ -92,16 +100,16 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
vslines = append(vslines, "") vslines = append(vslines, "")
} }
for i, t := range p.Uniforms { for i, t := range p.Uniforms {
vslines = append(vslines, fmt.Sprintf("uniform %s;", p.glslVarDecl(&t, fmt.Sprintf("U%d", i)))) vslines = append(vslines, fmt.Sprintf("uniform %s;", c.glslVarDecl(p, &t, fmt.Sprintf("U%d", i))))
} }
for i := 0; i < p.TextureNum; i++ { for i := 0; i < p.TextureNum; i++ {
vslines = append(vslines, fmt.Sprintf("uniform sampler2D T%d;", i)) vslines = append(vslines, fmt.Sprintf("uniform sampler2D T%d;", i))
} }
for i, t := range p.Attributes { for i, t := range p.Attributes {
vslines = append(vslines, fmt.Sprintf("attribute %s;", p.glslVarDecl(&t, fmt.Sprintf("A%d", i)))) vslines = append(vslines, fmt.Sprintf("attribute %s;", c.glslVarDecl(p, &t, fmt.Sprintf("A%d", i))))
} }
for i, t := range p.Varyings { for i, t := range p.Varyings {
vslines = append(vslines, fmt.Sprintf("varying %s;", p.glslVarDecl(&t, fmt.Sprintf("V%d", i)))) vslines = append(vslines, fmt.Sprintf("varying %s;", c.glslVarDecl(p, &t, fmt.Sprintf("V%d", i))))
} }
} }
if len(p.Funcs) > 0 { if len(p.Funcs) > 0 {
@ -109,13 +117,13 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
vslines = append(vslines, "") vslines = append(vslines, "")
} }
for _, f := range p.Funcs { for _, f := range p.Funcs {
vslines = append(vslines, p.glslFunc(&f, true)...) vslines = append(vslines, c.glslFunc(p, &f, true)...)
} }
for _, f := range p.Funcs { for _, f := range p.Funcs {
if len(vslines) > 0 && vslines[len(vslines)-1] != "" { if len(vslines) > 0 && vslines[len(vslines)-1] != "" {
vslines = append(vslines, "") vslines = append(vslines, "")
} }
vslines = append(vslines, p.glslFunc(&f, false)...) vslines = append(vslines, c.glslFunc(p, &f, false)...)
} }
} }
@ -124,7 +132,7 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
vslines = append(vslines, "") vslines = append(vslines, "")
} }
vslines = append(vslines, "void main(void) {") vslines = append(vslines, "void main(void) {")
vslines = append(vslines, p.glslBlock(&p.VertexFunc.Block, &p.VertexFunc.Block, 0, 0)...) vslines = append(vslines, c.glslBlock(p, &p.VertexFunc.Block, &p.VertexFunc.Block, 0, 0)...)
vslines = append(vslines, "}") vslines = append(vslines, "}")
} }
} }
@ -137,13 +145,13 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
fslines = append(fslines, "") fslines = append(fslines, "")
} }
for i, t := range p.Uniforms { for i, t := range p.Uniforms {
fslines = append(fslines, fmt.Sprintf("uniform %s;", p.glslVarDecl(&t, fmt.Sprintf("U%d", i)))) fslines = append(fslines, fmt.Sprintf("uniform %s;", c.glslVarDecl(p, &t, fmt.Sprintf("U%d", i))))
} }
for i := 0; i < p.TextureNum; i++ { for i := 0; i < p.TextureNum; i++ {
fslines = append(fslines, fmt.Sprintf("uniform sampler2D T%d;", i)) fslines = append(fslines, fmt.Sprintf("uniform sampler2D T%d;", i))
} }
for i, t := range p.Varyings { for i, t := range p.Varyings {
fslines = append(fslines, fmt.Sprintf("varying %s;", p.glslVarDecl(&t, fmt.Sprintf("V%d", i)))) fslines = append(fslines, fmt.Sprintf("varying %s;", c.glslVarDecl(p, &t, fmt.Sprintf("V%d", i))))
} }
} }
if len(p.Funcs) > 0 { if len(p.Funcs) > 0 {
@ -151,13 +159,13 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
fslines = append(fslines, "") fslines = append(fslines, "")
} }
for _, f := range p.Funcs { for _, f := range p.Funcs {
fslines = append(fslines, p.glslFunc(&f, true)...) fslines = append(fslines, c.glslFunc(p, &f, true)...)
} }
for _, f := range p.Funcs { for _, f := range p.Funcs {
if len(fslines) > 0 && fslines[len(fslines)-1] != "" { if len(fslines) > 0 && fslines[len(fslines)-1] != "" {
fslines = append(fslines, "") fslines = append(fslines, "")
} }
fslines = append(fslines, p.glslFunc(&f, false)...) fslines = append(fslines, c.glslFunc(p, &f, false)...)
} }
} }
@ -166,21 +174,21 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
fslines = append(fslines, "") fslines = append(fslines, "")
} }
fslines = append(fslines, "void main(void) {") fslines = append(fslines, "void main(void) {")
fslines = append(fslines, p.glslBlock(&p.FragmentFunc.Block, &p.FragmentFunc.Block, 0, 0)...) fslines = append(fslines, c.glslBlock(p, &p.FragmentFunc.Block, &p.FragmentFunc.Block, 0, 0)...)
fslines = append(fslines, "}") fslines = append(fslines, "}")
} }
} }
var tmpvslines []string var tmpvslines []string
tmpfslines := strings.Split(GlslFragmentPrelude, "\n") tmpfslines := strings.Split(FragmentPrelude, "\n")
// Struct types are determined after converting the program. // Struct types are determined after converting the program.
var stlines []string var stlines []string
if len(p.structTypes) > 0 { if len(c.structTypes) > 0 {
for i, t := range p.structTypes { for i, t := range c.structTypes {
stlines = append(stlines, fmt.Sprintf("struct S%d {", i)) stlines = append(stlines, fmt.Sprintf("struct S%d {", i))
for j, st := range t.Sub { for j, st := range t.Sub {
stlines = append(stlines, fmt.Sprintf("\t%s;", p.glslVarDecl(&st, fmt.Sprintf("M%d", j)))) stlines = append(stlines, fmt.Sprintf("\t%s;", c.glslVarDecl(p, &st, fmt.Sprintf("M%d", j))))
} }
stlines = append(stlines, "};") stlines = append(stlines, "};")
} }
@ -211,76 +219,76 @@ func (p *Program) Glsl() (vertexShader, fragmentShader string) {
return strings.Join(vslines, "\n") + "\n", strings.Join(fslines, "\n") + "\n" return strings.Join(vslines, "\n") + "\n", strings.Join(fslines, "\n") + "\n"
} }
func (p *Program) glslType(t *Type) (string, string) { func (c *compileContext) glslType(p *shaderir.Program, t *shaderir.Type) (string, string) {
switch t.Main { switch t.Main {
case None: case shaderir.None:
return "void", "" return "void", ""
case Struct: case shaderir.Struct:
return p.structName(t), "" return c.structName(p, t), ""
default: default:
return t.Glsl() return typeString(t)
} }
} }
func (p *Program) glslVarDecl(t *Type, varname string) string { func (c *compileContext) glslVarDecl(p *shaderir.Program, t *shaderir.Type, varname string) string {
switch t.Main { switch t.Main {
case None: case shaderir.None:
return "?(none)" return "?(none)"
case Struct: case shaderir.Struct:
return fmt.Sprintf("%s %s", p.structName(t), varname) return fmt.Sprintf("%s %s", c.structName(p, t), varname)
default: default:
t0, t1 := t.Glsl() t0, t1 := typeString(t)
return fmt.Sprintf("%s %s%s", t0, varname, t1) return fmt.Sprintf("%s %s%s", t0, varname, t1)
} }
} }
func (p *Program) glslVarInit(t *Type) string { func (c *compileContext) glslVarInit(p *shaderir.Program, t *shaderir.Type) string {
switch t.Main { switch t.Main {
case None: case shaderir.None:
return "?(none)" return "?(none)"
case Array: case shaderir.Array:
init := p.glslVarInit(&t.Sub[0]) init := c.glslVarInit(p, &t.Sub[0])
es := make([]string, 0, t.Length) es := make([]string, 0, t.Length)
for i := 0; i < t.Length; i++ { for i := 0; i < t.Length; i++ {
es = append(es, init) es = append(es, init)
} }
t0, t1 := t.Glsl() t0, t1 := typeString(t)
return fmt.Sprintf("%s%s(%s)", t0, t1, strings.Join(es, ", ")) return fmt.Sprintf("%s%s(%s)", t0, t1, strings.Join(es, ", "))
case Struct: case shaderir.Struct:
panic("not implemented") panic("not implemented")
case Bool: case shaderir.Bool:
return "false" return "false"
case Int: case shaderir.Int:
return "0" return "0"
case Float: case shaderir.Float:
return "float(0)" return "float(0)"
case Vec2: case shaderir.Vec2:
return "vec2(0)" return "vec2(0)"
case Vec3: case shaderir.Vec3:
return "vec3(0)" return "vec3(0)"
case Vec4: case shaderir.Vec4:
return "vec4(0)" return "vec4(0)"
case Mat2: case shaderir.Mat2:
return "mat2(0)" return "mat2(0)"
case Mat3: case shaderir.Mat3:
return "mat3(0)" return "mat3(0)"
case Mat4: case shaderir.Mat4:
return "mat4(0)" return "mat4(0)"
default: default:
t0, t1 := p.glslType(t) t0, t1 := c.glslType(p, t)
panic(fmt.Sprintf("?(unexpected type: %s%s)", t0, t1)) panic(fmt.Sprintf("?(unexpected type: %s%s)", t0, t1))
} }
} }
func (p *Program) glslFunc(f *Func, prototype bool) []string { func (c *compileContext) glslFunc(p *shaderir.Program, f *shaderir.Func, prototype bool) []string {
var args []string var args []string
var idx int var idx int
for _, t := range f.InParams { for _, t := range f.InParams {
args = append(args, "in "+p.glslVarDecl(&t, fmt.Sprintf("l%d", idx))) args = append(args, "in "+c.glslVarDecl(p, &t, fmt.Sprintf("l%d", idx)))
idx++ idx++
} }
for _, t := range f.OutParams { for _, t := range f.OutParams {
args = append(args, "out "+p.glslVarDecl(&t, fmt.Sprintf("l%d", idx))) args = append(args, "out "+c.glslVarDecl(p, &t, fmt.Sprintf("l%d", idx)))
idx++ idx++
} }
argsstr := "void" argsstr := "void"
@ -288,7 +296,7 @@ func (p *Program) glslFunc(f *Func, prototype bool) []string {
argsstr = strings.Join(args, ", ") argsstr = strings.Join(args, ", ")
} }
t0, t1 := p.glslType(&f.Return) t0, t1 := c.glslType(p, &f.Return)
sig := fmt.Sprintf("%s%s F%d(%s)", t0, t1, f.Index, argsstr) sig := fmt.Sprintf("%s%s F%d(%s)", t0, t1, f.Index, argsstr)
var lines []string var lines []string
@ -297,15 +305,15 @@ func (p *Program) glslFunc(f *Func, prototype bool) []string {
return lines return lines
} }
lines = append(lines, fmt.Sprintf("%s {", sig)) lines = append(lines, fmt.Sprintf("%s {", sig))
lines = append(lines, p.glslBlock(&f.Block, &f.Block, 0, idx)...) lines = append(lines, c.glslBlock(p, &f.Block, &f.Block, 0, idx)...)
lines = append(lines, "}") lines = append(lines, "}")
return lines return lines
} }
func constantToNumberLiteral(t ConstType, v constant.Value) string { func constantToNumberLiteral(t shaderir.ConstType, v constant.Value) string {
switch t { switch t {
case ConstTypeNone: case shaderir.ConstTypeNone:
if v.Kind() == constant.Bool { if v.Kind() == constant.Bool {
if constant.BoolVal(v) { if constant.BoolVal(v) {
return "true" return "true"
@ -313,7 +321,7 @@ func constantToNumberLiteral(t ConstType, v constant.Value) string {
return "false" return "false"
} }
fallthrough fallthrough
case ConstTypeFloat: case shaderir.ConstTypeFloat:
if i := constant.ToInt(v); i.Kind() == constant.Int { if i := constant.ToInt(v); i.Kind() == constant.Int {
x, _ := constant.Int64Val(i) x, _ := constant.Int64Val(i)
return fmt.Sprintf("%d.0", x) return fmt.Sprintf("%d.0", x)
@ -322,7 +330,7 @@ func constantToNumberLiteral(t ConstType, v constant.Value) string {
x, _ := constant.Float64Val(i) x, _ := constant.Float64Val(i)
return fmt.Sprintf("%.9e", x) return fmt.Sprintf("%.9e", x)
} }
case ConstTypeInt: case shaderir.ConstTypeInt:
if i := constant.ToInt(v); i.Kind() == constant.Int { if i := constant.ToInt(v); i.Kind() == constant.Int {
x, _ := constant.Int64Val(i) x, _ := constant.Int64Val(i)
return fmt.Sprintf("%d", x) return fmt.Sprintf("%d", x)
@ -331,7 +339,7 @@ func constantToNumberLiteral(t ConstType, v constant.Value) string {
return fmt.Sprintf("?(unexpected literal: %s)", v) return fmt.Sprintf("?(unexpected literal: %s)", v)
} }
func (p *Program) localVariableName(topBlock *Block, idx int) string { func localVariableName(p *shaderir.Program, topBlock *shaderir.Block, idx int) string {
switch topBlock { switch topBlock {
case &p.VertexFunc.Block: case &p.VertexFunc.Block:
na := len(p.Attributes) na := len(p.Attributes)
@ -363,63 +371,63 @@ func (p *Program) localVariableName(topBlock *Block, idx int) string {
} }
} }
func (p *Program) glslBlock(topBlock, block *Block, level int, localVarIndex int) []string { func (c *compileContext) glslBlock(p *shaderir.Program, topBlock, block *shaderir.Block, level int, localVarIndex int) []string {
idt := strings.Repeat("\t", level+1) idt := strings.Repeat("\t", level+1)
var lines []string var lines []string
for _, t := range block.LocalVars { for _, t := range block.LocalVars {
// The type is None e.g., when the variable is a for-loop counter. // The type is None e.g., when the variable is a for-loop counter.
if t.Main != None { if t.Main != shaderir.None {
lines = append(lines, fmt.Sprintf("%s%s = %s;", idt, p.glslVarDecl(&t, fmt.Sprintf("l%d", localVarIndex)), p.glslVarInit(&t))) lines = append(lines, fmt.Sprintf("%s%s = %s;", idt, c.glslVarDecl(p, &t, fmt.Sprintf("l%d", localVarIndex)), c.glslVarInit(p, &t)))
} }
localVarIndex++ localVarIndex++
} }
var glslExpr func(e *Expr) string var glslExpr func(e *shaderir.Expr) string
glslExpr = func(e *Expr) string { glslExpr = func(e *shaderir.Expr) string {
switch e.Type { switch e.Type {
case NumberExpr: case shaderir.NumberExpr:
return constantToNumberLiteral(e.ConstType, e.Const) return constantToNumberLiteral(e.ConstType, e.Const)
case UniformVariable: case shaderir.UniformVariable:
return fmt.Sprintf("U%d", e.Index) return fmt.Sprintf("U%d", e.Index)
case TextureVariable: case shaderir.TextureVariable:
return fmt.Sprintf("T%d", e.Index) return fmt.Sprintf("T%d", e.Index)
case LocalVariable: case shaderir.LocalVariable:
return p.localVariableName(topBlock, e.Index) return localVariableName(p, topBlock, e.Index)
case StructMember: case shaderir.StructMember:
return fmt.Sprintf("M%d", e.Index) return fmt.Sprintf("M%d", e.Index)
case BuiltinFuncExpr: case shaderir.BuiltinFuncExpr:
return e.BuiltinFunc.Glsl() return builtinFuncString(e.BuiltinFunc)
case SwizzlingExpr: case shaderir.SwizzlingExpr:
if !isValidSwizzling(e.Swizzling) { if !isValidSwizzling(e.Swizzling) {
return fmt.Sprintf("?(unexpected swizzling: %s)", e.Swizzling) return fmt.Sprintf("?(unexpected swizzling: %s)", e.Swizzling)
} }
return e.Swizzling return e.Swizzling
case FunctionExpr: case shaderir.FunctionExpr:
return fmt.Sprintf("F%d", e.Index) return fmt.Sprintf("F%d", e.Index)
case Unary: case shaderir.Unary:
var op string var op string
switch e.Op { switch e.Op {
case Add, Sub, NotOp: case shaderir.Add, shaderir.Sub, shaderir.NotOp:
op = string(e.Op) op = string(e.Op)
default: default:
op = fmt.Sprintf("?(unexpected op: %s)", string(e.Op)) op = fmt.Sprintf("?(unexpected op: %s)", string(e.Op))
} }
return fmt.Sprintf("%s(%s)", op, glslExpr(&e.Exprs[0])) return fmt.Sprintf("%s(%s)", op, glslExpr(&e.Exprs[0]))
case Binary: case shaderir.Binary:
return fmt.Sprintf("(%s) %s (%s)", glslExpr(&e.Exprs[0]), e.Op, glslExpr(&e.Exprs[1])) return fmt.Sprintf("(%s) %s (%s)", glslExpr(&e.Exprs[0]), e.Op, glslExpr(&e.Exprs[1]))
case Selection: case shaderir.Selection:
return fmt.Sprintf("(%s) ? (%s) : (%s)", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1]), glslExpr(&e.Exprs[2])) return fmt.Sprintf("(%s) ? (%s) : (%s)", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1]), glslExpr(&e.Exprs[2]))
case Call: case shaderir.Call:
var args []string var args []string
for _, exp := range e.Exprs[1:] { for _, exp := range e.Exprs[1:] {
args = append(args, glslExpr(&exp)) args = append(args, glslExpr(&exp))
} }
// Using parentheses at the callee is illegal. // Using parentheses at the callee is illegal.
return fmt.Sprintf("%s(%s)", glslExpr(&e.Exprs[0]), strings.Join(args, ", ")) return fmt.Sprintf("%s(%s)", glslExpr(&e.Exprs[0]), strings.Join(args, ", "))
case FieldSelector: case shaderir.FieldSelector:
return fmt.Sprintf("(%s).%s", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1])) return fmt.Sprintf("(%s).%s", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1]))
case Index: case shaderir.Index:
return fmt.Sprintf("(%s)[%s]", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1])) return fmt.Sprintf("(%s)[%s]", glslExpr(&e.Exprs[0]), glslExpr(&e.Exprs[1]))
default: default:
return fmt.Sprintf("?(unexpected expr: %d)", e.Type) return fmt.Sprintf("?(unexpected expr: %d)", e.Type)
@ -428,33 +436,33 @@ func (p *Program) glslBlock(topBlock, block *Block, level int, localVarIndex int
for _, s := range block.Stmts { for _, s := range block.Stmts {
switch s.Type { switch s.Type {
case ExprStmt: case shaderir.ExprStmt:
lines = append(lines, fmt.Sprintf("%s%s;", idt, glslExpr(&s.Exprs[0]))) lines = append(lines, fmt.Sprintf("%s%s;", idt, glslExpr(&s.Exprs[0])))
case BlockStmt: case shaderir.BlockStmt:
lines = append(lines, idt+"{") lines = append(lines, idt+"{")
lines = append(lines, p.glslBlock(topBlock, &s.Blocks[0], level+1, localVarIndex)...) lines = append(lines, c.glslBlock(p, topBlock, &s.Blocks[0], level+1, localVarIndex)...)
lines = append(lines, idt+"}") lines = append(lines, idt+"}")
case Assign: case shaderir.Assign:
// TODO: Give an appropriate context // TODO: Give an appropriate context
lines = append(lines, fmt.Sprintf("%s%s = %s;", idt, glslExpr(&s.Exprs[0]), glslExpr(&s.Exprs[1]))) lines = append(lines, fmt.Sprintf("%s%s = %s;", idt, glslExpr(&s.Exprs[0]), glslExpr(&s.Exprs[1])))
case If: case shaderir.If:
lines = append(lines, fmt.Sprintf("%sif (%s) {", idt, glslExpr(&s.Exprs[0]))) lines = append(lines, fmt.Sprintf("%sif (%s) {", idt, glslExpr(&s.Exprs[0])))
lines = append(lines, p.glslBlock(topBlock, &s.Blocks[0], level+1, localVarIndex)...) lines = append(lines, c.glslBlock(p, topBlock, &s.Blocks[0], level+1, localVarIndex)...)
if len(s.Blocks) > 1 { if len(s.Blocks) > 1 {
lines = append(lines, fmt.Sprintf("%s} else {", idt)) lines = append(lines, fmt.Sprintf("%s} else {", idt))
lines = append(lines, p.glslBlock(topBlock, &s.Blocks[1], level+1, localVarIndex)...) lines = append(lines, c.glslBlock(p, topBlock, &s.Blocks[1], level+1, localVarIndex)...)
} }
lines = append(lines, fmt.Sprintf("%s}", idt)) lines = append(lines, fmt.Sprintf("%s}", idt))
case For: case shaderir.For:
var ct ConstType var ct shaderir.ConstType
switch s.ForVarType.Main { switch s.ForVarType.Main {
case Int: case shaderir.Int:
ct = ConstTypeInt ct = shaderir.ConstTypeInt
case Float: case shaderir.Float:
ct = ConstTypeFloat ct = shaderir.ConstTypeFloat
} }
v := p.localVariableName(topBlock, s.ForVarIndex) v := localVariableName(p, topBlock, s.ForVarIndex)
var delta string var delta string
switch val, _ := constant.Float64Val(s.ForDelta); val { switch val, _ := constant.Float64Val(s.ForDelta); val {
case 0: case 0:
@ -474,7 +482,7 @@ func (p *Program) glslBlock(topBlock, block *Block, level int, localVarIndex int
} }
var op string var op string
switch s.ForOp { switch s.ForOp {
case LessThanOp, LessThanEqualOp, GreaterThanOp, GreaterThanEqualOp, EqualOp, NotEqualOp: case shaderir.LessThanOp, shaderir.LessThanEqualOp, shaderir.GreaterThanOp, shaderir.GreaterThanEqualOp, shaderir.EqualOp, shaderir.NotEqualOp:
op = string(s.ForOp) op = string(s.ForOp)
default: default:
op = fmt.Sprintf("?(unexpected op: %s)", string(s.ForOp)) op = fmt.Sprintf("?(unexpected op: %s)", string(s.ForOp))
@ -483,22 +491,22 @@ func (p *Program) glslBlock(topBlock, block *Block, level int, localVarIndex int
t := s.ForVarType t := s.ForVarType
init := constantToNumberLiteral(ct, s.ForInit) init := constantToNumberLiteral(ct, s.ForInit)
end := constantToNumberLiteral(ct, s.ForEnd) end := constantToNumberLiteral(ct, s.ForEnd)
t0, t1 := t.Glsl() t0, t1 := typeString(&t)
lines = append(lines, fmt.Sprintf("%sfor (%s %s%s = %s; %s %s %s; %s) {", idt, t0, v, t1, init, v, op, end, delta)) lines = append(lines, fmt.Sprintf("%sfor (%s %s%s = %s; %s %s %s; %s) {", idt, t0, v, t1, init, v, op, end, delta))
lines = append(lines, p.glslBlock(topBlock, &s.Blocks[0], level+1, localVarIndex)...) lines = append(lines, c.glslBlock(p, topBlock, &s.Blocks[0], level+1, localVarIndex)...)
lines = append(lines, fmt.Sprintf("%s}", idt)) lines = append(lines, fmt.Sprintf("%s}", idt))
case Continue: case shaderir.Continue:
lines = append(lines, idt+"continue;") lines = append(lines, idt+"continue;")
case Break: case shaderir.Break:
lines = append(lines, idt+"break;") lines = append(lines, idt+"break;")
case Return: case shaderir.Return:
if len(s.Exprs) == 0 { if len(s.Exprs) == 0 {
lines = append(lines, idt+"return;") lines = append(lines, idt+"return;")
} else { } else {
// TODO: Give an appropriate context. // TODO: Give an appropriate context.
lines = append(lines, fmt.Sprintf("%sreturn %s;", idt, glslExpr(&s.Exprs[0]))) lines = append(lines, fmt.Sprintf("%sreturn %s;", idt, glslExpr(&s.Exprs[0])))
} }
case Discard: case shaderir.Discard:
lines = append(lines, idt+"discard;") lines = append(lines, idt+"discard;")
default: default:
lines = append(lines, fmt.Sprintf("%s?(unexpected stmt: %d)", idt, s.Type)) lines = append(lines, fmt.Sprintf("%s?(unexpected stmt: %d)", idt, s.Type))

View File

@ -0,0 +1,85 @@
// 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 glsl
import (
"fmt"
"github.com/hajimehoshi/ebiten/internal/shaderir"
)
func typeString(t *shaderir.Type) (string, string) {
switch t.Main {
case shaderir.Array:
t0, t1 := typeString(&t.Sub[0])
return t0 + t1, fmt.Sprintf("[%d]", t.Length)
case shaderir.Struct:
panic("shaderir: a struct is not implemented")
default:
return basicTypeString(t.Main), ""
}
}
func basicTypeString(t shaderir.BasicType) string {
switch t {
case shaderir.None:
return "?(none)"
case shaderir.Bool:
return "bool"
case shaderir.Int:
return "int"
case shaderir.Float:
return "float"
case shaderir.Vec2:
return "vec2"
case shaderir.Vec3:
return "vec3"
case shaderir.Vec4:
return "vec4"
case shaderir.Mat2:
return "mat2"
case shaderir.Mat3:
return "mat3"
case shaderir.Mat4:
return "mat4"
case shaderir.Array:
return "?(array)"
case shaderir.Struct:
return "?(struct)"
default:
return fmt.Sprintf("?(unknown type: %d)", t)
}
}
func builtinFuncString(f shaderir.BuiltinFunc) string {
switch f {
case shaderir.LessThan:
return "lessThan"
case shaderir.LessThanEqual:
return "lessThanEqual"
case shaderir.GreaterThan:
return "greaterThan"
case shaderir.GreaterThanEqual:
return "greaterThanEqual"
case shaderir.NotEqual:
return "notEqual"
case shaderir.Dfdx:
return "dFdx"
case shaderir.Dfdy:
return "dFdy"
default:
return string(f)
}
}

View File

@ -19,6 +19,7 @@ import (
"testing" "testing"
. "github.com/hajimehoshi/ebiten/internal/shaderir" . "github.com/hajimehoshi/ebiten/internal/shaderir"
"github.com/hajimehoshi/ebiten/internal/shaderir/glsl"
) )
func block(localVars []Type, stmts ...Stmt) Block { func block(localVars []Type, stmts ...Stmt) Block {
@ -149,7 +150,7 @@ func fieldSelectorExpr(a, b Expr) Expr {
} }
func TestOutput(t *testing.T) { func TestOutput(t *testing.T) {
prelude := GlslFragmentPrelude + "\n" prelude := glsl.FragmentPrelude + "\n"
tests := []struct { tests := []struct {
Name string Name string
@ -161,7 +162,7 @@ func TestOutput(t *testing.T) {
Name: "Empty", Name: "Empty",
Program: Program{}, Program: Program{},
GlslVS: ``, GlslVS: ``,
GlslFS: GlslFragmentPrelude, GlslFS: glsl.FragmentPrelude,
}, },
{ {
Name: "Uniform", Name: "Uniform",
@ -821,7 +822,7 @@ void main(void) {
for _, tc := range tests { for _, tc := range tests {
tc := tc tc := tc
t.Run(tc.Name, func(t *testing.T) { t.Run(tc.Name, func(t *testing.T) {
vs, fs := tc.Program.Glsl() vs, fs := glsl.Compile(&tc.Program)
{ {
got := vs got := vs
want := tc.GlslVS + "\n" want := tc.GlslVS + "\n"

View File

@ -28,9 +28,6 @@ type Program struct {
Funcs []Func Funcs []Func
VertexFunc VertexFunc VertexFunc VertexFunc
FragmentFunc FragmentFunc FragmentFunc FragmentFunc
structNames map[string]string
structTypes []Type
} }
type Func struct { type Func struct {
@ -315,24 +312,3 @@ func ParseBuiltinFunc(str string) (BuiltinFunc, bool) {
} }
return "", false return "", false
} }
func (f BuiltinFunc) Glsl() string {
switch f {
case LessThan:
return "lessThan"
case LessThanEqual:
return "lessThanEqual"
case GreaterThan:
return "greaterThan"
case GreaterThanEqual:
return "greaterThanEqual"
case NotEqual:
return "notEqual"
case Dfdx:
return "dFdx"
case Dfdy:
return "dFdy"
default:
return string(f)
}
}

View File

@ -81,22 +81,6 @@ func (t *Type) String() string {
} }
} }
func (t *Type) serialize() string {
return t.String()
}
func (t *Type) Glsl() (string, string) {
switch t.Main {
case Array:
t0, t1 := t.Sub[0].Glsl()
return t0 + t1, fmt.Sprintf("[%d]", t.Length)
case Struct:
panic("shaderir: a struct is not implemented")
default:
return t.Main.glsl(), ""
}
}
func (t *Type) FloatNum() int { func (t *Type) FloatNum() int {
switch t.Main { switch t.Main {
case Float: case Float:
@ -136,34 +120,3 @@ const (
Array Array
Struct Struct
) )
func (t BasicType) glsl() string {
switch t {
case None:
return "?(none)"
case Bool:
return "bool"
case Int:
return "int"
case Float:
return "float"
case Vec2:
return "vec2"
case Vec3:
return "vec3"
case Vec4:
return "vec4"
case Mat2:
return "mat2"
case Mat3:
return "mat3"
case Mat4:
return "mat4"
case Array:
return "?(array)"
case Struct:
return "?(struct)"
default:
return fmt.Sprintf("?(unknown type: %d)", t)
}
}