tweak expr.typecastTo() a bit

This commit is contained in:
Irmen de Jong 2021-11-11 00:15:09 +01:00
parent 222bcb808f
commit 69f4a4d4f8
4 changed files with 20 additions and 18 deletions

View File

@ -46,7 +46,8 @@ internal class BeforeAsmGenerationAstChanger(val program: Program, private val o
if(binExpr.operator in associativeOperators) { if(binExpr.operator in associativeOperators) {
// A = <something-without-A> <associativeoperator> <otherthing-with-A> // A = <something-without-A> <associativeoperator> <otherthing-with-A>
// use the other part of the expression to split. // use the other part of the expression to split.
val (_, right) = binExpr.right.typecastTo(assignment.target.inferType(program).getOr(DataType.UNDEFINED), program, implicit=true) val sourceDt = binExpr.right.inferType(program).getOrElse { throw AssemblyError("invalid dt") }
val (_, right) = binExpr.right.typecastTo(assignment.target.inferType(program).getOr(DataType.UNDEFINED), sourceDt, implicit=true)
val assignRight = Assignment(assignment.target, right, assignment.position) val assignRight = Assignment(assignment.target, right, assignment.position)
return listOf( return listOf(
IAstModification.InsertBefore(assignment, assignRight, parent as IStatementContainer), IAstModification.InsertBefore(assignment, assignRight, parent as IStatementContainer),
@ -54,7 +55,8 @@ internal class BeforeAsmGenerationAstChanger(val program: Program, private val o
IAstModification.ReplaceNode(binExpr.left, assignment.target.toExpression(), binExpr)) IAstModification.ReplaceNode(binExpr.left, assignment.target.toExpression(), binExpr))
} }
} else { } else {
val (_, left) = binExpr.left.typecastTo(assignment.target.inferType(program).getOr(DataType.UNDEFINED), program, implicit=true) val sourceDt = binExpr.left.inferType(program).getOrElse { throw AssemblyError("invalid dt") }
val (_, left) = binExpr.left.typecastTo(assignment.target.inferType(program).getOr(DataType.UNDEFINED), sourceDt, implicit=true)
val assignLeft = Assignment(assignment.target, left, assignment.position) val assignLeft = Assignment(assignment.target, left, assignment.position)
return listOf( return listOf(
IAstModification.InsertBefore(assignment, assignLeft, parent as IStatementContainer), IAstModification.InsertBefore(assignment, assignLeft, parent as IStatementContainer),
@ -335,13 +337,15 @@ internal class BeforeAsmGenerationAstChanger(val program: Program, private val o
if(dt1 in ByteDatatypes) { if(dt1 in ByteDatatypes) {
if(dt2 in ByteDatatypes) if(dt2 in ByteDatatypes)
return noModifications return noModifications
val cast1 = TypecastExpression(arg1, if(dt1==DataType.UBYTE) DataType.UWORD else DataType.WORD, true, functionCallStatement.position) val (replaced, cast) = arg1.typecastTo(if(dt1==DataType.UBYTE) DataType.UWORD else DataType.WORD, dt1, true)
return listOf(IAstModification.ReplaceNode(arg1, cast1, functionCallStatement)) if(replaced)
return listOf(IAstModification.ReplaceNode(arg1, cast, functionCallStatement))
} else { } else {
if(dt2 in WordDatatypes) if(dt2 in WordDatatypes)
return noModifications return noModifications
val cast2 = TypecastExpression(arg2, if(dt2==DataType.UBYTE) DataType.UWORD else DataType.WORD, true, functionCallStatement.position) val (replaced, cast) = arg2.typecastTo(if(dt2==DataType.UBYTE) DataType.UWORD else DataType.WORD, dt2, true)
return listOf(IAstModification.ReplaceNode(arg2, cast2, functionCallStatement)) if(replaced)
return listOf(IAstModification.ReplaceNode(arg2, cast, functionCallStatement))
} }
} }
return noModifications return noModifications

View File

@ -169,8 +169,9 @@ internal class StatementReorderer(val program: Program, val errors: IErrorReport
is Subroutine -> { is Subroutine -> {
val paramType = callee.parameters[argnum].type val paramType = callee.parameters[argnum].type
if(leftDt isAssignableTo paramType) { if(leftDt isAssignableTo paramType) {
val cast = TypecastExpression(expr.left, paramType, true, parent.position) val (replaced, cast) = expr.left.typecastTo(paramType, leftDt.getOr(DataType.UNDEFINED), true)
return listOf(IAstModification.ReplaceNode(expr.left, cast, expr)) if(replaced)
return listOf(IAstModification.ReplaceNode(expr.left, cast, expr))
} }
} }
is BuiltinFunctionStatementPlaceholder -> { is BuiltinFunctionStatementPlaceholder -> {
@ -178,8 +179,9 @@ internal class StatementReorderer(val program: Program, val errors: IErrorReport
val paramTypes = func.parameters[argnum].possibleDatatypes val paramTypes = func.parameters[argnum].possibleDatatypes
for(type in paramTypes) { for(type in paramTypes) {
if(leftDt isAssignableTo type) { if(leftDt isAssignableTo type) {
val cast = TypecastExpression(expr.left, type, true, parent.position) val (replaced, cast) = expr.left.typecastTo(type, leftDt.getOr(DataType.UNDEFINED), true)
return listOf(IAstModification.ReplaceNode(expr.left, cast, expr)) if(replaced)
return listOf(IAstModification.ReplaceNode(expr.left, cast, expr))
} }
} }
} }

View File

@ -63,17 +63,14 @@ sealed class Expression: Node {
} }
} }
fun typecastTo(targetDt: DataType, program: Program, sourceDt: DataType?=null, implicit: Boolean=false): Pair<Boolean, Expression> { fun typecastTo(targetDt: DataType, sourceDt: DataType, implicit: Boolean=false): Pair<Boolean, Expression> {
require(sourceDt==null || sourceDt!=DataType.UNDEFINED) require(sourceDt!=DataType.UNDEFINED && targetDt!=DataType.UNDEFINED)
require(targetDt!=DataType.UNDEFINED) if(sourceDt==targetDt)
return Pair(false, this)
if(this is TypecastExpression) { if(this is TypecastExpression) {
this.type = targetDt this.type = targetDt
return Pair(false, this) return Pair(false, this)
} }
val exprDt = sourceDt ?: this.inferType(program).getOr(DataType.UNDEFINED)
if(exprDt==targetDt)
return Pair(false, this)
val typecast = TypecastExpression(this, targetDt, implicit, this.position) val typecast = TypecastExpression(this, targetDt, implicit, this.position)
return Pair(true, typecast) return Pair(true, typecast)
} }

View File

@ -3,7 +3,6 @@ TODO
For next compiler release (7.3) For next compiler release (7.3)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
- let typecasting code use expression.typecastTo()
- add expression simplification to while and until loops as well. - add expression simplification to while and until loops as well.