@@ -431,12 +431,12 @@ void CoroutinesCodeGenerator::InsertCoroutine(const FunctionDecl& fd, const Coro
431
431
}
432
432
}
433
433
434
- auto & str = ofm.GetString ();
434
+ auto str = std::move ( ofm.GetString () );
435
435
ReplaceAll (str, " <" sv, " " sv);
436
436
ReplaceAll (str, " :" sv, " " sv);
437
437
ReplaceAll (str, " >" sv, " " sv);
438
- ReplaceAll (str, " , " sv, " " sv);
439
- ReplaceAll ( str, " " sv, " " sv );
438
+
439
+ str = BuildTemplateParamObjectName (str );
440
440
441
441
if (fd.isOverloadedOperator ()) {
442
442
return StrCat (MakeLineColumnName (ctx.getSourceManager (), stmt->getBeginLoc (), " operator_" sv), str);
@@ -737,7 +737,6 @@ void CoroutinesCodeGenerator::InsertArg(const CoroutineBodyStmt* stmt)
737
737
}
738
738
739
739
if (const auto * coReturnVoid = dyn_cast_or_null<CoreturnStmt>(stmt->getFallthroughHandler ())) {
740
- coReturnVoid->dump ();
741
740
funcBodyStmts.Add (coReturnVoid);
742
741
}
743
742
@@ -817,10 +816,11 @@ void CoroutinesCodeGenerator::InsertArg(const CallExpr* stmt)
817
816
}
818
817
// -----------------------------------------------------------------------------
819
818
820
- static std::optional<std::string> FindValue (llvm::DenseMap<const Expr*, std::string>& map, const Expr* key)
819
+ static std::optional<std::string>
820
+ FindValue (llvm::DenseMap<const Expr*, std::pair<const DeclRefExpr*, std::string>>& map, const Expr* key)
821
821
{
822
822
if (const auto & s = map.find (key); s != map.end ()) {
823
- return s->second ;
823
+ return s->second . second ;
824
824
}
825
825
826
826
return {};
@@ -838,18 +838,23 @@ void CoroutinesCodeGenerator::InsertArg(const OpaqueValueExpr* stmt)
838
838
// Needs to be internal because a user can create the same type and it gets put into the stack frame
839
839
std::string name{BuildSuspendVarName (stmt)};
840
840
841
+ // In case of a coroutine-template the same suspension point can occur multiple times. But to know when to add
842
+ // the _1 we must match the one from each instantiation. The DeclRefExpr is what distinguishes the same
843
+ // OpaqueValueExpr between multiple instantiations.
844
+ const auto * dref = FindDeclRef (sourceExpr);
845
+
841
846
// The initial_suspend and final_suspend expressions carry the same location info. If we hit such a case,
842
847
// make up another name.
843
848
// Below is a std::find_if. However, the same code looks unreadable with std::find_if
844
- for (const auto lookupName{StrCat (CORO_FRAME_ACCESS, name)}; const auto & [k, v ] : mOpaqueValues ) {
845
- if (v == lookupName) {
849
+ for (const auto lookupName{StrCat (CORO_FRAME_ACCESS, name)}; const auto & [k, value ] : mOpaqueValues ) {
850
+ if (auto [thisDeref, v] = value; (thisDeref == dref) and (v == lookupName) ) {
846
851
name += " _1" sv;
847
852
break ;
848
853
}
849
854
}
850
855
851
856
const auto accessName{StrCat (CORO_FRAME_ACCESS, name)};
852
- mOpaqueValues .insert (std::make_pair (sourceExpr, accessName));
857
+ mOpaqueValues .insert (std::make_pair (sourceExpr, std::make_pair (dref, accessName) ));
853
858
854
859
OutputFormatHelper ofm{};
855
860
CoroutinesCodeGenerator codeGenerator{ofm, mPosBeforeFunc , mFSMName , mSuspendsCount , mASTData };
0 commit comments