Skip to content

Commit 759f520

Browse files
committed
[Macros] Only freestanding expression macros can have a non-Void result type
Fixes rdar://108871352.
1 parent 9991468 commit 759f520

File tree

4 files changed

+43
-1
lines changed

4 files changed

+43
-1
lines changed

include/swift/AST/DiagnosticsSema.def

+8
Original file line numberDiff line numberDiff line change
@@ -7101,6 +7101,14 @@ ERROR(macro_in_nested,none,
71017101
ERROR(macro_without_role,none,
71027102
"macro %0 must declare its applicable roles via '@freestanding' or @attached'",
71037103
(DeclName))
7104+
ERROR(macro_result_type_cannot_be_used,none,
7105+
"only a freestanding expression macro can produce a result of type %0",
7106+
(Type))
7107+
NOTE(macro_remove_result_type,none,
7108+
"remove the result type if the macro does not produce a value",
7109+
())
7110+
NOTE(macro_make_freestanding_expression,none,
7111+
"make this macro a freestanding expression macro", ())
71047112
ERROR(macro_expansion_missing_pound,none,
71057113
"expansion of macro %0 requires leading '#'", (DeclName))
71067114
ERROR(macro_expansion_missing_arguments,none,

lib/AST/Decl.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -3357,12 +3357,18 @@ TypeRepr *ValueDecl::getResultTypeRepr() const {
33573357
returnRepr = FD->getResultTypeRepr();
33583358
} else if (auto *SD = dyn_cast<SubscriptDecl>(this)) {
33593359
returnRepr = SD->getElementTypeRepr();
3360+
} else if (auto *MD = dyn_cast<MacroDecl>(this)) {
3361+
returnRepr = MD->resultType.getTypeRepr();
33603362
}
33613363

33623364
return returnRepr;
33633365
}
33643366

33653367
TypeRepr *ValueDecl::getOpaqueResultTypeRepr() const {
3368+
// FIXME: Macros don't allow opaque result types yet.
3369+
if (isa<MacroDecl>(this))
3370+
return nullptr;
3371+
33663372
auto *returnRepr = this->getResultTypeRepr();
33673373

33683374
auto *dc = getDeclContext();

lib/Sema/TypeCheckDeclPrimary.cpp

+17
Original file line numberDiff line numberDiff line change
@@ -2060,6 +2060,23 @@ class DeclChecker : public DeclVisitor<DeclChecker> {
20602060
break;
20612061
}
20622062
}
2063+
2064+
// If the macro has a (non-Void) result type, it must have the freestanding
2065+
// expression role. Other roles cannot have result types.
2066+
if (auto resultTypeRepr = MD->getResultTypeRepr()) {
2067+
if (!MD->getMacroRoles().contains(MacroRole::Expression) &&
2068+
!MD->getResultInterfaceType()->isEqual(Ctx.getVoidType())) {
2069+
auto resultType = MD->getResultInterfaceType();
2070+
Ctx.Diags.diagnose(
2071+
MD->arrowLoc, diag::macro_result_type_cannot_be_used, resultType)
2072+
.highlight(resultTypeRepr->getSourceRange());
2073+
Ctx.Diags.diagnose(MD->arrowLoc, diag::macro_make_freestanding_expression)
2074+
.fixItInsert(MD->getAttributeInsertionLoc(false),
2075+
"@freestanding(expression)\n");
2076+
Ctx.Diags.diagnose(MD->arrowLoc, diag::macro_remove_result_type)
2077+
.fixItRemove(SourceRange(MD->arrowLoc, resultTypeRepr->getEndLoc()));
2078+
}
2079+
}
20632080
}
20642081

20652082
void visitMacroExpansionDecl(MacroExpansionDecl *MED) {

test/Macros/macros_diagnostics.swift

+12-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ internal struct X { } // expected-note{{type declared here}}
4646
// expected-warning@-1{{external macro implementation type}}
4747

4848
struct ZZZ {
49-
macro m5() -> Int = #externalMacro(module: "BuiltinMacros", type: "Blah")
49+
macro m5() = #externalMacro(module: "BuiltinMacros", type: "Blah")
5050
// expected-error@-1{{macro 'm5()' can only be declared at file scope}}
5151
// expected-error@-2{{macro 'm5()' must declare its applicable roles}}
5252
// expected-warning@-3{{external macro implementation type}}
@@ -200,3 +200,14 @@ struct SomeType {
200200
// expected-error@-2{{use of protocol 'Hashable' as a type must be written 'any Hashable'}}
201201
// expected-error@-3{{external macro implementation type}}
202202
}
203+
204+
205+
206+
@freestanding(declaration) macro nonExpressionReturnsInt<T>(_: T) -> Int = #externalMacro(module: "A", type: "B")
207+
// expected-warning@-1{{external macro implementation type}}
208+
// expected-error@-2{{only a freestanding expression macro can produce a result of type 'Int'}}
209+
// expected-note@-3{{make this macro a freestanding expression macro}}{{1-1=@freestanding(expression)\n}}
210+
// expected-note@-4{{remove the result type if the macro does not produce a value}}{{67-74=}}
211+
212+
@freestanding(declaration) macro nonExpressionReturnsVoid<T>(_: T) -> Void = #externalMacro(module: "A", type: "B")
213+
// expected-warning@-1{{external macro implementation type}}

0 commit comments

Comments
 (0)