13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021-2024 Huawei Device Co., Ltd.
33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License.
53af6ab5fSopenharmony_ci * You may obtain a copy of the License at
63af6ab5fSopenharmony_ci *
73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
83af6ab5fSopenharmony_ci *
93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and
133af6ab5fSopenharmony_ci * limitations under the License.
143af6ab5fSopenharmony_ci */
153af6ab5fSopenharmony_ci
163af6ab5fSopenharmony_ci#include "ir/expressions/assignmentExpression.h"
173af6ab5fSopenharmony_ci#include "ir/expressions/binaryExpression.h"
183af6ab5fSopenharmony_ci#include "ir/expressions/memberExpression.h"
193af6ab5fSopenharmony_ci#include "ir/expressions/templateLiteral.h"
203af6ab5fSopenharmony_ci#include "ir/ts/tsQualifiedName.h"
213af6ab5fSopenharmony_ci
223af6ab5fSopenharmony_ci#include "checker/TSchecker.h"
233af6ab5fSopenharmony_ci
243af6ab5fSopenharmony_cinamespace ark::es2panda::checker {
253af6ab5fSopenharmony_ciir::MemberExpression *TSChecker::ResolveLeftMostMemberExpression(ir::MemberExpression *expr)
263af6ab5fSopenharmony_ci{
273af6ab5fSopenharmony_ci    ir::MemberExpression *iter = expr;
283af6ab5fSopenharmony_ci
293af6ab5fSopenharmony_ci    while (iter->Object()->IsMemberExpression()) {
303af6ab5fSopenharmony_ci        iter = iter->Object()->AsMemberExpression();
313af6ab5fSopenharmony_ci    }
323af6ab5fSopenharmony_ci
333af6ab5fSopenharmony_ci    return iter;
343af6ab5fSopenharmony_ci}
353af6ab5fSopenharmony_ci
363af6ab5fSopenharmony_cibool TSChecker::InAssignment(ir::AstNode *node)
373af6ab5fSopenharmony_ci{
383af6ab5fSopenharmony_ci    ir::AstNode *parent = node;
393af6ab5fSopenharmony_ci
403af6ab5fSopenharmony_ci    while (parent->Parent() != nullptr) {
413af6ab5fSopenharmony_ci        if (parent->Parent()->IsAssignmentExpression()) {
423af6ab5fSopenharmony_ci            return parent->Parent()->AsAssignmentExpression()->Left() == parent;
433af6ab5fSopenharmony_ci        }
443af6ab5fSopenharmony_ci
453af6ab5fSopenharmony_ci        if (parent->Parent()->IsBinaryExpression()) {
463af6ab5fSopenharmony_ci            ir::BinaryExpression *binaryExpr = parent->Parent()->AsBinaryExpression();
473af6ab5fSopenharmony_ci            return IsAssignmentOperator(binaryExpr->OperatorType()) && binaryExpr->Left() == parent;
483af6ab5fSopenharmony_ci        }
493af6ab5fSopenharmony_ci
503af6ab5fSopenharmony_ci        if (parent->Parent()->IsUpdateExpression()) {
513af6ab5fSopenharmony_ci            return true;
523af6ab5fSopenharmony_ci        }
533af6ab5fSopenharmony_ci
543af6ab5fSopenharmony_ci        parent = parent->Parent();
553af6ab5fSopenharmony_ci    }
563af6ab5fSopenharmony_ci    return false;
573af6ab5fSopenharmony_ci}
583af6ab5fSopenharmony_ci
593af6ab5fSopenharmony_cibool TSChecker::IsAssignmentOperator(lexer::TokenType op)
603af6ab5fSopenharmony_ci{
613af6ab5fSopenharmony_ci    switch (op) {
623af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_LESS_THAN_EQUAL:
633af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_GREATER_THAN_EQUAL:
643af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_MULTIPLY_EQUAL:
653af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_EXPONENTIATION_EQUAL:
663af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_DIVIDE_EQUAL:
673af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_MOD_EQUAL:
683af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_MINUS_EQUAL:
693af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_LEFT_SHIFT_EQUAL:
703af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_RIGHT_SHIFT_EQUAL:
713af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_UNSIGNED_RIGHT_SHIFT_EQUAL:
723af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_BITWISE_AND_EQUAL:
733af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_BITWISE_OR_EQUAL:
743af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_BITWISE_XOR_EQUAL:
753af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_PLUS_EQUAL:
763af6ab5fSopenharmony_ci        case lexer::TokenType::PUNCTUATOR_SUBSTITUTION:
773af6ab5fSopenharmony_ci            return true;
783af6ab5fSopenharmony_ci        default:
793af6ab5fSopenharmony_ci            return false;
803af6ab5fSopenharmony_ci    }
813af6ab5fSopenharmony_ci}
823af6ab5fSopenharmony_ci
833af6ab5fSopenharmony_cibool TSChecker::IsLiteralType(const Type *type)
843af6ab5fSopenharmony_ci{
853af6ab5fSopenharmony_ci    if (type->IsBooleanType()) {
863af6ab5fSopenharmony_ci        return true;
873af6ab5fSopenharmony_ci    }
883af6ab5fSopenharmony_ci
893af6ab5fSopenharmony_ci    if (type->IsUnionType()) {
903af6ab5fSopenharmony_ci        auto &constituentTypes = type->AsUnionType()->ConstituentTypes();
913af6ab5fSopenharmony_ci        bool result = true;
923af6ab5fSopenharmony_ci        for (auto *it : constituentTypes) {
933af6ab5fSopenharmony_ci            result &= it->HasTypeFlag(TypeFlag::UNIT);
943af6ab5fSopenharmony_ci        }
953af6ab5fSopenharmony_ci        return result;
963af6ab5fSopenharmony_ci    }
973af6ab5fSopenharmony_ci
983af6ab5fSopenharmony_ci    return type->HasTypeFlag(TypeFlag::UNIT);
993af6ab5fSopenharmony_ci}
1003af6ab5fSopenharmony_ci
1013af6ab5fSopenharmony_ciir::AstNode *TSChecker::FindAncestorUntilGivenType(ir::AstNode *node, ir::AstNodeType stop)
1023af6ab5fSopenharmony_ci{
1033af6ab5fSopenharmony_ci    while (node->Parent()->Type() != stop) {
1043af6ab5fSopenharmony_ci        if (node->Parent() != nullptr) {
1053af6ab5fSopenharmony_ci            node = node->Parent();
1063af6ab5fSopenharmony_ci            continue;
1073af6ab5fSopenharmony_ci        }
1083af6ab5fSopenharmony_ci
1093af6ab5fSopenharmony_ci        return nullptr;
1103af6ab5fSopenharmony_ci    }
1113af6ab5fSopenharmony_ci
1123af6ab5fSopenharmony_ci    return node;
1133af6ab5fSopenharmony_ci}
1143af6ab5fSopenharmony_ci
1153af6ab5fSopenharmony_cibool TSChecker::MaybeTypeOfKind(const Type *type, TypeFlag flags)
1163af6ab5fSopenharmony_ci{
1173af6ab5fSopenharmony_ci    if (type->HasTypeFlag(flags)) {
1183af6ab5fSopenharmony_ci        return true;
1193af6ab5fSopenharmony_ci    }
1203af6ab5fSopenharmony_ci
1213af6ab5fSopenharmony_ci    if (type->HasTypeFlag(TypeFlag::UNION_OR_INTERSECTION) && type->IsUnionType()) {
1223af6ab5fSopenharmony_ci        const auto &constituentTypes = type->AsUnionType()->ConstituentTypes();
1233af6ab5fSopenharmony_ci        for (auto *it : constituentTypes) {
1243af6ab5fSopenharmony_ci            if (MaybeTypeOfKind(it, flags)) {
1253af6ab5fSopenharmony_ci                return true;
1263af6ab5fSopenharmony_ci            }
1273af6ab5fSopenharmony_ci        }
1283af6ab5fSopenharmony_ci    }
1293af6ab5fSopenharmony_ci
1303af6ab5fSopenharmony_ci    return false;
1313af6ab5fSopenharmony_ci}
1323af6ab5fSopenharmony_ci
1333af6ab5fSopenharmony_cibool TSChecker::MaybeTypeOfKind(const Type *type, ObjectType::ObjectTypeKind kind)
1343af6ab5fSopenharmony_ci{
1353af6ab5fSopenharmony_ci    if (type->IsObjectType() && type->AsObjectType()->Kind() == kind) {
1363af6ab5fSopenharmony_ci        return true;
1373af6ab5fSopenharmony_ci    }
1383af6ab5fSopenharmony_ci
1393af6ab5fSopenharmony_ci    if (type->HasTypeFlag(TypeFlag::UNION_OR_INTERSECTION) && type->IsUnionType()) {
1403af6ab5fSopenharmony_ci        const auto &constituentTypes = type->AsUnionType()->ConstituentTypes();
1413af6ab5fSopenharmony_ci        for (auto *it : constituentTypes) {
1423af6ab5fSopenharmony_ci            if (MaybeTypeOfKind(it, kind)) {
1433af6ab5fSopenharmony_ci                return true;
1443af6ab5fSopenharmony_ci            }
1453af6ab5fSopenharmony_ci        }
1463af6ab5fSopenharmony_ci    }
1473af6ab5fSopenharmony_ci
1483af6ab5fSopenharmony_ci    return false;
1493af6ab5fSopenharmony_ci}
1503af6ab5fSopenharmony_ci
1513af6ab5fSopenharmony_cibool TSChecker::IsConstantMemberAccess(ir::Expression *expr)
1523af6ab5fSopenharmony_ci{
1533af6ab5fSopenharmony_ci    switch (expr->Type()) {
1543af6ab5fSopenharmony_ci        case ir::AstNodeType::IDENTIFIER: {
1553af6ab5fSopenharmony_ci            return true;
1563af6ab5fSopenharmony_ci        }
1573af6ab5fSopenharmony_ci        case ir::AstNodeType::MEMBER_EXPRESSION: {
1583af6ab5fSopenharmony_ci            bool res = IsConstantMemberAccess(expr->AsMemberExpression()->Object());
1593af6ab5fSopenharmony_ci            return !expr->AsMemberExpression()->IsComputed()
1603af6ab5fSopenharmony_ci                       ? res
1613af6ab5fSopenharmony_ci                       : (res && IsStringLike(expr->AsMemberExpression()->Property()));
1623af6ab5fSopenharmony_ci        }
1633af6ab5fSopenharmony_ci        default:
1643af6ab5fSopenharmony_ci            return false;
1653af6ab5fSopenharmony_ci    }
1663af6ab5fSopenharmony_ci}
1673af6ab5fSopenharmony_ci
1683af6ab5fSopenharmony_cibool TSChecker::IsStringLike(ir::Expression *expr)
1693af6ab5fSopenharmony_ci{
1703af6ab5fSopenharmony_ci    if (expr->IsStringLiteral()) {
1713af6ab5fSopenharmony_ci        return true;
1723af6ab5fSopenharmony_ci    }
1733af6ab5fSopenharmony_ci
1743af6ab5fSopenharmony_ci    if (expr->IsTemplateLiteral() && expr->AsTemplateLiteral()->Quasis().empty()) {
1753af6ab5fSopenharmony_ci        return true;
1763af6ab5fSopenharmony_ci    }
1773af6ab5fSopenharmony_ci
1783af6ab5fSopenharmony_ci    return false;
1793af6ab5fSopenharmony_ci}
1803af6ab5fSopenharmony_ci}  // namespace ark::es2panda::checker
181