1
0
mirror of https://gitlab.com/camelot/kickc.git synced 2025-01-02 20:29:38 +00:00

- Resolved issue of nobank without parameters. Now it works ... The CPreprocessor adds () and the KickCParser.g4 demands brackets as part of the syntax. But you can write without in the code.

- Retested all test cases.
This commit is contained in:
Sven Van de Velde 2023-04-11 21:48:26 +02:00
parent 5c5ffd2736
commit 82a6c56e0a
7 changed files with 75 additions and 46 deletions

View File

@ -136,8 +136,8 @@ parameterDecl
; ;
pragma pragma
: PRAGMA NAME : PRAGMA NAME PAR_BEGIN PAR_END #pragmaNoParameters
| PRAGMA NAME (PAR_BEGIN pragmaParam (COMMA pragmaParam)* PAR_END)? | PRAGMA NAME PAR_BEGIN pragmaParam (COMMA pragmaParam)* PAR_END #pragmaParameters
; ;
pragmaParam pragmaParam

View File

@ -42,7 +42,7 @@ public class Pass0GenerateStatementSequence extends KickCParserBaseVisitor<Objec
/** Used to build the scopes of the source file. */ /** Used to build the scopes of the source file. */
private final Stack<Scope> scopeStack; private final Stack<Scope> scopeStack;
/** All #pragma constructor_for() statements. Collected during parsing and handled by {@link #generate()} before returning. */ /** All #pragma constructor_for() statements. Collected during parsing and handled by {@link #generate()} before returning. */
private final List<KickCParser.PragmaContext> pragmaConstructorFors; private final List<KickCParser.PragmaParametersContext> pragmaConstructorFors;
public Pass0GenerateStatementSequence(CParser cParser, KickCParser.FileContext fileCtx, Program program, Procedure.CallingConvention initialCallingConvention, StringEncoding defaultEncoding, String defaultInterruptType) { public Pass0GenerateStatementSequence(CParser cParser, KickCParser.FileContext fileCtx, Program program, Procedure.CallingConvention initialCallingConvention, StringEncoding defaultEncoding, String defaultInterruptType) {
@ -148,7 +148,7 @@ public class Pass0GenerateStatementSequence extends KickCParserBaseVisitor<Objec
// Handle #pragma constructor_for() // Handle #pragma constructor_for()
List<ProcedureRef> constructorProcs = new ArrayList<>(); List<ProcedureRef> constructorProcs = new ArrayList<>();
for(KickCParser.PragmaContext pragmaConstructorFor : pragmaConstructorFors) { for(KickCParser.PragmaParametersContext pragmaConstructorFor : pragmaConstructorFors) {
final List<KickCParser.PragmaParamContext> names = pragmaConstructorFor.pragmaParam(); final List<KickCParser.PragmaParamContext> names = pragmaConstructorFor.pragmaParam();
if(names.size() < 2) if(names.size() < 2)
throw new CompileError("#pragma constructor_for requires at least 2 parameters.", new StatementSource(pragmaConstructorFor)); throw new CompileError("#pragma constructor_for requires at least 2 parameters.", new StatementSource(pragmaConstructorFor));
@ -237,8 +237,29 @@ public class Pass0GenerateStatementSequence extends KickCParserBaseVisitor<Objec
return null; return null;
} }
// #pragma NAME without parameters are registered in the parser as #pragma NAME ().
// The CPreprocessor.java parses first the #pragma statements and modifies them
// and selects which pragmas are allowed to be passed to the main parser.
// #pragma nobank is such a pragma, that must be passed. The CPreprocessor adds
// parentheses after the NAME, which comply with the KickCParser.g4 definition.
// But the programmer can write #pragma nobank without issues.
public Object visitPragmaNoParameters(KickCParser.PragmaNoParametersContext ctx) {
final String pragmaName = ctx.NAME().getText();
switch(pragmaName) {
case CParser.PRAGMA_TARGET:
throw new InternalError("Error! #pragma target() should be handled in preprocessor!");
case CParser.PRAGMA_NOBANK:
this.currentBank = null; // When the current segment is null, the procedure will not be declared as far.
break;
default:
program.getLog().append("Warning! Unknown #pragma " + pragmaName);
}
return null;
}
@Override @Override
public Object visitPragma(KickCParser.PragmaContext ctx) { public Object visitPragmaParameters(KickCParser.PragmaParametersContext ctx) {
final String pragmaName = ctx.NAME().getText(); final String pragmaName = ctx.NAME().getText();
switch(pragmaName) { switch(pragmaName) {
case CParser.PRAGMA_TARGET: case CParser.PRAGMA_TARGET:
@ -340,7 +361,7 @@ public class Pass0GenerateStatementSequence extends KickCParserBaseVisitor<Objec
* @param ctx The #pragma * @param ctx The #pragma
* @return The single parameter * @return The single parameter
*/ */
private static KickCParser.PragmaParamContext pragmaParamSingle(KickCParser.PragmaContext ctx) { private static KickCParser.PragmaParamContext pragmaParamSingle(KickCParser.PragmaParametersContext ctx) {
if(ctx.pragmaParam().size() != 1) if(ctx.pragmaParam().size() != 1)
throw new CompileError("#pragma expects a single parameter!", new StatementSource(ctx)); throw new CompileError("#pragma expects a single parameter!", new StatementSource(ctx));
return ctx.pragmaParam().get(0); return ctx.pragmaParam().get(0);

View File

@ -245,6 +245,14 @@ public class CPreprocessor implements TokenSource {
pragmaTokens.addAll(pragmaBody); pragmaTokens.addAll(pragmaBody);
// Pass on the #pragma to the parser - and mark it as already handled // Pass on the #pragma to the parser - and mark it as already handled
cTokenSource.addSourceFirst(new ListTokenSource(pragmaTokens)); cTokenSource.addSourceFirst(new ListTokenSource(pragmaTokens));
} else {
ArrayList<Token> parenthesizedBody = new ArrayList<>();
parenthesizedBody.add(new CommonToken(KickCLexer.PAR_BEGIN, "("));
parenthesizedBody.add(new CommonToken(KickCLexer.PAR_END, ")"));
pragmaBody = parenthesizedBody;
pragmaTokens.addAll(pragmaBody);
// Pass on the #pragma to the parser - and mark it as already handled
cTokenSource.addSourceFirst(new ListTokenSource(pragmaTokens));
} }
parserPragmas.add(inputToken); parserPragmas.add(inputToken);
return true; return true;

View File

@ -30,10 +30,6 @@
char* const SCREEN = (char*)0x0400; char* const SCREEN = (char*)0x0400;
#pragma code_seg(Code)
void main(void) {
SCREEN[0] = plus('0', 7); // close call
}
#pragma code_seg(RAM_Bank1) #pragma code_seg(RAM_Bank1)
#pragma bank(cx16_ram, 1) #pragma bank(cx16_ram, 1)
@ -47,4 +43,8 @@ char min(char a, char b) {
return a+b; return a+b;
} }
#pragma code_seg(Code)
void main(void) {
SCREEN[0] = plus('0', 7); // close call
}

View File

@ -20,7 +20,7 @@ plus::@return: scope:[plus] from plus
[6] return [6] return
to:@return to:@return
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
min: scope:[min] from plus min: scope:[min] from plus
[7] phi() [7] phi()
to:min::@return to:min::@return

View File

@ -2,22 +2,6 @@ Loading link script "call-banked-phi.ld"
CONTROL FLOW GRAPH SSA CONTROL FLOW GRAPH SSA
void main()
main: scope:[main] from __start
plus::a#0 = '0'
plus::b#0 = 7
call plus
plus::return#0 = plus::return#2
to:main::@1
main::@1: scope:[main] from main
plus::return#3 = phi( main/plus::return#0 )
main::$0 = plus::return#3
SCREEN[0] = main::$0
to:main::@return
main::@return: scope:[main] from main::@1
return
to:@return
__bank(cx16_ram, 1) char plus(char a , char b) __bank(cx16_ram, 1) char plus(char a , char b)
plus: scope:[plus] from main plus: scope:[plus] from main
plus::b#1 = phi( main/plus::b#0 ) plus::b#1 = phi( main/plus::b#0 )
@ -30,15 +14,15 @@ plus: scope:[plus] from main
plus::@1: scope:[plus] from plus plus::@1: scope:[plus] from plus
min::return#3 = phi( plus/min::return#0 ) min::return#3 = phi( plus/min::return#0 )
plus::$0 = min::return#3 plus::$0 = min::return#3
plus::return#1 = plus::$0 plus::return#0 = plus::$0
to:plus::@return to:plus::@return
plus::@return: scope:[plus] from plus::@1 plus::@return: scope:[plus] from plus::@1
plus::return#4 = phi( plus::@1/plus::return#1 ) plus::return#3 = phi( plus::@1/plus::return#0 )
plus::return#2 = plus::return#4 plus::return#1 = plus::return#3
return return
to:@return to:@return
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
min: scope:[min] from plus min: scope:[min] from plus
min::b#1 = phi( plus/min::b#0 ) min::b#1 = phi( plus/min::b#0 )
min::a#1 = phi( plus/min::a#0 ) min::a#1 = phi( plus/min::a#0 )
@ -51,6 +35,22 @@ min::@return: scope:[min] from min
return return
to:@return to:@return
void main()
main: scope:[main] from __start
plus::a#0 = '0'
plus::b#0 = 7
call plus
plus::return#2 = plus::return#1
to:main::@1
main::@1: scope:[main] from main
plus::return#4 = phi( main/plus::return#2 )
main::$0 = plus::return#4
SCREEN[0] = main::$0
to:main::@return
main::@return: scope:[main] from main::@1
return
to:@return
void __start() void __start()
__start: scope:[__start] from __start: scope:[__start] from
call main call main
@ -66,7 +66,7 @@ __constant char * const SCREEN = (char *)$400
void __start() void __start()
void main() void main()
char main::$0 char main::$0
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
char min::$0 char min::$0
char min::a char min::a
char min::a#0 char min::a#0
@ -107,10 +107,10 @@ Successful SSA optimization PassNCastSimplification
Finalized unsigned number type (char) 7 Finalized unsigned number type (char) 7
Finalized unsigned number type (char) 0 Finalized unsigned number type (char) 0
Successful SSA optimization PassNFinalizeNumberTypeConversions Successful SSA optimization PassNFinalizeNumberTypeConversions
Alias plus::return#0 = plus::return#3
Alias min::return#0 = min::return#3 Alias min::return#0 = min::return#3
Alias plus::return#1 = plus::$0 plus::return#4 plus::return#2 Alias plus::return#0 = plus::$0 plus::return#3 plus::return#1
Alias min::return#1 = min::$0 min::return#4 min::return#2 Alias min::return#1 = min::$0 min::return#4 min::return#2
Alias plus::return#2 = plus::return#4
Successful SSA optimization Pass2AliasElimination Successful SSA optimization Pass2AliasElimination
Identical Phi Values plus::a#1 plus::a#0 Identical Phi Values plus::a#1 plus::a#0
Identical Phi Values plus::b#1 plus::b#0 Identical Phi Values plus::b#1 plus::b#0
@ -123,31 +123,31 @@ Successful SSA optimization Pass2ConstantIdentification
Constant min::a#0 = plus::a#0 Constant min::a#0 = plus::a#0
Constant min::b#0 = plus::b#0 Constant min::b#0 = plus::b#0
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Simplifying expression containing zero SCREEN in [5] SCREEN[0] = main::$0 Simplifying expression containing zero SCREEN in [15] SCREEN[0] = main::$0
Successful SSA optimization PassNSimplifyExpressionWithZero Successful SSA optimization PassNSimplifyExpressionWithZero
Removing unused procedure __start Removing unused procedure __start
Removing unused procedure block __start Removing unused procedure block __start
Removing unused procedure block __start::@1 Removing unused procedure block __start::@1
Removing unused procedure block __start::@return Removing unused procedure block __start::@return
Successful SSA optimization PassNEliminateEmptyStart Successful SSA optimization PassNEliminateEmptyStart
Constant right-side identified [9] min::return#1 = min::a#0 + min::b#0 Constant right-side identified [4] min::return#1 = min::a#0 + min::b#0
Successful SSA optimization Pass2ConstantRValueConsolidation Successful SSA optimization Pass2ConstantRValueConsolidation
Constant min::return#1 = min::a#0+min::b#0 Constant min::return#1 = min::a#0+min::b#0
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Constant min::return#0 = min::return#1 Constant min::return#0 = min::return#1
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Constant plus::return#1 = min::return#0 Constant plus::return#0 = min::return#0
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Constant plus::return#0 = plus::return#1 Constant plus::return#2 = plus::return#0
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Constant main::$0 = plus::return#0 Constant main::$0 = plus::return#2
Successful SSA optimization Pass2ConstantIdentification Successful SSA optimization Pass2ConstantIdentification
Inlining constant with different constant siblings min::return#0 Inlining constant with different constant siblings min::return#0
Constant inlined plus::return#0 = min::return#1 Constant inlined plus::return#0 = min::return#1
Constant inlined main::$0 = min::return#1 Constant inlined main::$0 = min::return#1
Constant inlined plus::return#1 = min::return#1
Constant inlined min::a#0 = plus::a#0 Constant inlined min::a#0 = plus::a#0
Constant inlined min::return#0 = min::return#1 Constant inlined min::return#0 = min::return#1
Constant inlined plus::return#2 = min::return#1
Constant inlined min::b#0 = plus::b#0 Constant inlined min::b#0 = plus::b#0
Successful SSA optimization Pass2ConstantInlining Successful SSA optimization Pass2ConstantInlining
Adding NOP phi() at start of main Adding NOP phi() at start of main
@ -188,7 +188,7 @@ plus::@return: scope:[plus] from plus
[6] return [6] return
to:@return to:@return
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
min: scope:[min] from plus min: scope:[min] from plus
[7] phi() [7] phi()
to:min::@return to:min::@return
@ -199,7 +199,7 @@ min::@return: scope:[min] from min
VARIABLE REGISTER WEIGHTS VARIABLE REGISTER WEIGHTS
void main() void main()
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
char min::a char min::a
char min::b char min::b
char min::return char min::return
@ -215,14 +215,14 @@ Statement [1] call plus [ ] ( [ ] { } ) always clobbers reg byte a
Statement [2] *SCREEN = min::return#1 [ ] ( [ ] { } ) always clobbers reg byte a Statement [2] *SCREEN = min::return#1 [ ] ( [ ] { } ) always clobbers reg byte a
REGISTER UPLIFT SCOPES REGISTER UPLIFT SCOPES
Uplift Scope [main]
Uplift Scope [plus] Uplift Scope [plus]
Uplift Scope [min] Uplift Scope [min]
Uplift Scope [main]
Uplift Scope [] Uplift Scope []
Uplifting [main] best 90 combination
Uplifting [plus] best 90 combination Uplifting [plus] best 90 combination
Uplifting [min] best 90 combination Uplifting [min] best 90 combination
Uplifting [main] best 90 combination
Uplifting [] best 90 combination Uplifting [] best 90 combination
ASSEMBLER BEFORE OPTIMIZATION ASSEMBLER BEFORE OPTIMIZATION
@ -341,7 +341,7 @@ Succesful ASM optimization Pass5UnusedLabelElimination
FINAL SYMBOL TABLE FINAL SYMBOL TABLE
__constant char * const SCREEN = (char *) 1024 __constant char * const SCREEN = (char *) 1024
void main() void main()
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
char min::a char min::a
char min::b char min::b
char min::return char min::return

View File

@ -1,6 +1,6 @@
__constant char * const SCREEN = (char *) 1024 __constant char * const SCREEN = (char *) 1024
void main() void main()
__bank(cx16_ram, 1) char min(char a , char b) char min(char a , char b)
char min::a char min::a
char min::b char min::b
char min::return char min::return