13af6ab5fSopenharmony_ci/*
23af6ab5fSopenharmony_ci * Copyright (c) 2022 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 "concurrent.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include <binder/scope.h>
193af6ab5fSopenharmony_ci#include <compiler/core/compilerContext.h>
203af6ab5fSopenharmony_ci#include <compiler/core/pandagen.h>
213af6ab5fSopenharmony_ci#include <ir/astNode.h>
223af6ab5fSopenharmony_ci#include <ir/base/scriptFunction.h>
233af6ab5fSopenharmony_ci#include <ir/expressions/literals/stringLiteral.h>
243af6ab5fSopenharmony_ci#include <ir/statements/blockStatement.h>
253af6ab5fSopenharmony_ci#include <ir/statements/expressionStatement.h>
263af6ab5fSopenharmony_ci#include <lexer/token/sourceLocation.h>
273af6ab5fSopenharmony_ci#include <parser/module/sourceTextModuleRecord.h>
283af6ab5fSopenharmony_ci
293af6ab5fSopenharmony_cinamespace panda::es2panda::util {
303af6ab5fSopenharmony_ci
313af6ab5fSopenharmony_civoid Concurrent::SetConcurrent(ir::ScriptFunction *func, const ir::AstNode * node, const lexer::LineIndex &lineIndex)
323af6ab5fSopenharmony_ci{
333af6ab5fSopenharmony_ci    // concurrent function should only be function declaration
343af6ab5fSopenharmony_ci    if (!func->CanBeConcurrent()) {
353af6ab5fSopenharmony_ci        ThrowInvalidConcurrentFunction(lineIndex, node, ConcurrentInvalidFlag::NOT_ORDINARY_FUNCTION);
363af6ab5fSopenharmony_ci    }
373af6ab5fSopenharmony_ci
383af6ab5fSopenharmony_ci    func->AddFlag(ir::ScriptFunctionFlags::CONCURRENT);
393af6ab5fSopenharmony_ci}
403af6ab5fSopenharmony_ci
413af6ab5fSopenharmony_civoid Concurrent::ThrowInvalidConcurrentFunction(const lexer::LineIndex &lineIndex, const ir::AstNode *expr,
423af6ab5fSopenharmony_ci                                                ConcurrentInvalidFlag errFlag, util::StringView varName)
433af6ab5fSopenharmony_ci{
443af6ab5fSopenharmony_ci    auto line = expr->Range().start.line;
453af6ab5fSopenharmony_ci    auto column = (const_cast<lexer::LineIndex &>(lineIndex)).GetLocation(expr->Range().start).col - 1;
463af6ab5fSopenharmony_ci    switch (errFlag) {
473af6ab5fSopenharmony_ci        case ConcurrentInvalidFlag::NOT_ORDINARY_FUNCTION: {
483af6ab5fSopenharmony_ci            throw Error {ErrorType::GENERIC, "Concurrent function should only be function declaration", line,
493af6ab5fSopenharmony_ci                         column};
503af6ab5fSopenharmony_ci            break;
513af6ab5fSopenharmony_ci        }
523af6ab5fSopenharmony_ci        case ConcurrentInvalidFlag::NOT_IMPORT_VARIABLE: {
533af6ab5fSopenharmony_ci            std::stringstream ss;
543af6ab5fSopenharmony_ci            ss << "Concurrent function should only use import variable or local variable, '" << varName
553af6ab5fSopenharmony_ci               << "' is not one of them";
563af6ab5fSopenharmony_ci            throw Error {ErrorType::GENERIC, ss.str(), line, column};
573af6ab5fSopenharmony_ci            break;
583af6ab5fSopenharmony_ci        }
593af6ab5fSopenharmony_ci        default:
603af6ab5fSopenharmony_ci            break;
613af6ab5fSopenharmony_ci    }
623af6ab5fSopenharmony_ci}
633af6ab5fSopenharmony_ci
643af6ab5fSopenharmony_civoid Concurrent::ProcessConcurrent(const lexer::LineIndex &lineIndex, const ir::AstNode *node,
653af6ab5fSopenharmony_ci                                   const binder::ScopeFindResult &result, parser::Program *program)
663af6ab5fSopenharmony_ci{
673af6ab5fSopenharmony_ci    if (!result.concurrentFunc) {
683af6ab5fSopenharmony_ci        return;
693af6ab5fSopenharmony_ci    }
703af6ab5fSopenharmony_ci
713af6ab5fSopenharmony_ci    if (result.variable->IsModuleVariable() && result.variable->Declaration()->IsImportDecl()) {
723af6ab5fSopenharmony_ci        CollectRelativeModule(result, program);
733af6ab5fSopenharmony_ci        return;
743af6ab5fSopenharmony_ci    }
753af6ab5fSopenharmony_ci
763af6ab5fSopenharmony_ci    ThrowInvalidConcurrentFunction(lineIndex, node, ConcurrentInvalidFlag::NOT_IMPORT_VARIABLE,
773af6ab5fSopenharmony_ci                                   result.variable->Declaration()->Name());
783af6ab5fSopenharmony_ci}
793af6ab5fSopenharmony_ci
803af6ab5fSopenharmony_civoid Concurrent::CollectRelativeModule(const binder::ScopeFindResult &result, parser::Program *program)
813af6ab5fSopenharmony_ci{
823af6ab5fSopenharmony_ci    int moduleRequestIdx = program->ModuleRecord()->GetModuleRequestIdx(result.variable->Name());
833af6ab5fSopenharmony_ci    if (moduleRequestIdx != parser::SourceTextModuleRecord::INVALID_MODULEREQUEST_ID) {
843af6ab5fSopenharmony_ci        result.concurrentFunc->AddConcurrentModuleRequest(moduleRequestIdx);
853af6ab5fSopenharmony_ci    }
863af6ab5fSopenharmony_ci}
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_ci} // namespace panda::es2panda::util