Skip to content

Commit be670da

Browse files
authoredMar 21, 2025
Merge pull request #79980 from hborla/closure-body-macro
[Macros] Implement support for function body macros on closures.
2 parents bff9169 + b005df3 commit be670da

20 files changed

+593
-62
lines changed
 

‎include/swift/AST/ASTMangler.h

+9-1
Original file line numberDiff line numberDiff line change
@@ -412,10 +412,18 @@ class ASTMangler : public Mangler {
412412
std::string mangleMacroExpansion(const FreestandingMacroExpansion *expansion);
413413
std::string mangleAttachedMacroExpansion(
414414
const Decl *decl, CustomAttr *attr, MacroRole role);
415+
std::string mangleAttachedMacroExpansion(
416+
ClosureExpr *attachedTo, CustomAttr *attr, MacroDecl *macro);
415417

416418
void appendMacroExpansion(const FreestandingMacroExpansion *expansion);
417419
void appendMacroExpansionContext(SourceLoc loc, DeclContext *origDC,
418-
const FreestandingMacroExpansion *expansion);
420+
Identifier macroName,
421+
unsigned discriminator);
422+
423+
void appendMacroExpansion(ClosureExpr *attachedTo,
424+
CustomAttr *attr,
425+
MacroDecl *macro);
426+
419427
void appendMacroExpansionOperator(
420428
StringRef macroName, MacroRole role, unsigned discriminator);
421429

‎include/swift/AST/AnyFunctionRef.h

+48
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,54 @@ class AnyFunctionRef {
243243
llvm_unreachable("unexpected AnyFunctionRef representation");
244244
}
245245

246+
DeclAttributes getDeclAttributes() const {
247+
if (auto afd = TheFunction.dyn_cast<AbstractFunctionDecl *>()) {
248+
return afd->getExpandedAttrs();
249+
}
250+
251+
if (auto ace = TheFunction.dyn_cast<AbstractClosureExpr *>()) {
252+
if (auto *ce = dyn_cast<ClosureExpr>(ace)) {
253+
return ce->getAttrs();
254+
}
255+
}
256+
257+
return DeclAttributes();
258+
}
259+
260+
MacroDecl *getResolvedMacro(CustomAttr *attr) const {
261+
if (auto afd = TheFunction.dyn_cast<AbstractFunctionDecl *>()) {
262+
return afd->getResolvedMacro(attr);
263+
}
264+
265+
if (auto ace = TheFunction.dyn_cast<AbstractClosureExpr *>()) {
266+
if (auto *ce = dyn_cast<ClosureExpr>(ace)) {
267+
return ce->getResolvedMacro(attr);
268+
}
269+
}
270+
271+
return nullptr;
272+
}
273+
274+
using MacroCallback = llvm::function_ref<void(CustomAttr *, MacroDecl *)>;
275+
276+
void
277+
forEachAttachedMacro(MacroRole role,
278+
MacroCallback macroCallback) const {
279+
auto attrs = getDeclAttributes();
280+
for (auto customAttrConst : attrs.getAttributes<CustomAttr>()) {
281+
auto customAttr = const_cast<CustomAttr *>(customAttrConst);
282+
auto *macroDecl = getResolvedMacro(customAttr);
283+
284+
if (!macroDecl)
285+
continue;
286+
287+
if (!macroDecl->getMacroRoles().contains(role))
288+
continue;
289+
290+
macroCallback(customAttr, macroDecl);
291+
}
292+
}
293+
246294
friend bool operator==(AnyFunctionRef lhs, AnyFunctionRef rhs) {
247295
return lhs.TheFunction == rhs.TheFunction;
248296
}

‎include/swift/AST/DiagnosticsSema.def

+3
Original file line numberDiff line numberDiff line change
@@ -7843,6 +7843,9 @@ ERROR(conformance_macro,none,
78437843
ERROR(experimental_macro,none,
78447844
"macro %0 is experimental",
78457845
(DeclName))
7846+
ERROR(experimental_closure_body_macro,none,
7847+
"function body macros on closures is experimental",
7848+
())
78467849

78477850
ERROR(macro_resolve_circular_reference, none,
78487851
"circular reference resolving %select{freestanding|attached}0 macro %1",

‎include/swift/AST/Expr.h

+6
Original file line numberDiff line numberDiff line change
@@ -4305,6 +4305,8 @@ class ClosureExpr : public AbstractClosureExpr {
43054305
BraceStmt *getBody() const { return Body; }
43064306
void setBody(BraceStmt *S) { Body = S; }
43074307

4308+
BraceStmt *getExpandedBody();
4309+
43084310
DeclAttributes &getAttrs() { return Attributes; }
43094311
const DeclAttributes &getAttrs() const { return Attributes; }
43104312

@@ -4422,6 +4424,10 @@ class ClosureExpr : public AbstractClosureExpr {
44224424
return ExplicitResultTypeAndBodyState.getPointer()->getTypeRepr();
44234425
}
44244426

4427+
/// Returns the resolved macro for the given custom attribute
4428+
/// attached to this closure expression.
4429+
MacroDecl *getResolvedMacro(CustomAttr *customAttr);
4430+
44254431
/// Determine whether the closure has a single expression for its
44264432
/// body.
44274433
///

‎include/swift/AST/TypeCheckRequests.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -4708,7 +4708,7 @@ class ExpandPreambleMacroRequest
47084708

47094709
class ExpandBodyMacroRequest
47104710
: public SimpleRequest<ExpandBodyMacroRequest,
4711-
std::optional<unsigned>(AbstractFunctionDecl *),
4711+
std::optional<unsigned>(AnyFunctionRef),
47124712
RequestFlags::Cached> {
47134713
public:
47144714
using SimpleRequest::SimpleRequest;
@@ -4717,7 +4717,7 @@ class ExpandBodyMacroRequest
47174717
friend SimpleRequest;
47184718

47194719
std::optional<unsigned> evaluate(Evaluator &evaluator,
4720-
AbstractFunctionDecl *fn) const;
4720+
AnyFunctionRef fn) const;
47214721

47224722
public:
47234723
bool isCached() const { return true; }

‎include/swift/AST/TypeCheckerTypeIDZone.def

+1-1
Original file line numberDiff line numberDiff line change
@@ -531,7 +531,7 @@ SWIFT_REQUEST(TypeChecker, ExpandPreambleMacroRequest,
531531
ArrayRef<unsigned>(AbstractFunctionDecl *),
532532
Cached, NoLocationInfo)
533533
SWIFT_REQUEST(TypeChecker, ExpandBodyMacroRequest,
534-
std::optional<unsigned>(AbstractFunctionDecl *),
534+
std::optional<unsigned>(AnyFunctionRef),
535535
Cached, NoLocationInfo)
536536
SWIFT_REQUEST(TypeChecker, LocalDiscriminatorsRequest,
537537
unsigned(DeclContext *),

‎include/swift/Basic/Features.def

+3
Original file line numberDiff line numberDiff line change
@@ -512,6 +512,9 @@ EXPERIMENTAL_FEATURE(ConcurrencySyntaxSugar, true)
512512
/// Allow declaration of compile-time values
513513
EXPERIMENTAL_FEATURE(CompileTimeValues, true)
514514

515+
/// Allow function body macros applied to closures.
516+
EXPERIMENTAL_FEATURE(ClosureBodyMacro, true)
517+
515518
#undef EXPERIMENTAL_FEATURE_EXCLUDED_FROM_MODULE_INTERFACE
516519
#undef EXPERIMENTAL_FEATURE
517520
#undef UPCOMING_FEATURE

‎lib/AST/ASTMangler.cpp

+56-10
Original file line numberDiff line numberDiff line change
@@ -4824,7 +4824,8 @@ static Identifier encodeLocalPrecheckedDiscriminator(
48244824

48254825
void ASTMangler::appendMacroExpansionContext(
48264826
SourceLoc loc, DeclContext *origDC,
4827-
const FreestandingMacroExpansion *expansion
4827+
Identifier macroName,
4828+
unsigned macroDiscriminator
48284829
) {
48294830
origDC = MacroDiscriminatorContext::getInnermostMacroContext(origDC);
48304831
BaseEntitySignature nullBase(nullptr);
@@ -4833,9 +4834,9 @@ void ASTMangler::appendMacroExpansionContext(
48334834
if (auto outermostLocalDC = getOutermostLocalContext(origDC)) {
48344835
auto innermostNonlocalDC = outermostLocalDC->getParent();
48354836
appendContext(innermostNonlocalDC, nullBase, StringRef());
4836-
Identifier name = expansion->getMacroName().getBaseIdentifier();
4837+
Identifier name = macroName;
48374838
ASTContext &ctx = origDC->getASTContext();
4838-
unsigned discriminator = expansion->getDiscriminator();
4839+
unsigned discriminator = macroDiscriminator;
48394840
name = encodeLocalPrecheckedDiscriminator(ctx, name, discriminator);
48404841
appendIdentifier(name.str());
48414842
} else {
@@ -4940,7 +4941,10 @@ void ASTMangler::appendMacroExpansionContext(
49404941
return appendMacroExpansionLoc();
49414942

49424943
// Append our own context and discriminator.
4943-
appendMacroExpansionContext(outerExpansionLoc, origDC, expansion);
4944+
appendMacroExpansionContext(
4945+
outerExpansionLoc, origDC,
4946+
macroName,
4947+
macroDiscriminator);
49444948
appendMacroExpansionOperator(
49454949
baseName.userFacingName(), role, discriminator);
49464950
}
@@ -4967,16 +4971,14 @@ void ASTMangler::appendMacroExpansionOperator(
49674971
}
49684972

49694973
static StringRef getPrivateDiscriminatorIfNecessary(
4970-
const MacroExpansionExpr *expansion) {
4971-
auto dc = MacroDiscriminatorContext::getInnermostMacroContext(
4972-
expansion->getDeclContext());
4973-
auto decl = dc->getAsDecl();
4974+
const DeclContext *macroDC) {
4975+
auto decl = macroDC->getAsDecl();
49744976
if (decl && !decl->isOutermostPrivateOrFilePrivateScope())
49754977
return StringRef();
49764978

49774979
// Mangle non-local private declarations with a textual discriminator
49784980
// based on their enclosing file.
4979-
auto topLevelSubcontext = dc->getModuleScopeContext();
4981+
auto topLevelSubcontext = macroDC->getModuleScopeContext();
49804982
SourceFile *sf = dyn_cast<SourceFile>(topLevelSubcontext);
49814983
if (!sf)
49824984
return StringRef();
@@ -4992,6 +4994,13 @@ static StringRef getPrivateDiscriminatorIfNecessary(
49924994
return discriminator.str();
49934995
}
49944996

4997+
static StringRef getPrivateDiscriminatorIfNecessary(
4998+
const MacroExpansionExpr *expansion) {
4999+
auto dc = MacroDiscriminatorContext::getInnermostMacroContext(
5000+
expansion->getDeclContext());
5001+
return getPrivateDiscriminatorIfNecessary(dc);
5002+
}
5003+
49955004
static StringRef getPrivateDiscriminatorIfNecessary(
49965005
const FreestandingMacroExpansion *expansion) {
49975006
switch (expansion->getFreestandingMacroKind()) {
@@ -5008,7 +5017,8 @@ void
50085017
ASTMangler::appendMacroExpansion(const FreestandingMacroExpansion *expansion) {
50095018
appendMacroExpansionContext(expansion->getPoundLoc(),
50105019
expansion->getDeclContext(),
5011-
expansion);
5020+
expansion->getMacroName().getBaseIdentifier(),
5021+
expansion->getDiscriminator());
50125022
auto privateDiscriminator = getPrivateDiscriminatorIfNecessary(expansion);
50135023
if (!privateDiscriminator.empty()) {
50145024
appendIdentifier(privateDiscriminator);
@@ -5020,6 +5030,42 @@ ASTMangler::appendMacroExpansion(const FreestandingMacroExpansion *expansion) {
50205030
expansion->getDiscriminator());
50215031
}
50225032

5033+
void ASTMangler::appendMacroExpansion(ClosureExpr *attachedTo,
5034+
CustomAttr *attr,
5035+
MacroDecl *macro) {
5036+
auto &ctx = attachedTo->getASTContext();
5037+
auto discriminator =
5038+
ctx.getNextMacroDiscriminator(attachedTo,
5039+
macro->getBaseName());
5040+
5041+
appendMacroExpansionContext(
5042+
attr->getLocation(),
5043+
attachedTo,
5044+
macro->getBaseName().getIdentifier(),
5045+
discriminator);
5046+
5047+
auto privateDiscriminator =
5048+
getPrivateDiscriminatorIfNecessary(attachedTo);
5049+
if (!privateDiscriminator.empty()) {
5050+
appendIdentifier(privateDiscriminator);
5051+
appendOperator("Ll");
5052+
}
5053+
5054+
appendMacroExpansionOperator(
5055+
macro->getBaseName().userFacingName(),
5056+
MacroRole::Body,
5057+
discriminator);
5058+
}
5059+
5060+
std::string
5061+
ASTMangler::mangleAttachedMacroExpansion(ClosureExpr *attachedTo,
5062+
CustomAttr *attr,
5063+
MacroDecl *macro) {
5064+
beginMangling();
5065+
appendMacroExpansion(attachedTo, attr, macro);
5066+
return finalize();
5067+
}
5068+
50235069
std::string
50245070
ASTMangler::mangleMacroExpansion(const FreestandingMacroExpansion *expansion) {
50255071
beginMangling();

‎lib/AST/ASTScopeCreation.cpp

+15-6
Original file line numberDiff line numberDiff line change
@@ -309,13 +309,22 @@ ASTSourceFileScope::ASTSourceFileScope(SourceFile *SF,
309309
break;
310310
}
311311
case MacroRole::Body: {
312-
// Use the end location of the function decl itself as the parentLoc
313-
// for the new function body scope. This is different from the end
314-
// location of the original source range, which is after the end of the
315-
// function decl.
316312
auto expansion = SF->getMacroExpansion();
317-
parentLoc = expansion.getEndLoc();
318-
bodyForDecl = cast<AbstractFunctionDecl>(expansion.get<Decl *>());
313+
if (expansion.is<Decl *>()) {
314+
// Use the end location of the function decl itself as the parentLoc
315+
// for the new function body scope. This is different from the end
316+
// location of the original source range, which is after the end of the
317+
// function decl.
318+
bodyForDecl = cast<AbstractFunctionDecl>(expansion.get<Decl *>());
319+
parentLoc = expansion.getEndLoc();
320+
break;
321+
}
322+
323+
// Otherwise, we have a closure body macro.
324+
auto insertionRange = SF->getMacroInsertionRange();
325+
parentLoc = insertionRange.End;
326+
if (insertionRange.Start != insertionRange.End)
327+
parentLoc = parentLoc.getAdvancedLoc(-1);
319328
break;
320329
}
321330
}

‎lib/AST/Expr.cpp

+35
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
#include "swift/Basic/Assertions.h"
1919
#include "swift/Basic/Statistic.h"
2020
#include "swift/Basic/Unicode.h"
21+
#include "swift/Basic/SourceManager.h"
2122
#include "swift/AST/ASTContext.h"
2223
#include "swift/AST/ASTVisitor.h"
2324
#include "swift/AST/Decl.h" // FIXME: Bad dependency
@@ -2016,6 +2017,29 @@ BraceStmt * AbstractClosureExpr::getBody() const {
20162017
llvm_unreachable("Unknown closure expression");
20172018
}
20182019

2020+
BraceStmt *ClosureExpr::getExpandedBody() {
2021+
auto &ctx = getASTContext();
2022+
2023+
// Expand a body macro, if there is one.
2024+
BraceStmt *macroExpandedBody = nullptr;
2025+
if (auto bufferID = evaluateOrDefault(
2026+
ctx.evaluator,
2027+
ExpandBodyMacroRequest{this},
2028+
std::nullopt)) {
2029+
CharSourceRange bufferRange = ctx.SourceMgr.getRangeForBuffer(*bufferID);
2030+
auto bufferStart = bufferRange.getStart();
2031+
auto module = getParentModule();
2032+
auto macroSourceFile = module->getSourceFileContainingLocation(bufferStart);
2033+
2034+
if (macroSourceFile->getTopLevelItems().size() == 1) {
2035+
auto stmt = macroSourceFile->getTopLevelItems()[0].dyn_cast<Stmt *>();
2036+
macroExpandedBody = dyn_cast<BraceStmt>(stmt);
2037+
}
2038+
}
2039+
2040+
return macroExpandedBody;
2041+
}
2042+
20192043
bool AbstractClosureExpr::bodyHasExplicitReturnStmt() const {
20202044
return AnyFunctionRef(const_cast<AbstractClosureExpr *>(this))
20212045
.bodyHasExplicitReturnStmt();
@@ -2175,6 +2199,17 @@ void ClosureExpr::setExplicitResultType(Type ty) {
21752199
->setType(MetatypeType::get(ty));
21762200
}
21772201

2202+
MacroDecl *
2203+
ClosureExpr::getResolvedMacro(CustomAttr *customAttr) {
2204+
auto &ctx = getASTContext();
2205+
auto declRef = evaluateOrDefault(
2206+
ctx.evaluator,
2207+
ResolveMacroRequest{customAttr, this},
2208+
ConcreteDeclRef());
2209+
2210+
return dyn_cast_or_null<MacroDecl>(declRef.getDecl());
2211+
}
2212+
21782213
FORWARD_SOURCE_LOCS_TO(AutoClosureExpr, Body)
21792214

21802215
void AutoClosureExpr::setBody(Expr *E) {

‎lib/AST/FeatureSet.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -360,6 +360,10 @@ static bool usesFeatureCompileTimeValues(Decl *decl) {
360360
return decl->getAttrs().hasAttribute<ConstValAttr>();
361361
}
362362

363+
static bool usesFeatureClosureBodyMacro(Decl *decl) {
364+
return false;
365+
}
366+
363367
static bool usesFeatureMemorySafetyAttributes(Decl *decl) {
364368
if (decl->getAttrs().hasAttribute<SafeAttr>() ||
365369
decl->getAttrs().hasAttribute<UnsafeAttr>())

‎lib/AST/TypeCheckRequests.cpp

+8-10
Original file line numberDiff line numberDiff line change
@@ -2681,19 +2681,17 @@ void ExpandPreambleMacroRequest::noteCycleStep(DiagnosticEngine &diags) const {
26812681
//----------------------------------------------------------------------------//
26822682

26832683
void ExpandBodyMacroRequest::diagnoseCycle(DiagnosticEngine &diags) const {
2684-
auto decl = std::get<0>(getStorage());
2685-
diags.diagnose(decl->getLoc(),
2686-
diag::macro_expand_circular_reference_entity,
2687-
"body",
2688-
decl->getName());
2684+
auto fn = std::get<0>(getStorage());
2685+
diags.diagnose(fn.getLoc(),
2686+
diag::macro_expand_circular_reference_unnamed,
2687+
"body");
26892688
}
26902689

26912690
void ExpandBodyMacroRequest::noteCycleStep(DiagnosticEngine &diags) const {
2692-
auto decl = std::get<0>(getStorage());
2693-
diags.diagnose(decl->getLoc(),
2694-
diag::macro_expand_circular_reference_entity_through,
2695-
"body",
2696-
decl->getName());
2691+
auto fn = std::get<0>(getStorage());
2692+
diags.diagnose(fn.getLoc(),
2693+
diag::macro_expand_circular_reference_unnamed_through,
2694+
"body");
26972695
}
26982696

26992697
//----------------------------------------------------------------------------//

0 commit comments

Comments
 (0)
Please sign in to comment.