// Copyright (c) 2019 PaddlePaddle Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "lite/core/scope.h" #define SCOPE_KIDS_READER_LOCK lite::fluid::AutoRDLock auto_lock(kids_lock_); #define SCOPE_KIDS_WRITER_LOCK lite::fluid::AutoWRLock auto_lock(kids_lock_); #define SCOPE_VARS_READER_LOCK lite::fluid::AutoRDLock auto_lock(vars_lock_); #define SCOPE_VARS_WRITER_LOCK lite::fluid::AutoWRLock auto_lock(vars_lock_); namespace paddle { namespace lite { Scope::~Scope() { SCOPE_KIDS_WRITER_LOCK for (auto *x : kids_) { if (x) { delete x; } } } Scope &Scope::NewScope() const { SCOPE_KIDS_WRITER_LOCK kids_.push_back(new Scope); kids_.back()->parent_ = this; return *kids_.back(); } Variable *Scope::Var(const std::string &name) { SCOPE_VARS_WRITER_LOCK auto *var = FindVar(name); if (var) return var; // create a new variable. vars_.emplace(name, std::unique_ptr(new Variable)); return vars_[name].get(); } Variable *Scope::LocalVar(const std::string &name) { SCOPE_VARS_WRITER_LOCK auto *var = FindLocalVar(name); if (var) return var; // create a new variable. vars_.emplace(name, std::unique_ptr(new Variable)); return vars_[name].get(); } Variable *Scope::FindVar(const std::string &name) const { Variable *var{nullptr}; var = FindLocalVar(name); const Scope *cur_scope = this; rwlock_->RDLock(); while (!var && cur_scope->parent()) { cur_scope = cur_scope->parent(); var = cur_scope->FindLocalVar(name); } rwlock_->UNLock(); return var; } Variable *Scope::FindLocalVar(const std::string &name) const { rwlock_->RDLock(); auto it = vars_.find(name); if (it != vars_.end()) { rwlock_->UNLock(); return it->second.get(); } rwlock_->UNLock(); return nullptr; } // AttributeVarNames will get persistive attribute names stored in parent scope std::vector Scope::AttributeVarNames() const { std::vector resulted_keys; const Scope *cur_scope = this; while (cur_scope->parent()) { cur_scope = cur_scope->parent(); auto keys = cur_scope->LocalVarNames(); resulted_keys.insert(resulted_keys.end(), keys.begin(), keys.end()); } // remove feed and fetch std::vector skiped_vars = {"feed", "fetch"}; for (int i = 0; i < skiped_vars.size(); i++) { auto iter = std::find(resulted_keys.begin(), resulted_keys.end(), skiped_vars[i]); while (iter != resulted_keys.end()) { resulted_keys.erase(iter); iter = std::find(resulted_keys.begin(), resulted_keys.end(), skiped_vars[i]); } } return resulted_keys; } std::vector Scope::LocalVarNames() const { std::vector keys; { rwlock_->RDLock(); for (const auto &item : vars_) { keys.push_back(item.first); } rwlock_->UNLock(); } return keys; } } // namespace lite } // namespace paddle