mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2025-01-26 10:42:42 +01:00
internal/shader: implement strict type checks in assignments
Closes #1972
This commit is contained in:
parent
93b4c0d9b1
commit
15fe7158fd
@ -336,6 +336,7 @@ func (cs *compileState) parseDecl(b *block, d ast.Decl) ([]shaderir.Stmt, bool)
|
|||||||
if !ok {
|
if !ok {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
stmts = append(stmts, ss...)
|
stmts = append(stmts, ss...)
|
||||||
if b == &cs.global {
|
if b == &cs.global {
|
||||||
// TODO: Should rhs be ignored?
|
// TODO: Should rhs be ignored?
|
||||||
@ -473,7 +474,7 @@ func (s *compileState) parseVariable(block *block, vs *ast.ValueSpec) ([]variabl
|
|||||||
|
|
||||||
init := vs.Values[i]
|
init := vs.Values[i]
|
||||||
|
|
||||||
es, origts, ss, ok := s.parseExpr(block, init, true)
|
es, rts, ss, ok := s.parseExpr(block, init, true)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, nil, nil, false
|
return nil, nil, nil, false
|
||||||
}
|
}
|
||||||
@ -481,7 +482,7 @@ func (s *compileState) parseVariable(block *block, vs *ast.ValueSpec) ([]variabl
|
|||||||
if t.Main == shaderir.None {
|
if t.Main == shaderir.None {
|
||||||
ts, ok := s.functionReturnTypes(block, init)
|
ts, ok := s.functionReturnTypes(block, init)
|
||||||
if !ok {
|
if !ok {
|
||||||
ts = origts
|
ts = rts
|
||||||
}
|
}
|
||||||
if len(ts) > 1 {
|
if len(ts) > 1 {
|
||||||
s.addError(vs.Pos(), fmt.Sprintf("the numbers of lhs and rhs don't match"))
|
s.addError(vs.Pos(), fmt.Sprintf("the numbers of lhs and rhs don't match"))
|
||||||
@ -498,6 +499,12 @@ func (s *compileState) parseVariable(block *block, vs *ast.ValueSpec) ([]variabl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for i, rt := range rts {
|
||||||
|
if !canAssign(&es[i], &t, &rt) {
|
||||||
|
s.addError(vs.Pos(), fmt.Sprintf("cannot use type %s as type %s in variable declaration", rt.String(), t.String()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inits = append(inits, es...)
|
inits = append(inits, es...)
|
||||||
stmts = append(stmts, ss...)
|
stmts = append(stmts, ss...)
|
||||||
|
|
||||||
@ -526,10 +533,15 @@ func (s *compileState) parseVariable(block *block, vs *ast.ValueSpec) ([]variabl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(inittypes) > 0 {
|
|
||||||
|
if t.Main == shaderir.None && len(inittypes) > 0 {
|
||||||
t = inittypes[i]
|
t = inittypes[i]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if !canAssign(&initexprs[i], &t, &inittypes[i]) {
|
||||||
|
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.
|
// Add the same initexprs for each variable.
|
||||||
inits = append(inits, initexprs...)
|
inits = append(inits, initexprs...)
|
||||||
}
|
}
|
||||||
|
@ -562,7 +562,7 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
for i, e := range lhs {
|
for i, e := range lhs {
|
||||||
if len(lhs) == len(rhs) {
|
if len(lhs) == len(rhs) {
|
||||||
// Prase RHS first for the order of the statements.
|
// Prase RHS first for the order of the statements.
|
||||||
r, origts, ss, ok := cs.parseExpr(block, rhs[i], true)
|
r, rts, ss, ok := cs.parseExpr(block, rhs[i], true)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
@ -580,7 +580,7 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
}
|
}
|
||||||
ts, ok := cs.functionReturnTypes(block, rhs[i])
|
ts, ok := cs.functionReturnTypes(block, rhs[i])
|
||||||
if !ok {
|
if !ok {
|
||||||
ts = origts
|
ts = rts
|
||||||
}
|
}
|
||||||
if len(ts) > 1 {
|
if len(ts) > 1 {
|
||||||
cs.addError(pos, fmt.Sprintf("single-value context and multiple-value context cannot be mixed"))
|
cs.addError(pos, fmt.Sprintf("single-value context and multiple-value context cannot be mixed"))
|
||||||
@ -599,7 +599,7 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
l, _, ss, ok := cs.parseExpr(block, lhs[i], false)
|
l, lts, ss, ok := cs.parseExpr(block, lhs[i], false)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
@ -646,6 +646,13 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for i := range lts {
|
||||||
|
if !canAssign(&r[i], <s[i], &rts[i]) {
|
||||||
|
cs.addError(pos, fmt.Sprintf("cannot use type %s as type %s in variable declaration", rts[i].String(), lts[i].String()))
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if len(lhs) == 1 {
|
if len(lhs) == 1 {
|
||||||
stmts = append(stmts, shaderir.Stmt{
|
stmts = append(stmts, shaderir.Stmt{
|
||||||
Type: shaderir.Assign,
|
Type: shaderir.Assign,
|
||||||
@ -653,7 +660,7 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
// For variable swapping, use temporary variables.
|
// For variable swapping, use temporary variables.
|
||||||
t := origts[0]
|
t := rts[0]
|
||||||
if t.Main == shaderir.None {
|
if t.Main == shaderir.None {
|
||||||
t = toDefaultType(r[0].Const)
|
t = toDefaultType(r[0].Const)
|
||||||
}
|
}
|
||||||
@ -716,7 +723,7 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
block.addNamedLocalVariable(name, t, e.Pos())
|
block.addNamedLocalVariable(name, t, e.Pos())
|
||||||
}
|
}
|
||||||
|
|
||||||
l, _, ss, ok := cs.parseExpr(block, lhs[i], false)
|
l, lts, ss, ok := cs.parseExpr(block, lhs[i], false)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
@ -727,6 +734,13 @@ func (cs *compileState) assign(block *block, fname string, pos token.Pos, lhs, r
|
|||||||
}
|
}
|
||||||
allblank = false
|
allblank = false
|
||||||
|
|
||||||
|
for i, lt := range lts {
|
||||||
|
if !canAssign(&rhsExprs[i], <, &rhsTypes[i]) {
|
||||||
|
cs.addError(pos, fmt.Sprintf("cannot use type %s as type %s in variable declaration", rhsTypes[i].String(), lt.String()))
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
stmts = append(stmts, shaderir.Stmt{
|
stmts = append(stmts, shaderir.Stmt{
|
||||||
Type: shaderir.Assign,
|
Type: shaderir.Assign,
|
||||||
Exprs: []shaderir.Expr{l[0], rhsExprs[i]},
|
Exprs: []shaderir.Expr{l[0], rhsExprs[i]},
|
||||||
@ -754,3 +768,13 @@ func toDefaultType(v gconstant.Value) shaderir.Type {
|
|||||||
// TODO: Should this be an error?
|
// TODO: Should this be an error?
|
||||||
return shaderir.Type{}
|
return shaderir.Type{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func canAssign(re *shaderir.Expr, lt *shaderir.Type, rt *shaderir.Type) bool {
|
||||||
|
if lt.Equal(rt) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if re.Type == shaderir.NumberExpr && (lt.Main == shaderir.Float || lt.Main == shaderir.Int || lt.Main == shaderir.Bool) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
@ -1183,3 +1183,127 @@ func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
|||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Issue #1972
|
||||||
|
func TestSyntaxType(t *testing.T) {
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x vec2 = vec3(0)
|
||||||
|
_ = x
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x, y vec2 = vec2(0), vec3(0)
|
||||||
|
_, _ = x, y
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x vec2
|
||||||
|
x = vec3(0)
|
||||||
|
_ = x
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x, y vec2
|
||||||
|
x, y = vec2(0), vec3(0)
|
||||||
|
_ = x
|
||||||
|
_ = y
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x vec2
|
||||||
|
x = 0
|
||||||
|
_ = x
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Foo() (vec3, vec3) {
|
||||||
|
return vec3(0), vec3(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x, y vec2 = Foo()
|
||||||
|
_ = x
|
||||||
|
_ = y
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Foo() (vec3, vec3) {
|
||||||
|
return vec3(0), vec3(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var x, y vec2
|
||||||
|
x, y = Foo()
|
||||||
|
_ = x
|
||||||
|
_ = y
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Issue #1972
|
||||||
|
func TestSyntaxTypeBlankVar(t *testing.T) {
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var _ vec2 = vec3(0)
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var _, _ vec2 = vec2(0), vec3(0)
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
if _, err := compileToIR([]byte(`package main
|
||||||
|
|
||||||
|
func Foo() (vec3, vec3) {
|
||||||
|
return vec3(0), vec3(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Fragment(position vec4, texCoord vec2, color vec4) vec4 {
|
||||||
|
var _, _ vec2 = Foo()
|
||||||
|
return color
|
||||||
|
}
|
||||||
|
`)); err == nil {
|
||||||
|
t.Errorf("error must be non-nil but was nil")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
2
internal/shader/testdata/array.expected.vs
vendored
2
internal/shader/testdata/array.expected.vs
vendored
@ -22,7 +22,7 @@ void F1(out vec2 l0[2]) {
|
|||||||
(l1)[0] = vec2(1.0);
|
(l1)[0] = vec2(1.0);
|
||||||
l2[0] = l1[0];
|
l2[0] = l1[0];
|
||||||
l2[1] = l1[1];
|
l2[1] = l1[1];
|
||||||
((l2)[1]).y = vec2(2.0);
|
(l2)[1] = vec2(2.0);
|
||||||
l0[0] = l2[0];
|
l0[0] = l2[0];
|
||||||
l0[1] = l2[1];
|
l0[1] = l2[1];
|
||||||
return;
|
return;
|
||||||
|
2
internal/shader/testdata/array.go
vendored
2
internal/shader/testdata/array.go
vendored
@ -9,6 +9,6 @@ func Foo() [2]vec2 {
|
|||||||
|
|
||||||
func Bar() [2]vec2 {
|
func Bar() [2]vec2 {
|
||||||
x := [2]vec2{vec2(1)}
|
x := [2]vec2{vec2(1)}
|
||||||
x[1].y = vec2(2)
|
x[1] = vec2(2)
|
||||||
return x
|
return x
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user