From 3990c7e1e508be2fbdf6ec5750aa5f4f820a6474 Mon Sep 17 00:00:00 2001 From: "Edward A. Lee" Date: Mon, 18 Nov 2024 11:43:20 -0800 Subject: [PATCH] Make macros available in methods --- .../lflang/generator/c/CMethodGenerator.java | 4 ++ test/C/src/MethodMacros.lf | 48 +++++++++++++++++++ 2 files changed, 52 insertions(+) create mode 100644 test/C/src/MethodMacros.lf diff --git a/core/src/main/java/org/lflang/generator/c/CMethodGenerator.java b/core/src/main/java/org/lflang/generator/c/CMethodGenerator.java index acb6d348ce..dfb7cabdeb 100644 --- a/core/src/main/java/org/lflang/generator/c/CMethodGenerator.java +++ b/core/src/main/java/org/lflang/generator/c/CMethodGenerator.java @@ -7,6 +7,7 @@ import org.lflang.generator.CodeBuilder; import org.lflang.lf.Method; import org.lflang.lf.Reactor; +import org.lflang.util.StringUtil; /** * Collection of functions to generate C code to declare methods. @@ -60,6 +61,8 @@ public static String generateMethod( var body = ASTUtils.toText(method.getCode()); code.prSourceLineNumber(method, suppressLineDirectives); + // Define macros for functions such as lf_tag(), lf_time_logical(), lf_set(), etc. + code.pr("#include " + StringUtil.addDoubleQuotes(CCoreFilesUtils.getCTargetSetHeader())); code.prComment("Implementation of method " + method.getName() + "()"); code.pr(generateMethodSignature(method, tpr, types) + " {"); @@ -83,6 +86,7 @@ public static String generateMethod( code.pr(body); code.unindent(); code.pr("}"); + code.pr("#include " + StringUtil.addDoubleQuotes(CCoreFilesUtils.getCTargetSetUndefHeader())); code.prEndSourceLineNumber(suppressLineDirectives); return code.toString(); } diff --git a/test/C/src/MethodMacros.lf b/test/C/src/MethodMacros.lf new file mode 100644 index 0000000000..3eafeec608 --- /dev/null +++ b/test/C/src/MethodMacros.lf @@ -0,0 +1,48 @@ +// Test ability of methods to call each other and (recursively) themselves. +target C { + timeout: 10 ms +} + +reactor Fib { + input in: int + output out: int + state foo: int = 2 + + // Return the n-th Fibonacci number. + // Type name convention for ports is _class_method_t, all lowercase. + method fib(out: _fib_out_t*, n: int): int {= + int result = 1; + if (n > 1) result = add(fib(NULL, n-1), fib(NULL, n-2)); + if (out != NULL) { + lf_set(out, result); + lf_print("At elapsed time " PRINTF_TIME ", fib(%d) = %d", lf_time_logical_elapsed(), n, result); + } + return result; + =} + + method add(x: int, y: int): int {= + return x + y; + =} + + reaction(in) -> out {= + fib(out, in->value); + =} +} + +main reactor { + state count: int = 0 + timer t(0, 1 ms) + fib = new Fib() + + reaction(t) -> fib.in {= + lf_set(fib.in, self->count); + =} + + reaction(fib.out) {= + lf_print("fib(%d) = %d", self->count, fib.out->value); + int answers[] = {1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89}; + if (fib.out->value != answers[self->count++]) { + lf_print_error_and_exit("Expected %d", answers[self->count-1]); + } + =} +}