13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021 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 "templateLiteral.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include <compiler/core/pandagen.h>
193af6ab5fSopenharmony_ci#include <typescript/checker.h>
203af6ab5fSopenharmony_ci#include <ir/astDump.h>
213af6ab5fSopenharmony_ci#include <ir/base/templateElement.h>
223af6ab5fSopenharmony_ci
233af6ab5fSopenharmony_cinamespace panda::es2panda::ir {
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_civoid TemplateLiteral::Iterate(const NodeTraverser &cb) const
263af6ab5fSopenharmony_ci{
273af6ab5fSopenharmony_ci    for (auto *it : expressions_) {
283af6ab5fSopenharmony_ci        cb(it);
293af6ab5fSopenharmony_ci    }
303af6ab5fSopenharmony_ci
313af6ab5fSopenharmony_ci    for (auto *it : quasis_) {
323af6ab5fSopenharmony_ci        cb(it);
333af6ab5fSopenharmony_ci    }
343af6ab5fSopenharmony_ci}
353af6ab5fSopenharmony_ci
363af6ab5fSopenharmony_civoid TemplateLiteral::Dump(ir::AstDumper *dumper) const
373af6ab5fSopenharmony_ci{
383af6ab5fSopenharmony_ci    dumper->Add({{"type", "TemplateLiteral"}, {"expressions", expressions_}, {"quasis", quasis_}});
393af6ab5fSopenharmony_ci}
403af6ab5fSopenharmony_ci
413af6ab5fSopenharmony_civoid TemplateLiteral::Compile(compiler::PandaGen *pg) const
423af6ab5fSopenharmony_ci{
433af6ab5fSopenharmony_ci    auto quasisIt = quasis_.begin();
443af6ab5fSopenharmony_ci    auto expressionIt = expressions_.begin();
453af6ab5fSopenharmony_ci
463af6ab5fSopenharmony_ci    pg->LoadAccumulatorString(this, (*quasisIt)->Cooked());
473af6ab5fSopenharmony_ci
483af6ab5fSopenharmony_ci    quasisIt++;
493af6ab5fSopenharmony_ci
503af6ab5fSopenharmony_ci    bool isQuais = false;
513af6ab5fSopenharmony_ci    size_t total = quasis_.size() + expressions_.size();
523af6ab5fSopenharmony_ci
533af6ab5fSopenharmony_ci    compiler::RegScope rs(pg);
543af6ab5fSopenharmony_ci    compiler::VReg lhs = pg->AllocReg();
553af6ab5fSopenharmony_ci
563af6ab5fSopenharmony_ci    while (total != 1) {
573af6ab5fSopenharmony_ci        const ir::AstNode *node = nullptr;
583af6ab5fSopenharmony_ci
593af6ab5fSopenharmony_ci        if (isQuais) {
603af6ab5fSopenharmony_ci            pg->StoreAccumulator(*quasisIt, lhs);
613af6ab5fSopenharmony_ci            pg->LoadAccumulatorString(this, (*quasisIt)->Cooked());
623af6ab5fSopenharmony_ci
633af6ab5fSopenharmony_ci            node = *quasisIt;
643af6ab5fSopenharmony_ci            quasisIt++;
653af6ab5fSopenharmony_ci        } else {
663af6ab5fSopenharmony_ci            const ir::Expression *element = *expressionIt;
673af6ab5fSopenharmony_ci            pg->StoreAccumulator(element, lhs);
683af6ab5fSopenharmony_ci
693af6ab5fSopenharmony_ci            element->Compile(pg);
703af6ab5fSopenharmony_ci
713af6ab5fSopenharmony_ci            node = element;
723af6ab5fSopenharmony_ci            expressionIt++;
733af6ab5fSopenharmony_ci        }
743af6ab5fSopenharmony_ci
753af6ab5fSopenharmony_ci        pg->Binary(node, lexer::TokenType::PUNCTUATOR_PLUS, lhs);
763af6ab5fSopenharmony_ci
773af6ab5fSopenharmony_ci        isQuais = !isQuais;
783af6ab5fSopenharmony_ci        total--;
793af6ab5fSopenharmony_ci    }
803af6ab5fSopenharmony_ci}
813af6ab5fSopenharmony_ci
823af6ab5fSopenharmony_cichecker::Type *TemplateLiteral::Check(checker::Checker *checker) const
833af6ab5fSopenharmony_ci{
843af6ab5fSopenharmony_ci    // TODO(aszilagyi)
853af6ab5fSopenharmony_ci    return checker->GlobalAnyType();
863af6ab5fSopenharmony_ci}
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_civoid TemplateLiteral::UpdateSelf(const NodeUpdater &cb, [[maybe_unused]] binder::Binder *binder)
893af6ab5fSopenharmony_ci{
903af6ab5fSopenharmony_ci    for (auto iter = expressions_.begin(); iter != expressions_.end(); iter++) {
913af6ab5fSopenharmony_ci        *iter = std::get<ir::AstNode *>(cb(*iter))->AsExpression();
923af6ab5fSopenharmony_ci    }
933af6ab5fSopenharmony_ci
943af6ab5fSopenharmony_ci    for (auto iter = quasis_.begin(); iter != quasis_.end(); iter++) {
953af6ab5fSopenharmony_ci        *iter = std::get<ir::AstNode *>(cb(*iter))->AsTemplateElement();
963af6ab5fSopenharmony_ci    }
973af6ab5fSopenharmony_ci}
983af6ab5fSopenharmony_ci
993af6ab5fSopenharmony_ci}  // namespace panda::es2panda::ir
100