Skip to content

Commit

Permalink
[cpp] Fix unhandled is Derived d.
Browse files Browse the repository at this point in the history
  • Loading branch information
pfusik committed Oct 5, 2024
1 parent bc28dbd commit 863faca
Show file tree
Hide file tree
Showing 5 changed files with 107 additions and 46 deletions.
37 changes: 26 additions & 11 deletions GenCpp.fu
Original file line number Diff line number Diff line change
Expand Up @@ -1482,30 +1482,42 @@ public class GenCpp : GenCCpp
base.VisitBinaryExpr(expr, parent);
}

static bool HasLambdaCapture(FuExpr expr)
static bool HasLambdaCapture(FuExpr expr, List<FuVar>! lambdaVars)
{
switch (expr) {
case FuAggregateInitializer init:
return init.Items.Any(item => HasLambdaCapture(item));
return init.Items.Any(item => HasLambdaCapture(item, lambdaVars));
case FuLiteral:
case FuLambdaExpr: // TODO: nested lambdas
return false;
case FuInterpolatedString interp:
return interp.Parts.Any(part => HasLambdaCapture(part.Argument));
return interp.Parts.Any(part => HasLambdaCapture(part.Argument, lambdaVars));
case FuSymbolReference symbol:
if (symbol.Left != null)
return HasLambdaCapture(symbol.Left);
if (symbol.Symbol is FuMember member)
return HasLambdaCapture(symbol.Left, lambdaVars);
switch (symbol.Symbol) {
case FuMember member:
return !member.IsStatic();
return !(symbol.Symbol.Parent is FuLambdaExpr); // TODO: nested lambdas
case FuVar def:
return !lambdaVars.Contains(def);
default:
return false; // class, enum, const
}
case FuUnaryExpr unary:
return unary.Inner != null && HasLambdaCapture(unary.Inner);
return unary.Inner != null && HasLambdaCapture(unary.Inner, lambdaVars);
case FuBinaryExpr binary:
return HasLambdaCapture(binary.Left) || HasLambdaCapture(binary.Right);
if (HasLambdaCapture(binary.Left, lambdaVars))
return true;
if (binary.Op == FuToken.Is) {
if (binary.Right is FuVar def)
lambdaVars.Add(def);
return false;
}
return HasLambdaCapture(binary.Right, lambdaVars);
case FuSelectExpr select:
return HasLambdaCapture(select.Cond) || HasLambdaCapture(select.OnTrue) || HasLambdaCapture(select.OnFalse);
return HasLambdaCapture(select.Cond, lambdaVars) || HasLambdaCapture(select.OnTrue, lambdaVars) || HasLambdaCapture(select.OnFalse, lambdaVars);
case FuCallExpr call:
return HasLambdaCapture(call.Method) || call.Arguments.Any(arg => HasLambdaCapture(arg));
return HasLambdaCapture(call.Method, lambdaVars) || call.Arguments.Any(arg => HasLambdaCapture(arg, lambdaVars));
default:
assert false;
}
Expand All @@ -1514,7 +1526,10 @@ public class GenCpp : GenCCpp
internal override void VisitLambdaExpr!(FuLambdaExpr expr)
{
WriteChar('[');
if (HasLambdaCapture(expr.Body))
List<FuVar>() lambdaVars;
assert expr.First is FuVar it;
lambdaVars.Add(it);
if (HasLambdaCapture(expr.Body, lambdaVars))
WriteChar('&');
Write("](");
if (expr.First.Type is FuOwningType || expr.First.Type.Id == FuId.StringStorageType) {
Expand Down
36 changes: 25 additions & 11 deletions libfut.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -15084,37 +15084,51 @@ void GenCpp::visitBinaryExpr(const FuBinaryExpr * expr, FuPriority parent)
GenBase::visitBinaryExpr(expr, parent);
}

bool GenCpp::hasLambdaCapture(const FuExpr * expr)
bool GenCpp::hasLambdaCapture(const FuExpr * expr, std::vector<const FuVar *> * lambdaVars)
{
if (const FuAggregateInitializer *init = dynamic_cast<const FuAggregateInitializer *>(expr))
return std::any_of(init->items.begin(), init->items.end(), [](const std::shared_ptr<FuExpr> &item) { return hasLambdaCapture(item.get()); });
return std::any_of(init->items.begin(), init->items.end(), [&](const std::shared_ptr<FuExpr> &item) { return hasLambdaCapture(item.get(), lambdaVars); });
else if (dynamic_cast<const FuLiteral *>(expr) || dynamic_cast<const FuLambdaExpr *>(expr))
return false;
else if (const FuInterpolatedString *interp = dynamic_cast<const FuInterpolatedString *>(expr))
return std::any_of(interp->parts.begin(), interp->parts.end(), [](const FuInterpolatedPart &part) { return hasLambdaCapture(part.argument.get()); });
return std::any_of(interp->parts.begin(), interp->parts.end(), [&](const FuInterpolatedPart &part) { return hasLambdaCapture(part.argument.get(), lambdaVars); });
else if (const FuSymbolReference *symbol = dynamic_cast<const FuSymbolReference *>(expr)) {
if (symbol->left != nullptr)
return hasLambdaCapture(symbol->left.get());
return hasLambdaCapture(symbol->left.get(), lambdaVars);
if (const FuMember *member = dynamic_cast<const FuMember *>(symbol->symbol))
return !member->isStatic();
return !dynamic_cast<const FuLambdaExpr *>(symbol->symbol->parent);
else if (const FuVar *def = dynamic_cast<const FuVar *>(symbol->symbol))
return !(std::find(lambdaVars->begin(), lambdaVars->end(), def) != lambdaVars->end());
else
return false;
}
else if (const FuUnaryExpr *unary = dynamic_cast<const FuUnaryExpr *>(expr))
return unary->inner != nullptr && hasLambdaCapture(unary->inner.get());
else if (const FuBinaryExpr *binary = dynamic_cast<const FuBinaryExpr *>(expr))
return hasLambdaCapture(binary->left.get()) || hasLambdaCapture(binary->right.get());
return unary->inner != nullptr && hasLambdaCapture(unary->inner.get(), lambdaVars);
else if (const FuBinaryExpr *binary = dynamic_cast<const FuBinaryExpr *>(expr)) {
if (hasLambdaCapture(binary->left.get(), lambdaVars))
return true;
if (binary->op == FuToken::is) {
if (const FuVar *def = dynamic_cast<const FuVar *>(binary->right.get()))
lambdaVars->push_back(def);
return false;
}
return hasLambdaCapture(binary->right.get(), lambdaVars);
}
else if (const FuSelectExpr *select = dynamic_cast<const FuSelectExpr *>(expr))
return hasLambdaCapture(select->cond.get()) || hasLambdaCapture(select->onTrue.get()) || hasLambdaCapture(select->onFalse.get());
return hasLambdaCapture(select->cond.get(), lambdaVars) || hasLambdaCapture(select->onTrue.get(), lambdaVars) || hasLambdaCapture(select->onFalse.get(), lambdaVars);
else if (const FuCallExpr *call = dynamic_cast<const FuCallExpr *>(expr))
return hasLambdaCapture(call->method.get()) || std::any_of(call->arguments.begin(), call->arguments.end(), [](const std::shared_ptr<FuExpr> &arg) { return hasLambdaCapture(arg.get()); });
return hasLambdaCapture(call->method.get(), lambdaVars) || std::any_of(call->arguments.begin(), call->arguments.end(), [&](const std::shared_ptr<FuExpr> &arg) { return hasLambdaCapture(arg.get(), lambdaVars); });
else
std::abort();
}

void GenCpp::visitLambdaExpr(const FuLambdaExpr * expr)
{
writeChar('[');
if (hasLambdaCapture(expr->body.get()))
std::vector<const FuVar *> lambdaVars;
const FuVar * it = static_cast<const FuVar *>(expr->first);
lambdaVars.push_back(it);
if (hasLambdaCapture(expr->body.get(), &lambdaVars))
writeChar('&');
write("](");
if (dynamic_cast<const FuOwningType *>(expr->first->type.get()) || expr->first->type->id == FuId::stringStorageType) {
Expand Down
37 changes: 26 additions & 11 deletions libfut.cs
Original file line number Diff line number Diff line change
Expand Up @@ -15431,30 +15431,42 @@ internal override void VisitBinaryExpr(FuBinaryExpr expr, FuPriority parent)
base.VisitBinaryExpr(expr, parent);
}

static bool HasLambdaCapture(FuExpr expr)
static bool HasLambdaCapture(FuExpr expr, List<FuVar> lambdaVars)
{
switch (expr) {
case FuAggregateInitializer init:
return init.Items.Exists(item => HasLambdaCapture(item));
return init.Items.Exists(item => HasLambdaCapture(item, lambdaVars));
case FuLiteral:
case FuLambdaExpr:
return false;
case FuInterpolatedString interp:
return interp.Parts.Exists(part => HasLambdaCapture(part.Argument));
return interp.Parts.Exists(part => HasLambdaCapture(part.Argument, lambdaVars));
case FuSymbolReference symbol:
if (symbol.Left != null)
return HasLambdaCapture(symbol.Left);
if (symbol.Symbol is FuMember member)
return HasLambdaCapture(symbol.Left, lambdaVars);
switch (symbol.Symbol) {
case FuMember member:
return !member.IsStatic();
return !(symbol.Symbol.Parent is FuLambdaExpr);
case FuVar def:
return !lambdaVars.Contains(def);
default:
return false;
}
case FuUnaryExpr unary:
return unary.Inner != null && HasLambdaCapture(unary.Inner);
return unary.Inner != null && HasLambdaCapture(unary.Inner, lambdaVars);
case FuBinaryExpr binary:
return HasLambdaCapture(binary.Left) || HasLambdaCapture(binary.Right);
if (HasLambdaCapture(binary.Left, lambdaVars))
return true;
if (binary.Op == FuToken.Is) {
if (binary.Right is FuVar def)
lambdaVars.Add(def);
return false;
}
return HasLambdaCapture(binary.Right, lambdaVars);
case FuSelectExpr select:
return HasLambdaCapture(select.Cond) || HasLambdaCapture(select.OnTrue) || HasLambdaCapture(select.OnFalse);
return HasLambdaCapture(select.Cond, lambdaVars) || HasLambdaCapture(select.OnTrue, lambdaVars) || HasLambdaCapture(select.OnFalse, lambdaVars);
case FuCallExpr call:
return HasLambdaCapture(call.Method) || call.Arguments.Exists(arg => HasLambdaCapture(arg));
return HasLambdaCapture(call.Method, lambdaVars) || call.Arguments.Exists(arg => HasLambdaCapture(arg, lambdaVars));
default:
throw new NotImplementedException();
}
Expand All @@ -15463,7 +15475,10 @@ static bool HasLambdaCapture(FuExpr expr)
internal override void VisitLambdaExpr(FuLambdaExpr expr)
{
WriteChar('[');
if (HasLambdaCapture(expr.Body))
List<FuVar> lambdaVars = new List<FuVar>();
FuVar it = (FuVar) expr.First;
lambdaVars.Add(it);
if (HasLambdaCapture(expr.Body, lambdaVars))
WriteChar('&');
Write("](");
if (expr.First.Type is FuOwningType || expr.First.Type.Id == FuId.StringStorageType) {
Expand Down
2 changes: 1 addition & 1 deletion libfut.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -2452,7 +2452,7 @@ class GenCpp : public GenCCpp
void writeMatchProperty(const FuSymbolReference * expr, std::string_view name);
void writeGtRawPtr(const FuExpr * expr);
void writeIsVar(const FuExpr * expr, const FuVar * def, FuPriority parent);
static bool hasLambdaCapture(const FuExpr * expr);
static bool hasLambdaCapture(const FuExpr * expr, std::vector<const FuVar *> * lambdaVars);
static bool isIsVar(const FuExpr * expr);
bool hasVariables(const FuStatement * statement) const;
void openNamespace();
Expand Down
41 changes: 29 additions & 12 deletions libfut.js
Original file line number Diff line number Diff line change
Expand Up @@ -15907,42 +15907,56 @@ export class GenCpp extends GenCCpp
super.visitBinaryExpr(expr, parent);
}

static #hasLambdaCapture(expr)
static #hasLambdaCapture(expr, lambdaVars)
{
if (expr instanceof FuAggregateInitializer) {
const init = expr;
return init.items.some(item => GenCpp.#hasLambdaCapture(item));
return init.items.some(item => GenCpp.#hasLambdaCapture(item, lambdaVars));
}
else if (expr instanceof FuLiteral || expr instanceof FuLambdaExpr)
return false;
else if (expr instanceof FuInterpolatedString) {
const interp = expr;
return interp.parts.some(part => GenCpp.#hasLambdaCapture(part.argument));
return interp.parts.some(part => GenCpp.#hasLambdaCapture(part.argument, lambdaVars));
}
else if (expr instanceof FuSymbolReference) {
const symbol = expr;
if (symbol.left != null)
return GenCpp.#hasLambdaCapture(symbol.left);
let member;
if ((member = symbol.symbol) instanceof FuMember)
return GenCpp.#hasLambdaCapture(symbol.left, lambdaVars);
if (symbol.symbol instanceof FuMember) {
const member = symbol.symbol;
return !member.isStatic();
return !(symbol.symbol.parent instanceof FuLambdaExpr);
}
else if (symbol.symbol instanceof FuVar) {
const def = symbol.symbol;
return !lambdaVars.includes(def);
}
else
return false;
}
else if (expr instanceof FuUnaryExpr) {
const unary = expr;
return unary.inner != null && GenCpp.#hasLambdaCapture(unary.inner);
return unary.inner != null && GenCpp.#hasLambdaCapture(unary.inner, lambdaVars);
}
else if (expr instanceof FuBinaryExpr) {
const binary = expr;
return GenCpp.#hasLambdaCapture(binary.left) || GenCpp.#hasLambdaCapture(binary.right);
if (GenCpp.#hasLambdaCapture(binary.left, lambdaVars))
return true;
if (binary.op == FuToken.IS) {
let def;
if ((def = binary.right) instanceof FuVar)
lambdaVars.push(def);
return false;
}
return GenCpp.#hasLambdaCapture(binary.right, lambdaVars);
}
else if (expr instanceof FuSelectExpr) {
const select = expr;
return GenCpp.#hasLambdaCapture(select.cond) || GenCpp.#hasLambdaCapture(select.onTrue) || GenCpp.#hasLambdaCapture(select.onFalse);
return GenCpp.#hasLambdaCapture(select.cond, lambdaVars) || GenCpp.#hasLambdaCapture(select.onTrue, lambdaVars) || GenCpp.#hasLambdaCapture(select.onFalse, lambdaVars);
}
else if (expr instanceof FuCallExpr) {
const call = expr;
return GenCpp.#hasLambdaCapture(call.method) || call.arguments_.some(arg => GenCpp.#hasLambdaCapture(arg));
return GenCpp.#hasLambdaCapture(call.method, lambdaVars) || call.arguments_.some(arg => GenCpp.#hasLambdaCapture(arg, lambdaVars));
}
else
throw new Error();
Expand All @@ -15951,7 +15965,10 @@ export class GenCpp extends GenCCpp
visitLambdaExpr(expr)
{
this.writeChar(91);
if (GenCpp.#hasLambdaCapture(expr.body))
const lambdaVars = [];
let it = expr.first;
lambdaVars.push(it);
if (GenCpp.#hasLambdaCapture(expr.body, lambdaVars))
this.writeChar(38);
this.write("](");
if (expr.first.type instanceof FuOwningType || expr.first.type.id == FuId.STRING_STORAGE_TYPE) {
Expand Down

0 comments on commit 863faca

Please sign in to comment.