blob: 785afe98f082a9892b08f3dda19b66111cc1104e [file] [log] [blame]
/**
* Copyright (c) 2016-present, Facebook, Inc.
*
* 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.
*/
#ifndef CAFFE2_OPERATORS_CREATE_SCOPE_OP_H_
#define CAFFE2_OPERATORS_CREATE_SCOPE_OP_H_
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <vector>
#include "caffe2/core/context.h"
#include "caffe2/core/logging.h"
#include "caffe2/core/operator.h"
#include "caffe2/proto/caffe2.pb.h"
CAFFE2_DECLARE_bool(caffe2_workspace_stack_debug);
namespace caffe2 {
namespace detail {
/*
* Keeps track of forward and backward gradient workspaces in stack,
* reuses previously created workspaces, non-thread safe
*/
class WorkspaceStack {
public:
explicit WorkspaceStack() : parent_ws_(nullptr), top_(-1) {}
std::shared_ptr<Workspace> pushForwardWorkspace(
Workspace* parent_ws,
const std::unordered_map<std::string, std::string>& blob_bindings) {
checkStack();
if (FLAGS_caffe2_workspace_stack_debug) {
if (parent_ws_) {
CAFFE_ENFORCE_EQ(parent_ws_, parent_ws, "Parent workspace mismatch");
} else {
parent_ws_ = parent_ws;
}
if (!blob_bindings_.empty()) {
checkBindingsMatch(blob_bindings_, blob_bindings);
} else {
blob_bindings_ = blob_bindings;
}
}
if (top_ == workspaces_.size() - 1) {
workspaces_.push_back(
std::make_shared<Workspace>(parent_ws, blob_bindings));
}
return workspaces_[++top_];
}
std::shared_ptr<Workspace> popGradientWorkspace(
Workspace* parent_ws,
const std::unordered_map<std::string, std::string>& grad_blob_bindings) {
checkStack();
if (FLAGS_caffe2_workspace_stack_debug) {
if (parent_ws_) {
CAFFE_ENFORCE_EQ(parent_ws_, parent_ws, "Parent workspace mismatch");
} else {
parent_ws_ = parent_ws;
}
if (!grad_blob_bindings_.empty()) {
checkBindingsMatch(grad_blob_bindings_, grad_blob_bindings);
} else {
grad_blob_bindings_ = grad_blob_bindings;
}
}
if (top_ < 0) {
return nullptr;
}
auto& grad_workspace = workspaces_[top_];
grad_workspace->AddBlobMapping(parent_ws, grad_blob_bindings);
--top_;
return grad_workspace;
}
void clear() {
checkStack();
top_ = -1;
}
private:
void checkStack() const {
CAFFE_ENFORCE_GT(
(int)workspaces_.size(), top_, "Corrupted workspaces stack");
}
void checkBindingsMatch(
const std::unordered_map<std::string, std::string>& bindings,
const std::unordered_map<std::string, std::string>& test_bindings) const {
CAFFE_ENFORCE_EQ(
bindings.size(), test_bindings.size(), "Blob bindings mismatch");
for (const auto& blob_binding : bindings) {
CAFFE_ENFORCE(
test_bindings.count(blob_binding.first), "Blob bindings mismatch");
CAFFE_ENFORCE_EQ(
test_bindings.at(blob_binding.first),
blob_binding.second,
"Blob bindings mismatch");
}
}
std::unordered_map<std::string, std::string> blob_bindings_;
std::unordered_map<std::string, std::string> grad_blob_bindings_;
Workspace* parent_ws_;
int top_;
std::vector<std::shared_ptr<Workspace>> workspaces_;
};
}
template <class Context>
class CreateScopeOp final : public Operator<Context> {
public:
CreateScopeOp(const OperatorDef& operator_def, Workspace* ws)
: Operator<Context>(operator_def, ws) {}
USE_OPERATOR_CONTEXT_FUNCTIONS;
bool RunOnDevice() override;
};
} // namespace caffe2
#endif // CAFFE2_OPERATORS_CREATE_SCOPE_OP_H_