Files
2025-05-18 13:04:45 +08:00

228 lines
4.3 KiB
C++

// Copyright Epic Games, Inc. All Rights Reserved.
#include "MuT/ASTOpBoolAnd.h"
#include "MuT/ASTOpConstantBool.h"
#include "HAL/PlatformMath.h"
#include "MuR/ModelPrivate.h"
#include "MuR/RefCounted.h"
#include "MuR/Types.h"
namespace mu
{
ASTOpBoolAnd::ASTOpBoolAnd()
: A(this)
, B(this)
{
}
ASTOpBoolAnd::~ASTOpBoolAnd()
{
// Explicit call needed to avoid recursive destruction
ASTOp::RemoveChildren();
}
bool ASTOpBoolAnd::IsEqual(const ASTOp& OtherUntyped) const
{
if (OtherUntyped.GetOpType()==GetOpType())
{
const ASTOpBoolAnd* Other = static_cast<const ASTOpBoolAnd*>(&OtherUntyped);
return A == Other->A &&
B == Other->B;
}
return false;
}
uint64 ASTOpBoolAnd::Hash() const
{
uint64 Result = std::hash<void*>()(A.child().get());
hash_combine(Result, B.child().get());
return Result;
}
Ptr<ASTOp> ASTOpBoolAnd::Clone(MapChildFuncRef MapChild) const
{
Ptr<ASTOpBoolAnd> New = new ASTOpBoolAnd();
New->A = MapChild(A.child());
New->B = MapChild(B.child());
return New;
}
void ASTOpBoolAnd::ForEachChild(const TFunctionRef<void(ASTChild&)> Func)
{
Func(A);
Func(B);
}
void ASTOpBoolAnd::Link(FProgram& Program, FLinkerOptions*)
{
// Already linked?
if (!linkedAddress)
{
OP::BoolBinaryArgs Args;
FMemory::Memzero(Args);
if (A) Args.A = A->linkedAddress;
if (B) Args.B = B->linkedAddress;
linkedAddress = (OP::ADDRESS)Program.OpAddress.Num();
Program.OpAddress.Add(Program.ByteCode.Num());
AppendCode(Program.ByteCode, GetOpType());
AppendCode(Program.ByteCode, Args);
}
}
ASTOp::FBoolEvalResult ASTOpBoolAnd::EvaluateBool(ASTOpList& Facts, FEvaluateBoolCache* Cache) const
{
FEvaluateBoolCache LocalCache;
if (!Cache)
{
Cache = &LocalCache;
}
else
{
// Is this in the cache?
FEvaluateBoolCache::iterator it = Cache->find(this);
if (it != Cache->end())
{
return it->second;
}
}
FBoolEvalResult Result = BET_UNKNOWN;
FBoolEvalResult resultA = BET_UNKNOWN;
FBoolEvalResult resultB = BET_UNKNOWN;
for (int32 f = 0; f < Facts.Num(); ++f)
{
if (A && resultA == BET_UNKNOWN)
{
resultA = A->EvaluateBool(Facts, Cache);
if (resultA == BET_TRUE && resultB == BET_TRUE)
{
Result = BET_TRUE;
break;
}
if (resultA == BET_FALSE || resultB == BET_FALSE)
{
Result = BET_FALSE;
break;
}
}
if (B && resultB == BET_UNKNOWN)
{
resultB = B->EvaluateBool(Facts, Cache);
if (resultA == BET_TRUE && resultB == BET_TRUE)
{
Result = BET_TRUE;
break;
}
if (resultA == BET_FALSE || resultB == BET_FALSE)
{
Result = BET_FALSE;
break;
}
}
}
(*Cache)[this] = Result;
return Result;
}
Ptr<ASTOp> ASTOpBoolAnd::OptimiseSemantic(const FModelOptimizationOptions&, int32 Pass) const
{
Ptr<ASTOp> Result;
bool bChanged = false;
if (!A)
{
Result = B.child();
bChanged = true;
}
else if (!B)
{
Result = A.child();
bChanged = true;
}
else if (A->GetOpType() == EOpType::BO_CONSTANT)
{
if (static_cast<const ASTOpConstantBool*>(A.child().get())->bValue)
{
Result = B.child();
bChanged = true;
}
else
{
Result = A.child();
bChanged = true;
}
}
else if (B->GetOpType() == EOpType::BO_CONSTANT)
{
if (static_cast<const ASTOpConstantBool*>(B.child().get())->bValue)
{
Result = A.child();
bChanged = true;
}
else
{
Result = B.child();
bChanged = true;
}
}
// Common cases of repeated branch in children
else if (A->GetOpType() == EOpType::BO_AND)
{
const ASTOpBoolAnd* TypedA = static_cast<const ASTOpBoolAnd*>(A.child().get());
if (TypedA->A.child() == B.child()
||
TypedA->B.child() == B.child())
{
Result = A.child();
bChanged = true;
}
}
else if (B->GetOpType() == EOpType::BO_AND)
{
const ASTOpBoolAnd* TypedB = static_cast<const ASTOpBoolAnd*>(B.child().get());
if (TypedB->B.child() == A.child()
||
TypedB->B.child() == B.child())
{
Result = B.child();
bChanged = true;
}
}
else if (A.child() == B.child() || *A.child() == *B.child())
{
Result = A.child();
bChanged = true;
}
// if it became null, it means true (neutral AND argument)
if (bChanged && !Result)
{
Result = new ASTOpConstantBool(true);
}
return Result;
}
}