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#ifndef ES2PANDA_UTIL_HELPERS_H
173af6ab5fSopenharmony_ci#define ES2PANDA_UTIL_HELPERS_H
183af6ab5fSopenharmony_ci
193af6ab5fSopenharmony_ci#include "varbinder/variableFlags.h"
203af6ab5fSopenharmony_ci#include "mem/pool_manager.h"
213af6ab5fSopenharmony_ci#include "util/ustring.h"
223af6ab5fSopenharmony_ci#include "ir/module/importSpecifier.h"
233af6ab5fSopenharmony_ci#include "parser/program/program.h"
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_ci#include <cmath>
263af6ab5fSopenharmony_ci#include <string>
273af6ab5fSopenharmony_ci
283af6ab5fSopenharmony_cinamespace ark::es2panda::varbinder {
293af6ab5fSopenharmony_ciclass Variable;
303af6ab5fSopenharmony_ci}  // namespace ark::es2panda::varbinder
313af6ab5fSopenharmony_ci
323af6ab5fSopenharmony_cinamespace ark::es2panda::checker {
333af6ab5fSopenharmony_ciclass ETSObjectType;
343af6ab5fSopenharmony_ciclass Type;
353af6ab5fSopenharmony_ci}  // namespace ark::es2panda::checker
363af6ab5fSopenharmony_ci
373af6ab5fSopenharmony_cinamespace ark::es2panda::compiler {
383af6ab5fSopenharmony_ciclass Literal;
393af6ab5fSopenharmony_ci}  // namespace ark::es2panda::compiler
403af6ab5fSopenharmony_ci
413af6ab5fSopenharmony_cinamespace ark::es2panda::ir {
423af6ab5fSopenharmony_ciclass Expression;
433af6ab5fSopenharmony_ciclass ScriptFunction;
443af6ab5fSopenharmony_ciclass ClassDefinition;
453af6ab5fSopenharmony_ciclass ClassProperty;
463af6ab5fSopenharmony_ciclass Identifier;
473af6ab5fSopenharmony_ciclass MethodDefinition;
483af6ab5fSopenharmony_ciclass AstNode;
493af6ab5fSopenharmony_ciclass ReturnStatement;
503af6ab5fSopenharmony_ciclass CallExpression;
513af6ab5fSopenharmony_ciclass ClassStaticBlock;
523af6ab5fSopenharmony_ciclass TSInterfaceDeclaration;
533af6ab5fSopenharmony_ciclass TSEnumDeclaration;
543af6ab5fSopenharmony_ciclass ETSImportDeclaration;
553af6ab5fSopenharmony_cienum class AstNodeType;
563af6ab5fSopenharmony_ci}  // namespace ark::es2panda::ir
573af6ab5fSopenharmony_ci
583af6ab5fSopenharmony_cinamespace ark::es2panda::util {
593af6ab5fSopenharmony_cienum class LogLevel : std::uint8_t {
603af6ab5fSopenharmony_ci    DEBUG,
613af6ab5fSopenharmony_ci    INFO,
623af6ab5fSopenharmony_ci    WARNING,
633af6ab5fSopenharmony_ci    ERROR,
643af6ab5fSopenharmony_ci    FATAL,
653af6ab5fSopenharmony_ci    INVALID,
663af6ab5fSopenharmony_ci};
673af6ab5fSopenharmony_ci
683af6ab5fSopenharmony_ciclass NodeAllocator {
693af6ab5fSopenharmony_cipublic:
703af6ab5fSopenharmony_ci    template <typename T, typename... Args>
713af6ab5fSopenharmony_ci    static T *NoSetParent(ArenaAllocator *alloc, Args &&...args)
723af6ab5fSopenharmony_ci    {
733af6ab5fSopenharmony_ci        return alloc->New<T>(std::forward<Args>(args)...);
743af6ab5fSopenharmony_ci    }
753af6ab5fSopenharmony_ci
763af6ab5fSopenharmony_ci    template <typename T, typename... Args>
773af6ab5fSopenharmony_ci    static T *ForceSetParent(ArenaAllocator *alloc, Args &&...args)
783af6ab5fSopenharmony_ci    {
793af6ab5fSopenharmony_ci        auto *ret = NoSetParent<T>(alloc, std::forward<Args>(args)...);
803af6ab5fSopenharmony_ci        if (ret == nullptr) {
813af6ab5fSopenharmony_ci            return nullptr;
823af6ab5fSopenharmony_ci        }
833af6ab5fSopenharmony_ci        ret->Iterate([ret](ir::AstNode *child) { child->SetParent(ret); });
843af6ab5fSopenharmony_ci        return ret;
853af6ab5fSopenharmony_ci    }
863af6ab5fSopenharmony_ci
873af6ab5fSopenharmony_ci    template <typename T, typename... Args>
883af6ab5fSopenharmony_ci    static T *Alloc(ArenaAllocator *alloc, Args &&...args)
893af6ab5fSopenharmony_ci    {
903af6ab5fSopenharmony_ci        auto *ret = NoSetParent<T>(alloc, std::forward<Args>(args)...);
913af6ab5fSopenharmony_ci        if (ret == nullptr) {
923af6ab5fSopenharmony_ci            return nullptr;
933af6ab5fSopenharmony_ci        }
943af6ab5fSopenharmony_ci        ret->Iterate([ret](ir::AstNode *child) {
953af6ab5fSopenharmony_ci            ASSERT(child->Parent() == nullptr);
963af6ab5fSopenharmony_ci            child->SetParent(ret);
973af6ab5fSopenharmony_ci        });
983af6ab5fSopenharmony_ci        return ret;
993af6ab5fSopenharmony_ci    }
1003af6ab5fSopenharmony_ci};
1013af6ab5fSopenharmony_ci
1023af6ab5fSopenharmony_ciclass Helpers {
1033af6ab5fSopenharmony_cipublic:
1043af6ab5fSopenharmony_ci    Helpers() = delete;
1053af6ab5fSopenharmony_ci
1063af6ab5fSopenharmony_ci    static bool IsGlobalIdentifier(const util::StringView &str);
1073af6ab5fSopenharmony_ci    static bool ContainSpreadElement(const ArenaVector<ir::Expression *> &args);
1083af6ab5fSopenharmony_ci    static util::StringView LiteralToPropName(const ir::Expression *lit);
1093af6ab5fSopenharmony_ci
1103af6ab5fSopenharmony_ci    template <typename T>
1113af6ab5fSopenharmony_ci    static bool IsInteger(double number);
1123af6ab5fSopenharmony_ci    static bool IsIndex(double number);
1133af6ab5fSopenharmony_ci    static int64_t GetIndex(const util::StringView &str);
1143af6ab5fSopenharmony_ci
1153af6ab5fSopenharmony_ci    static std::string ToString(double number);
1163af6ab5fSopenharmony_ci    static util::StringView ToStringView(ArenaAllocator *allocator, double number);
1173af6ab5fSopenharmony_ci    static util::StringView ToStringView(ArenaAllocator *allocator, int32_t number);
1183af6ab5fSopenharmony_ci    static util::StringView ToStringView(ArenaAllocator *allocator, uint32_t number);
1193af6ab5fSopenharmony_ci    static bool EndsWith(const std::string &str, const std::string &suffix);
1203af6ab5fSopenharmony_ci
1213af6ab5fSopenharmony_ci    static const ir::ScriptFunction *GetContainingConstructor(const ir::AstNode *node);
1223af6ab5fSopenharmony_ci    static const ir::ScriptFunction *GetContainingConstructor(const ir::ClassProperty *node);
1233af6ab5fSopenharmony_ci
1243af6ab5fSopenharmony_ci    template <typename T,
1253af6ab5fSopenharmony_ci              typename U = std::enable_if_t<
1263af6ab5fSopenharmony_ci                  std::is_convertible_v<std::remove_const_t<std::remove_pointer_t<T>> *, ir::AstNode *>,
1273af6ab5fSopenharmony_ci                  std::conditional_t<std::is_const_v<std::remove_pointer_t<T>>, const ir::AstNode *, ir::AstNode *>>>
1283af6ab5fSopenharmony_ci    static U FindAncestorGivenByType(T node, ir::AstNodeType type)
1293af6ab5fSopenharmony_ci    {
1303af6ab5fSopenharmony_ci        U iter = node->Parent();
1313af6ab5fSopenharmony_ci
1323af6ab5fSopenharmony_ci        while (iter->Type() != type) {
1333af6ab5fSopenharmony_ci            if (iter->Parent() != nullptr) {
1343af6ab5fSopenharmony_ci                iter = iter->Parent();
1353af6ab5fSopenharmony_ci                continue;
1363af6ab5fSopenharmony_ci            }
1373af6ab5fSopenharmony_ci
1383af6ab5fSopenharmony_ci            return nullptr;
1393af6ab5fSopenharmony_ci        }
1403af6ab5fSopenharmony_ci
1413af6ab5fSopenharmony_ci        return iter;
1423af6ab5fSopenharmony_ci    }
1433af6ab5fSopenharmony_ci
1443af6ab5fSopenharmony_ci    static const checker::ETSObjectType *GetContainingObjectType(const ir::AstNode *node);
1453af6ab5fSopenharmony_ci    static const ir::TSEnumDeclaration *GetContainingEnumDeclaration(const ir::AstNode *node);
1463af6ab5fSopenharmony_ci    static const ir::ClassDefinition *GetContainingClassDefinition(const ir::AstNode *node);
1473af6ab5fSopenharmony_ci    static const ir::TSInterfaceDeclaration *GetContainingInterfaceDeclaration(const ir::AstNode *node);
1483af6ab5fSopenharmony_ci    static const ir::MethodDefinition *GetContainingClassMethodDefinition(const ir::AstNode *node);
1493af6ab5fSopenharmony_ci    static const ir::ClassStaticBlock *GetContainingClassStaticBlock(const ir::AstNode *node);
1503af6ab5fSopenharmony_ci    static const ir::ScriptFunction *GetContainingFunction(const ir::AstNode *node);
1513af6ab5fSopenharmony_ci    static const ir::ClassDefinition *GetClassDefiniton(const ir::ScriptFunction *node);
1523af6ab5fSopenharmony_ci    static bool IsSpecialPropertyKey(const ir::Expression *expr);
1533af6ab5fSopenharmony_ci    static bool IsConstantPropertyKey(const ir::Expression *expr, bool isComputed);
1543af6ab5fSopenharmony_ci    static compiler::Literal ToConstantLiteral(const ir::Expression *expr);
1553af6ab5fSopenharmony_ci    static bool IsBindingPattern(const ir::AstNode *node);
1563af6ab5fSopenharmony_ci    static bool IsPattern(const ir::AstNode *node);
1573af6ab5fSopenharmony_ci    static std::vector<ir::Identifier *> CollectBindingNames(ir::AstNode *node);
1583af6ab5fSopenharmony_ci    static util::StringView FunctionName(ArenaAllocator *allocator, const ir::ScriptFunction *func);
1593af6ab5fSopenharmony_ci    static void CheckImportedName(const ArenaVector<ir::ImportSpecifier *> &specifiers,
1603af6ab5fSopenharmony_ci                                  const ir::ImportSpecifier *specifier, const std::string &fileName);
1613af6ab5fSopenharmony_ci    static void CheckDefaultImportedName(const ArenaVector<ir::ImportDefaultSpecifier *> &specifiers,
1623af6ab5fSopenharmony_ci                                         const ir::ImportDefaultSpecifier *specifier, const std::string &fileName);
1633af6ab5fSopenharmony_ci    static void CheckDefaultImport(const ArenaVector<ir::ETSImportDeclaration *> &statements);
1643af6ab5fSopenharmony_ci    static std::tuple<util::StringView, bool> ParamName(ArenaAllocator *allocator, const ir::AstNode *param,
1653af6ab5fSopenharmony_ci                                                        uint32_t index);
1663af6ab5fSopenharmony_ci
1673af6ab5fSopenharmony_ci    template <typename T, typename V>
1683af6ab5fSopenharmony_ci    static ArenaVector<T *> ConvertVector(const ArenaVector<V *> &src)
1693af6ab5fSopenharmony_ci    {
1703af6ab5fSopenharmony_ci        ArenaVector<T *> dst(src.begin(), src.end(), src.get_allocator());
1713af6ab5fSopenharmony_ci        return dst;
1723af6ab5fSopenharmony_ci    }
1733af6ab5fSopenharmony_ci
1743af6ab5fSopenharmony_ci    template <typename Source, typename Target>
1753af6ab5fSopenharmony_ci    static bool IsTargetFitInSourceRange(Target target)
1763af6ab5fSopenharmony_ci    {
1773af6ab5fSopenharmony_ci        if (!std::isfinite(target)) {
1783af6ab5fSopenharmony_ci            return true;
1793af6ab5fSopenharmony_ci        }
1803af6ab5fSopenharmony_ci
1813af6ab5fSopenharmony_ci        // NOLINTNEXTLINE(misc-redundant-expression)
1823af6ab5fSopenharmony_ci        return target >= std::numeric_limits<Source>::lowest() &&
1833af6ab5fSopenharmony_ci               target <= static_cast<Target>(std::numeric_limits<Source>::max());
1843af6ab5fSopenharmony_ci    }
1853af6ab5fSopenharmony_ci
1863af6ab5fSopenharmony_ci    static const uint32_t INVALID_INDEX = 4294967295L;
1873af6ab5fSopenharmony_ci
1883af6ab5fSopenharmony_ci    static std::string CreateEscapedString(const std::string &str);
1893af6ab5fSopenharmony_ci    static std::string UTF16toUTF8(const char16_t c);
1903af6ab5fSopenharmony_ci
1913af6ab5fSopenharmony_ci    template <typename... Elements>
1923af6ab5fSopenharmony_ci    static void LogDebug(Elements &&...elems);
1933af6ab5fSopenharmony_ci    template <typename... Elements>
1943af6ab5fSopenharmony_ci    static void LogInfo(Elements &&...elems);
1953af6ab5fSopenharmony_ci    template <typename... Elements>
1963af6ab5fSopenharmony_ci    static void LogWarning(Elements &&...elems);
1973af6ab5fSopenharmony_ci    template <typename... Elements>
1983af6ab5fSopenharmony_ci    static void LogError(Elements &&...elems);
1993af6ab5fSopenharmony_ci    template <typename... Elements>
2003af6ab5fSopenharmony_ci    static void LogFatal(Elements &&...elems);
2013af6ab5fSopenharmony_ci
2023af6ab5fSopenharmony_ci    template <typename... Elements>
2033af6ab5fSopenharmony_ci    static std::string AppendAll(Elements &&...elems);
2043af6ab5fSopenharmony_ci
2053af6ab5fSopenharmony_ci    static std::pair<std::string_view, std::string_view> SplitSignature(std::string_view signature);
2063af6ab5fSopenharmony_ci
2073af6ab5fSopenharmony_ci    static std::vector<std::string> &StdLib();
2083af6ab5fSopenharmony_ci    static bool IsStdLib(const parser::Program *program);
2093af6ab5fSopenharmony_ci
2103af6ab5fSopenharmony_ciprivate:
2113af6ab5fSopenharmony_ci    template <LogLevel LOG_L, typename... Elements>
2123af6ab5fSopenharmony_ci    static void Log(Elements &&...elems);
2133af6ab5fSopenharmony_ci};
2143af6ab5fSopenharmony_ci
2153af6ab5fSopenharmony_citemplate <typename T>
2163af6ab5fSopenharmony_cibool Helpers::IsInteger(double number)
2173af6ab5fSopenharmony_ci{
2183af6ab5fSopenharmony_ci    if (std::fabs(number) <= static_cast<double>(std::numeric_limits<T>::max())) {
2193af6ab5fSopenharmony_ci        T intNum = static_cast<T>(number);
2203af6ab5fSopenharmony_ci
2213af6ab5fSopenharmony_ci        if (static_cast<double>(intNum) == number) {
2223af6ab5fSopenharmony_ci            return true;
2233af6ab5fSopenharmony_ci        }
2243af6ab5fSopenharmony_ci    }
2253af6ab5fSopenharmony_ci
2263af6ab5fSopenharmony_ci    return false;
2273af6ab5fSopenharmony_ci}
2283af6ab5fSopenharmony_ci
2293af6ab5fSopenharmony_citemplate <LogLevel LOG_L, typename... Elements>
2303af6ab5fSopenharmony_civoid Helpers::Log(Elements &&...elems)
2313af6ab5fSopenharmony_ci{
2323af6ab5fSopenharmony_ci    constexpr auto ES2PANDA = ark::Logger::Component::ES2PANDA;
2333af6ab5fSopenharmony_ci    constexpr auto LOG_LEVEL = []() {
2343af6ab5fSopenharmony_ci        switch (LOG_L) {
2353af6ab5fSopenharmony_ci            case LogLevel::DEBUG: {
2363af6ab5fSopenharmony_ci                return ark::Logger::Level::DEBUG;
2373af6ab5fSopenharmony_ci            }
2383af6ab5fSopenharmony_ci            case LogLevel::INFO: {
2393af6ab5fSopenharmony_ci                return ark::Logger::Level::INFO;
2403af6ab5fSopenharmony_ci            }
2413af6ab5fSopenharmony_ci            case LogLevel::WARNING: {
2423af6ab5fSopenharmony_ci                return ark::Logger::Level::WARNING;
2433af6ab5fSopenharmony_ci            }
2443af6ab5fSopenharmony_ci            case LogLevel::ERROR: {
2453af6ab5fSopenharmony_ci                return ark::Logger::Level::ERROR;
2463af6ab5fSopenharmony_ci            }
2473af6ab5fSopenharmony_ci            case LogLevel::FATAL: {
2483af6ab5fSopenharmony_ci                return ark::Logger::Level::FATAL;
2493af6ab5fSopenharmony_ci            }
2503af6ab5fSopenharmony_ci            default: {
2513af6ab5fSopenharmony_ci                UNREACHABLE_CONSTEXPR();
2523af6ab5fSopenharmony_ci            }
2533af6ab5fSopenharmony_ci        }
2543af6ab5fSopenharmony_ci    }();
2553af6ab5fSopenharmony_ci
2563af6ab5fSopenharmony_ci#ifndef NDEBUG
2573af6ab5fSopenharmony_ci    const bool isMessageSuppressed = ark::Logger::IsMessageSuppressed(LOG_LEVEL, ES2PANDA);
2583af6ab5fSopenharmony_ci#else
2593af6ab5fSopenharmony_ci    const bool isMessageSuppressed = false;
2603af6ab5fSopenharmony_ci#endif
2613af6ab5fSopenharmony_ci    if (!ark::Logger::IsLoggingOnOrAbort(LOG_LEVEL, ES2PANDA) || isMessageSuppressed) {
2623af6ab5fSopenharmony_ci        return;
2633af6ab5fSopenharmony_ci    }
2643af6ab5fSopenharmony_ci
2653af6ab5fSopenharmony_ci    (ark::Logger::Message(LOG_LEVEL, ES2PANDA, false).GetStream() << ... << std::forward<Elements>(elems));
2663af6ab5fSopenharmony_ci}
2673af6ab5fSopenharmony_ci
2683af6ab5fSopenharmony_citemplate <typename... Elements>
2693af6ab5fSopenharmony_civoid Helpers::LogDebug(Elements &&...elems)
2703af6ab5fSopenharmony_ci{
2713af6ab5fSopenharmony_ci    Helpers::Log<LogLevel::DEBUG>(std::forward<Elements>(elems)...);
2723af6ab5fSopenharmony_ci}
2733af6ab5fSopenharmony_ci
2743af6ab5fSopenharmony_citemplate <typename... Elements>
2753af6ab5fSopenharmony_civoid Helpers::LogInfo(Elements &&...elems)
2763af6ab5fSopenharmony_ci{
2773af6ab5fSopenharmony_ci    Helpers::Log<LogLevel::INFO>(std::forward<Elements>(elems)...);
2783af6ab5fSopenharmony_ci}
2793af6ab5fSopenharmony_ci
2803af6ab5fSopenharmony_citemplate <typename... Elements>
2813af6ab5fSopenharmony_civoid Helpers::LogWarning(Elements &&...elems)
2823af6ab5fSopenharmony_ci{
2833af6ab5fSopenharmony_ci    Helpers::Log<LogLevel::WARNING>(std::forward<Elements>(elems)...);
2843af6ab5fSopenharmony_ci}
2853af6ab5fSopenharmony_ci
2863af6ab5fSopenharmony_citemplate <typename... Elements>
2873af6ab5fSopenharmony_civoid Helpers::LogError(Elements &&...elems)
2883af6ab5fSopenharmony_ci{
2893af6ab5fSopenharmony_ci    Helpers::Log<LogLevel::ERROR>(std::forward<Elements>(elems)...);
2903af6ab5fSopenharmony_ci}
2913af6ab5fSopenharmony_ci
2923af6ab5fSopenharmony_citemplate <typename... Elements>
2933af6ab5fSopenharmony_civoid Helpers::LogFatal(Elements &&...elems)
2943af6ab5fSopenharmony_ci{
2953af6ab5fSopenharmony_ci    Helpers::Log<LogLevel::FATAL>(std::forward<Elements>(elems)...);
2963af6ab5fSopenharmony_ci}
2973af6ab5fSopenharmony_ci
2983af6ab5fSopenharmony_citemplate <typename... Elements>
2993af6ab5fSopenharmony_cistd::string Helpers::AppendAll(Elements &&...elems)
3003af6ab5fSopenharmony_ci{
3013af6ab5fSopenharmony_ci    std::string ret {};
3023af6ab5fSopenharmony_ci    ((ret += std::forward<Elements>(elems)), ...);
3033af6ab5fSopenharmony_ci    return ret;
3043af6ab5fSopenharmony_ci}
3053af6ab5fSopenharmony_ci
3063af6ab5fSopenharmony_ci}  // namespace ark::es2panda::util
3073af6ab5fSopenharmony_ci
3083af6ab5fSopenharmony_ci#endif
309