mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-23 17:32:02 +01:00
49582519c1
This change adds a new compiler directive 'kage:unit' to Kage. This takes one of these two values: 'pixel' and 'texel'. The default value is 'texel'. With the pixel-unit mode, all the built-in functions treats pixels instead of texels, and the texCoord argument of Fragment is in pixels. This simplifies shader programs as programs no longer have the notion of texels. With the texel-unit mode, the behavior is the same as the current behavior. Closes #1431
941 lines
22 KiB
Go
941 lines
22 KiB
Go
// 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 shader
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"go/ast"
|
|
gconstant "go/constant"
|
|
"go/parser"
|
|
"go/token"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/hajimehoshi/ebiten/v2/internal/shaderir"
|
|
)
|
|
|
|
type variable struct {
|
|
name string
|
|
typ shaderir.Type
|
|
forLoopCounter bool
|
|
}
|
|
|
|
type constant struct {
|
|
name string
|
|
typ shaderir.Type
|
|
ctyp shaderir.ConstType
|
|
value gconstant.Value
|
|
}
|
|
|
|
type function struct {
|
|
name string
|
|
|
|
ir shaderir.Func
|
|
}
|
|
|
|
type compileState struct {
|
|
fs *token.FileSet
|
|
|
|
vertexEntry string
|
|
fragmentEntry string
|
|
unit shaderir.Unit
|
|
|
|
ir shaderir.Program
|
|
|
|
funcs []function
|
|
|
|
global block
|
|
|
|
varyingParsed bool
|
|
|
|
errs []string
|
|
}
|
|
|
|
func (cs *compileState) findFunction(name string) (int, bool) {
|
|
for i, f := range cs.funcs {
|
|
if f.name == name {
|
|
return i, true
|
|
}
|
|
}
|
|
return 0, false
|
|
}
|
|
|
|
func (cs *compileState) findUniformVariable(name string) (int, bool) {
|
|
for i, u := range cs.ir.UniformNames {
|
|
if u == name {
|
|
return i, true
|
|
}
|
|
}
|
|
return 0, false
|
|
}
|
|
|
|
type typ struct {
|
|
name string
|
|
ir shaderir.Type
|
|
}
|
|
|
|
type block struct {
|
|
types []typ
|
|
vars []variable
|
|
unusedVars map[int]token.Pos
|
|
consts []constant
|
|
pos token.Pos
|
|
outer *block
|
|
|
|
ir *shaderir.Block
|
|
}
|
|
|
|
func (b *block) totalLocalVariableCount() int {
|
|
c := len(b.vars)
|
|
if b.outer != nil {
|
|
c += b.outer.totalLocalVariableCount()
|
|
}
|
|
return c
|
|
}
|
|
|
|
func (b *block) addNamedLocalVariable(name string, typ shaderir.Type, pos token.Pos) {
|
|
b.vars = append(b.vars, variable{
|
|
name: name,
|
|
typ: typ,
|
|
})
|
|
if name == "_" {
|
|
return
|
|
}
|
|
idx := len(b.vars) - 1
|
|
if b.unusedVars == nil {
|
|
b.unusedVars = map[int]token.Pos{}
|
|
}
|
|
b.unusedVars[idx] = pos
|
|
}
|
|
|
|
func (b *block) findLocalVariable(name string, markLocalVariableUsed bool) (int, shaderir.Type, bool) {
|
|
if name == "" || name == "_" {
|
|
panic("shader: variable name must be non-empty and non-underscore")
|
|
}
|
|
|
|
idx := 0
|
|
for outer := b.outer; outer != nil; outer = outer.outer {
|
|
idx += len(outer.vars)
|
|
}
|
|
for i, v := range b.vars {
|
|
if v.name == name {
|
|
if markLocalVariableUsed {
|
|
delete(b.unusedVars, i)
|
|
}
|
|
return idx + i, v.typ, true
|
|
}
|
|
}
|
|
if b.outer != nil {
|
|
return b.outer.findLocalVariable(name, markLocalVariableUsed)
|
|
}
|
|
return 0, shaderir.Type{}, false
|
|
}
|
|
|
|
func (b *block) findLocalVariableByIndex(idx int) (shaderir.Type, bool) {
|
|
bs := []*block{b}
|
|
for outer := b.outer; outer != nil; outer = outer.outer {
|
|
bs = append(bs, outer)
|
|
}
|
|
for i := len(bs) - 1; i >= 0; i-- {
|
|
if len(bs[i].vars) <= idx {
|
|
idx -= len(bs[i].vars)
|
|
continue
|
|
}
|
|
return bs[i].vars[idx].typ, true
|
|
}
|
|
return shaderir.Type{}, false
|
|
}
|
|
|
|
func (b *block) findConstant(name string) (constant, bool) {
|
|
if name == "" || name == "_" {
|
|
panic("shader: constant name must be non-empty and non-underscore")
|
|
}
|
|
|
|
for _, c := range b.consts {
|
|
if c.name == name {
|
|
return c, true
|
|
}
|
|
}
|
|
if b.outer != nil {
|
|
return b.outer.findConstant(name)
|
|
}
|
|
|
|
return constant{}, false
|
|
}
|
|
|
|
type ParseError struct {
|
|
errs []string
|
|
}
|
|
|
|
func (p *ParseError) Error() string {
|
|
return strings.Join(p.errs, "\n")
|
|
}
|
|
|
|
func Compile(src []byte, vertexEntry, fragmentEntry string, textureCount int) (*shaderir.Program, error) {
|
|
unit, err := ParseCompilerDirectives(src)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
fs := token.NewFileSet()
|
|
f, err := parser.ParseFile(fs, "", src, parser.AllErrors)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
s := &compileState{
|
|
fs: fs,
|
|
vertexEntry: vertexEntry,
|
|
fragmentEntry: fragmentEntry,
|
|
unit: unit,
|
|
}
|
|
s.global.ir = &shaderir.Block{}
|
|
s.parse(f)
|
|
|
|
if len(s.errs) > 0 {
|
|
return nil, &ParseError{s.errs}
|
|
}
|
|
|
|
// TODO: Resolve identifiers?
|
|
// TODO: Resolve constants
|
|
|
|
// TODO: Make a call graph and reorder the elements.
|
|
|
|
s.ir.TextureCount = textureCount
|
|
return &s.ir, nil
|
|
}
|
|
|
|
func ParseCompilerDirectives(src []byte) (shaderir.Unit, error) {
|
|
// TODO: Change the unit to pixels in v3 (#2645).
|
|
unit := shaderir.Texel
|
|
|
|
reUnit := regexp.MustCompile(`^//kage:unit\s+(.+)$`)
|
|
var unitParsed bool
|
|
|
|
buf := bytes.NewBuffer(src)
|
|
s := bufio.NewScanner(buf)
|
|
for s.Scan() {
|
|
m := reUnit.FindStringSubmatch(s.Text())
|
|
if m == nil {
|
|
continue
|
|
}
|
|
if unitParsed {
|
|
return 0, fmt.Errorf("shader: at most one //kage:unit can exist in a shader")
|
|
}
|
|
switch m[1] {
|
|
case "pixel":
|
|
unit = shaderir.Pixel
|
|
case "texel":
|
|
unit = shaderir.Texel
|
|
default:
|
|
return 0, fmt.Errorf("shader: invalid value for //kage:unit: %s", m[1])
|
|
}
|
|
unitParsed = true
|
|
}
|
|
|
|
return unit, nil
|
|
}
|
|
|
|
func (s *compileState) addError(pos token.Pos, str string) {
|
|
p := s.fs.Position(pos)
|
|
s.errs = append(s.errs, fmt.Sprintf("%s: %s", p, str))
|
|
}
|
|
|
|
func (cs *compileState) parse(f *ast.File) {
|
|
cs.ir.Unit = cs.unit
|
|
|
|
// Parse GenDecl for global variables, and then parse functions.
|
|
for _, d := range f.Decls {
|
|
if _, ok := d.(*ast.FuncDecl); !ok {
|
|
ss, ok := cs.parseDecl(&cs.global, "", d)
|
|
if !ok {
|
|
return
|
|
}
|
|
cs.global.ir.Stmts = append(cs.global.ir.Stmts, ss...)
|
|
}
|
|
}
|
|
|
|
// Sort the uniform variable so that special variable starting with __ should come first.
|
|
var unames []string
|
|
var utypes []shaderir.Type
|
|
for i, u := range cs.ir.UniformNames {
|
|
if strings.HasPrefix(u, "__") {
|
|
unames = append(unames, u)
|
|
utypes = append(utypes, cs.ir.Uniforms[i])
|
|
}
|
|
}
|
|
// TODO: Check len(unames) == graphics.PreservedUniformVariablesNum. Unfortunately this is not true on tests.
|
|
for i, u := range cs.ir.UniformNames {
|
|
if !strings.HasPrefix(u, "__") {
|
|
unames = append(unames, u)
|
|
utypes = append(utypes, cs.ir.Uniforms[i])
|
|
}
|
|
}
|
|
cs.ir.UniformNames = unames
|
|
cs.ir.Uniforms = utypes
|
|
|
|
// Parse function names so that any other function call the others.
|
|
// The function data is provisional and will be updated soon.
|
|
for _, d := range f.Decls {
|
|
fd, ok := d.(*ast.FuncDecl)
|
|
if !ok {
|
|
continue
|
|
}
|
|
n := fd.Name.Name
|
|
if n == cs.vertexEntry {
|
|
continue
|
|
}
|
|
if n == cs.fragmentEntry {
|
|
continue
|
|
}
|
|
|
|
for _, f := range cs.funcs {
|
|
if f.name == n {
|
|
cs.addError(d.Pos(), fmt.Sprintf("redeclared function: %s", n))
|
|
return
|
|
}
|
|
}
|
|
|
|
inParams, outParams, ret := cs.parseFuncParams(&cs.global, n, fd)
|
|
var inT, outT []shaderir.Type
|
|
for _, v := range inParams {
|
|
inT = append(inT, v.typ)
|
|
}
|
|
for _, v := range outParams {
|
|
outT = append(outT, v.typ)
|
|
}
|
|
|
|
cs.funcs = append(cs.funcs, function{
|
|
name: n,
|
|
ir: shaderir.Func{
|
|
Index: len(cs.funcs),
|
|
InParams: inT,
|
|
OutParams: outT,
|
|
Return: ret,
|
|
Block: &shaderir.Block{},
|
|
},
|
|
})
|
|
}
|
|
|
|
// Parse functions.
|
|
for _, d := range f.Decls {
|
|
if f, ok := d.(*ast.FuncDecl); ok {
|
|
ss, ok := cs.parseDecl(&cs.global, f.Name.Name, d)
|
|
if !ok {
|
|
return
|
|
}
|
|
cs.global.ir.Stmts = append(cs.global.ir.Stmts, ss...)
|
|
}
|
|
}
|
|
|
|
if len(cs.errs) > 0 {
|
|
return
|
|
}
|
|
|
|
for _, f := range cs.funcs {
|
|
cs.ir.Funcs = append(cs.ir.Funcs, f.ir)
|
|
}
|
|
}
|
|
|
|
func (cs *compileState) parseDecl(b *block, fname string, d ast.Decl) ([]shaderir.Stmt, bool) {
|
|
var stmts []shaderir.Stmt
|
|
|
|
switch d := d.(type) {
|
|
case *ast.GenDecl:
|
|
switch d.Tok {
|
|
case token.TYPE:
|
|
// TODO: Parse other types
|
|
for _, s := range d.Specs {
|
|
s := s.(*ast.TypeSpec)
|
|
t, ok := cs.parseType(b, fname, s.Type)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
n := s.Name.Name
|
|
for _, t := range b.types {
|
|
if t.name == n {
|
|
cs.addError(s.Pos(), fmt.Sprintf("%s redeclared in this block", n))
|
|
return nil, false
|
|
}
|
|
}
|
|
b.types = append(b.types, typ{
|
|
name: n,
|
|
ir: t,
|
|
})
|
|
}
|
|
case token.CONST:
|
|
for _, s := range d.Specs {
|
|
s := s.(*ast.ValueSpec)
|
|
cs, ok := cs.parseConstant(b, fname, s)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
b.consts = append(b.consts, cs...)
|
|
}
|
|
case token.VAR:
|
|
for _, s := range d.Specs {
|
|
s := s.(*ast.ValueSpec)
|
|
vs, inits, ss, ok := cs.parseVariable(b, fname, s)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
|
|
stmts = append(stmts, ss...)
|
|
if b == &cs.global {
|
|
// TODO: Should rhs be ignored?
|
|
for i, v := range vs {
|
|
if !strings.HasPrefix(v.name, "__") {
|
|
if v.name[0] < 'A' || 'Z' < v.name[0] {
|
|
cs.addError(s.Names[i].Pos(), fmt.Sprintf("global variables must be exposed: %s", v.name))
|
|
}
|
|
}
|
|
cs.ir.UniformNames = append(cs.ir.UniformNames, v.name)
|
|
cs.ir.Uniforms = append(cs.ir.Uniforms, v.typ)
|
|
}
|
|
continue
|
|
}
|
|
|
|
// base must be obtained before adding the variables.
|
|
base := b.totalLocalVariableCount()
|
|
for _, v := range vs {
|
|
b.addNamedLocalVariable(v.name, v.typ, d.Pos())
|
|
}
|
|
|
|
if len(inits) > 0 {
|
|
for i := range vs {
|
|
stmts = append(stmts, shaderir.Stmt{
|
|
Type: shaderir.Assign,
|
|
Exprs: []shaderir.Expr{
|
|
{
|
|
Type: shaderir.LocalVariable,
|
|
Index: base + i,
|
|
},
|
|
inits[i],
|
|
},
|
|
})
|
|
}
|
|
}
|
|
}
|
|
case token.IMPORT:
|
|
cs.addError(d.Pos(), "import is forbidden")
|
|
default:
|
|
cs.addError(d.Pos(), "unexpected token")
|
|
}
|
|
case *ast.FuncDecl:
|
|
f, ok := cs.parseFunc(b, d)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
if b != &cs.global {
|
|
cs.addError(d.Pos(), "non-global function is not implemented")
|
|
return nil, false
|
|
}
|
|
switch d.Name.Name {
|
|
case cs.vertexEntry:
|
|
cs.ir.VertexFunc.Block = f.ir.Block
|
|
case cs.fragmentEntry:
|
|
cs.ir.FragmentFunc.Block = f.ir.Block
|
|
default:
|
|
// The function is already registered for their names.
|
|
for i := range cs.funcs {
|
|
if cs.funcs[i].name == d.Name.Name {
|
|
// Index is already determined by the provisional parsing.
|
|
f.ir.Index = cs.funcs[i].ir.Index
|
|
cs.funcs[i] = f
|
|
break
|
|
}
|
|
}
|
|
}
|
|
default:
|
|
cs.addError(d.Pos(), "unexpected decl")
|
|
return nil, false
|
|
}
|
|
|
|
return stmts, true
|
|
}
|
|
|
|
// functionReturnTypes returns the original returning value types, if the given expression is call.
|
|
//
|
|
// Note that parseExpr returns the returning types for IR, not the original function.
|
|
func (cs *compileState) functionReturnTypes(block *block, expr ast.Expr) ([]shaderir.Type, bool) {
|
|
call, ok := expr.(*ast.CallExpr)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
|
|
ident, ok := call.Fun.(*ast.Ident)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
|
|
for _, f := range cs.funcs {
|
|
if f.name == ident.Name {
|
|
// TODO: Is it correct to combine out-params and return param?
|
|
ts := f.ir.OutParams
|
|
if f.ir.Return.Main != shaderir.None {
|
|
ts = append(ts, f.ir.Return)
|
|
}
|
|
return ts, true
|
|
}
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
func (s *compileState) parseVariable(block *block, fname string, vs *ast.ValueSpec) ([]variable, []shaderir.Expr, []shaderir.Stmt, bool) {
|
|
if len(vs.Names) != len(vs.Values) && len(vs.Values) != 1 && len(vs.Values) != 0 {
|
|
s.addError(vs.Pos(), fmt.Sprintf("the numbers of lhs and rhs don't match"))
|
|
return nil, nil, nil, false
|
|
}
|
|
|
|
var declt shaderir.Type
|
|
if vs.Type != nil {
|
|
var ok bool
|
|
declt, ok = s.parseType(block, fname, vs.Type)
|
|
if !ok {
|
|
return nil, nil, nil, false
|
|
}
|
|
}
|
|
|
|
var (
|
|
vars []variable
|
|
inits []shaderir.Expr
|
|
stmts []shaderir.Stmt
|
|
)
|
|
|
|
// These variables are used only in multiple-value context.
|
|
var inittypes []shaderir.Type
|
|
var initexprs []shaderir.Expr
|
|
|
|
for i, n := range vs.Names {
|
|
t := declt
|
|
switch {
|
|
case len(vs.Values) == 0:
|
|
// No initialization
|
|
|
|
case len(vs.Names) == len(vs.Values):
|
|
// Single-value context
|
|
|
|
init := vs.Values[i]
|
|
|
|
es, rts, ss, ok := s.parseExpr(block, fname, init, true)
|
|
if !ok {
|
|
return nil, nil, nil, false
|
|
}
|
|
|
|
if t.Main == shaderir.None {
|
|
ts, ok := s.functionReturnTypes(block, init)
|
|
if !ok {
|
|
ts = rts
|
|
}
|
|
if len(ts) > 1 {
|
|
s.addError(vs.Pos(), fmt.Sprintf("the numbers of lhs and rhs don't match"))
|
|
}
|
|
t = ts[0]
|
|
}
|
|
|
|
if es[0].Type == shaderir.NumberExpr {
|
|
switch t.Main {
|
|
case shaderir.Int:
|
|
es[0].ConstType = shaderir.ConstTypeInt
|
|
case shaderir.Float:
|
|
es[0].ConstType = shaderir.ConstTypeFloat
|
|
}
|
|
}
|
|
|
|
for i, rt := range rts {
|
|
if !canAssign(&t, &rt, es[i].Const) {
|
|
s.addError(vs.Pos(), fmt.Sprintf("cannot use type %s as type %s in variable declaration", rt.String(), t.String()))
|
|
}
|
|
}
|
|
|
|
inits = append(inits, es...)
|
|
stmts = append(stmts, ss...)
|
|
|
|
default:
|
|
// Multiple-value context
|
|
|
|
if i == 0 {
|
|
init := vs.Values[0]
|
|
|
|
var ss []shaderir.Stmt
|
|
var ok bool
|
|
initexprs, inittypes, ss, ok = s.parseExpr(block, fname, init, true)
|
|
if !ok {
|
|
return nil, nil, nil, false
|
|
}
|
|
stmts = append(stmts, ss...)
|
|
|
|
if t.Main == shaderir.None {
|
|
ts, ok := s.functionReturnTypes(block, init)
|
|
if ok {
|
|
inittypes = ts
|
|
}
|
|
if len(ts) != len(vs.Names) {
|
|
s.addError(vs.Pos(), fmt.Sprintf("the numbers of lhs and rhs don't match"))
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
if t.Main == shaderir.None && len(inittypes) > 0 {
|
|
t = inittypes[i]
|
|
}
|
|
|
|
if !canAssign(&t, &inittypes[i], initexprs[i].Const) {
|
|
s.addError(vs.Pos(), fmt.Sprintf("cannot use type %s as type %s in variable declaration", inittypes[i].String(), t.String()))
|
|
}
|
|
|
|
// Add the same initexprs for each variable.
|
|
inits = append(inits, initexprs...)
|
|
}
|
|
|
|
name := n.Name
|
|
for _, v := range append(block.vars, vars...) {
|
|
if v.name == name {
|
|
s.addError(vs.Pos(), fmt.Sprintf("duplicated local variable name: %s", name))
|
|
return nil, nil, nil, false
|
|
}
|
|
}
|
|
for _, c := range block.consts {
|
|
if c.name == name {
|
|
s.addError(vs.Pos(), fmt.Sprintf("duplicated local constant/variable name: %s", name))
|
|
return nil, nil, nil, false
|
|
}
|
|
}
|
|
vars = append(vars, variable{
|
|
name: name,
|
|
typ: t,
|
|
})
|
|
}
|
|
|
|
return vars, inits, stmts, true
|
|
}
|
|
|
|
func (s *compileState) parseConstant(block *block, fname string, vs *ast.ValueSpec) ([]constant, bool) {
|
|
var t shaderir.Type
|
|
if vs.Type != nil {
|
|
var ok bool
|
|
t, ok = s.parseType(block, fname, vs.Type)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
}
|
|
|
|
var cs []constant
|
|
for i, n := range vs.Names {
|
|
name := n.Name
|
|
for _, c := range block.consts {
|
|
if c.name == name {
|
|
s.addError(vs.Pos(), fmt.Sprintf("duplicated local constant name: %s", name))
|
|
return nil, false
|
|
}
|
|
}
|
|
for _, v := range block.vars {
|
|
if v.name == name {
|
|
s.addError(vs.Pos(), fmt.Sprintf("duplicated local constant/variable name: %s", name))
|
|
return nil, false
|
|
}
|
|
}
|
|
|
|
es, ts, ss, ok := s.parseExpr(block, fname, vs.Values[i], false)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
if len(ss) > 0 {
|
|
s.addError(vs.Pos(), fmt.Sprintf("invalid constant expression: %s", name))
|
|
return nil, false
|
|
}
|
|
if len(ts) != 1 || len(es) != 1 {
|
|
s.addError(vs.Pos(), fmt.Sprintf("invalid constant expression: %s", n))
|
|
return nil, false
|
|
}
|
|
if es[0].Type != shaderir.NumberExpr {
|
|
s.addError(vs.Pos(), fmt.Sprintf("constant expression must be a number but not: %s", n))
|
|
return nil, false
|
|
}
|
|
|
|
if !t.Equal(&shaderir.Type{}) && !canAssign(&t, &ts[0], es[0].Const) {
|
|
s.addError(vs.Pos(), fmt.Sprintf("cannot use %v as %s value in constant declaration", es[0].Const, t.String()))
|
|
return nil, false
|
|
}
|
|
|
|
c := es[0].Const
|
|
constType := es[0].ConstType
|
|
switch t.Main {
|
|
case shaderir.Bool:
|
|
constType = shaderir.ConstTypeBool
|
|
case shaderir.Int:
|
|
constType = shaderir.ConstTypeInt
|
|
c = gconstant.ToInt(c)
|
|
case shaderir.Float:
|
|
constType = shaderir.ConstTypeFloat
|
|
c = gconstant.ToFloat(c)
|
|
}
|
|
|
|
cs = append(cs, constant{
|
|
name: name,
|
|
typ: t,
|
|
ctyp: constType,
|
|
value: c,
|
|
})
|
|
}
|
|
return cs, true
|
|
}
|
|
|
|
func (cs *compileState) parseFuncParams(block *block, fname string, d *ast.FuncDecl) (in, out []variable, ret shaderir.Type) {
|
|
for _, f := range d.Type.Params.List {
|
|
t, ok := cs.parseType(block, fname, f.Type)
|
|
if !ok {
|
|
return
|
|
}
|
|
for _, n := range f.Names {
|
|
in = append(in, variable{
|
|
name: n.Name,
|
|
typ: t,
|
|
})
|
|
}
|
|
}
|
|
|
|
if d.Type.Results == nil {
|
|
return
|
|
}
|
|
|
|
for _, f := range d.Type.Results.List {
|
|
t, ok := cs.parseType(block, fname, f.Type)
|
|
if !ok {
|
|
return
|
|
}
|
|
if len(f.Names) == 0 {
|
|
out = append(out, variable{
|
|
name: "",
|
|
typ: t,
|
|
})
|
|
} else {
|
|
for _, n := range f.Names {
|
|
out = append(out, variable{
|
|
name: n.Name,
|
|
typ: t,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(out) == 1 && out[0].name == "" {
|
|
ret = out[0].typ
|
|
out = nil
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func (cs *compileState) parseFunc(block *block, d *ast.FuncDecl) (function, bool) {
|
|
if d.Name == nil {
|
|
cs.addError(d.Pos(), "function must have a name")
|
|
return function{}, false
|
|
}
|
|
if d.Name.Name == "init" {
|
|
cs.addError(d.Pos(), "init function is not implemented")
|
|
return function{}, false
|
|
}
|
|
if d.Body == nil {
|
|
cs.addError(d.Pos(), "function must have a body")
|
|
return function{}, false
|
|
}
|
|
|
|
inParams, outParams, returnType := cs.parseFuncParams(block, d.Name.Name, d)
|
|
|
|
checkVaryings := func(vs []variable) {
|
|
if len(cs.ir.Varyings) != len(vs) {
|
|
cs.addError(d.Pos(), fmt.Sprintf("the number of vertex entry point's returning values and the number of fragment entry point's params must be the same"))
|
|
return
|
|
}
|
|
for i, t := range cs.ir.Varyings {
|
|
if t.Main != vs[i].typ.Main {
|
|
cs.addError(d.Pos(), fmt.Sprintf("vertex entry point's returning value types and fragment entry point's param types must match"))
|
|
}
|
|
}
|
|
}
|
|
|
|
if block == &cs.global {
|
|
switch d.Name.Name {
|
|
case cs.vertexEntry:
|
|
for _, v := range inParams {
|
|
cs.ir.Attributes = append(cs.ir.Attributes, v.typ)
|
|
}
|
|
|
|
// For the vertex entry, a parameter (variable) is used as a returning value.
|
|
// For example, GLSL doesn't treat gl_Position as a returning value.
|
|
// TODO: This can be resolved by having an indirect function like what the fragment entry already does.
|
|
// See internal/shaderir/glsl.adjustProgram.
|
|
if len(outParams) == 0 {
|
|
outParams = append(outParams, variable{
|
|
typ: shaderir.Type{Main: shaderir.Vec4},
|
|
})
|
|
}
|
|
|
|
// The first out-param is treated as gl_Position in GLSL.
|
|
if outParams[0].typ.Main != shaderir.Vec4 {
|
|
cs.addError(d.Pos(), fmt.Sprintf("vertex entry point must have at least one returning vec4 value for a position"))
|
|
return function{}, false
|
|
}
|
|
|
|
if cs.varyingParsed {
|
|
checkVaryings(outParams[1:])
|
|
} else {
|
|
for _, v := range outParams[1:] {
|
|
// TODO: Check that these params are not arrays or structs
|
|
cs.ir.Varyings = append(cs.ir.Varyings, v.typ)
|
|
}
|
|
}
|
|
cs.varyingParsed = true
|
|
case cs.fragmentEntry:
|
|
if len(inParams) == 0 {
|
|
cs.addError(d.Pos(), fmt.Sprintf("fragment entry point must have at least one vec4 parameter for a position"))
|
|
return function{}, false
|
|
}
|
|
if inParams[0].typ.Main != shaderir.Vec4 {
|
|
cs.addError(d.Pos(), fmt.Sprintf("fragment entry point must have at least one vec4 parameter for a position"))
|
|
return function{}, false
|
|
}
|
|
|
|
if len(outParams) != 0 || returnType.Main != shaderir.Vec4 {
|
|
cs.addError(d.Pos(), fmt.Sprintf("fragment entry point must have one returning vec4 value for a color"))
|
|
return function{}, false
|
|
}
|
|
|
|
if cs.varyingParsed {
|
|
checkVaryings(inParams[1:])
|
|
} else {
|
|
for _, v := range inParams[1:] {
|
|
cs.ir.Varyings = append(cs.ir.Varyings, v.typ)
|
|
}
|
|
}
|
|
cs.varyingParsed = true
|
|
}
|
|
}
|
|
|
|
b, ok := cs.parseBlock(block, d.Name.Name, d.Body.List, inParams, outParams, returnType, true)
|
|
if !ok {
|
|
return function{}, false
|
|
}
|
|
|
|
if len(outParams) > 0 || returnType.Main != shaderir.None {
|
|
var hasReturn func(stmts []shaderir.Stmt) bool
|
|
hasReturn = func(stmts []shaderir.Stmt) bool {
|
|
for _, stmt := range stmts {
|
|
if stmt.Type == shaderir.Return {
|
|
return true
|
|
}
|
|
for _, b := range stmt.Blocks {
|
|
if hasReturn(b.Stmts) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
if !hasReturn(b.ir.Stmts) {
|
|
cs.addError(d.Pos(), fmt.Sprintf("function %s must have a return statement but not", d.Name))
|
|
return function{}, false
|
|
}
|
|
}
|
|
|
|
var inT, outT []shaderir.Type
|
|
for _, v := range inParams {
|
|
inT = append(inT, v.typ)
|
|
}
|
|
for _, v := range outParams {
|
|
outT = append(outT, v.typ)
|
|
}
|
|
|
|
return function{
|
|
name: d.Name.Name,
|
|
ir: shaderir.Func{
|
|
InParams: inT,
|
|
OutParams: outT,
|
|
Return: returnType,
|
|
Block: b.ir,
|
|
},
|
|
}, true
|
|
}
|
|
|
|
func (cs *compileState) parseBlock(outer *block, fname string, stmts []ast.Stmt, inParams, outParams []variable, returnType shaderir.Type, checkLocalVariableUsage bool) (*block, bool) {
|
|
var vars []variable
|
|
if outer == &cs.global {
|
|
vars = make([]variable, 0, len(inParams)+len(outParams))
|
|
vars = append(vars, inParams...)
|
|
vars = append(vars, outParams...)
|
|
}
|
|
|
|
var offset int
|
|
for b := outer; b != nil; b = b.outer {
|
|
offset += len(b.vars)
|
|
}
|
|
if outer == &cs.global {
|
|
offset += len(inParams) + len(outParams)
|
|
}
|
|
|
|
block := &block{
|
|
vars: vars,
|
|
outer: outer,
|
|
ir: &shaderir.Block{
|
|
LocalVarIndexOffset: offset,
|
|
},
|
|
}
|
|
|
|
defer func() {
|
|
var offset int
|
|
if outer == &cs.global {
|
|
offset = len(inParams) + len(outParams)
|
|
}
|
|
for _, v := range block.vars[offset:] {
|
|
if v.forLoopCounter {
|
|
block.ir.LocalVars = append(block.ir.LocalVars, shaderir.Type{})
|
|
continue
|
|
}
|
|
block.ir.LocalVars = append(block.ir.LocalVars, v.typ)
|
|
}
|
|
}()
|
|
|
|
if outer.outer == nil && len(outParams) > 0 && outParams[0].name != "" {
|
|
for i := range outParams {
|
|
block.ir.Stmts = append(block.ir.Stmts, shaderir.Stmt{
|
|
Type: shaderir.Init,
|
|
InitIndex: len(inParams) + i,
|
|
})
|
|
}
|
|
}
|
|
|
|
for _, stmt := range stmts {
|
|
ss, ok := cs.parseStmt(block, fname, stmt, inParams, outParams, returnType)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
block.ir.Stmts = append(block.ir.Stmts, ss...)
|
|
}
|
|
|
|
if checkLocalVariableUsage && len(block.unusedVars) > 0 {
|
|
for idx, pos := range block.unusedVars {
|
|
cs.addError(pos, fmt.Sprintf("local variable %s is not used", block.vars[idx].name))
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
return block, true
|
|
}
|