Use the migrated headers under tensorflow/core/platform. PiperOrigin-RevId: 304194185 Change-Id: I476004f49e554b07f5670b7f5b0c3c61a5ddc839
155 lines
5.4 KiB
C++
155 lines
5.4 KiB
C++
/* Copyright 2015 The TensorFlow 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 "tensorflow/c/checkpoint_reader.h"
|
|
|
|
#include <unordered_set>
|
|
#include <utility>
|
|
|
|
#include "tensorflow/core/platform/env.h"
|
|
#include "tensorflow/core/platform/status.h"
|
|
#include "tensorflow/core/platform/stringpiece.h"
|
|
#include "tensorflow/core/platform/types.h"
|
|
#include "tensorflow/core/util/saved_tensor_slice_util.h"
|
|
|
|
namespace tensorflow {
|
|
namespace checkpoint {
|
|
|
|
class TensorSliceReader;
|
|
|
|
CheckpointReader::CheckpointReader(const string& filename, TF_Status* status)
|
|
: reader_(nullptr),
|
|
v2_reader_(nullptr),
|
|
var_to_shape_map_(nullptr),
|
|
var_to_data_type_map_(nullptr) {
|
|
// Depending on whether this is a V2 ckpt, initializes "reader_" or
|
|
// "v2_reader_".
|
|
std::vector<string> v2_path;
|
|
if (Env::Default()->GetMatchingPaths(MetaFilename(filename), &v2_path).ok() &&
|
|
!v2_path.empty()) {
|
|
v2_reader_.reset(
|
|
new BundleReader(Env::Default(), filename /* prefix to a V2 ckpt */));
|
|
if (!v2_reader_->status().ok()) {
|
|
Set_TF_Status_from_Status(status, v2_reader_->status());
|
|
return;
|
|
}
|
|
auto result = BuildV2VarMaps();
|
|
var_to_shape_map_.swap(result.first);
|
|
var_to_data_type_map_.swap(result.second);
|
|
} else {
|
|
reader_.reset(new TensorSliceReader(filename));
|
|
if (!reader_->status().ok()) {
|
|
Set_TF_Status_from_Status(status, reader_->status());
|
|
return;
|
|
}
|
|
var_to_shape_map_.reset(
|
|
new TensorSliceReader::VarToShapeMap(reader_->GetVariableToShapeMap()));
|
|
var_to_data_type_map_.reset(new TensorSliceReader::VarToDataTypeMap(
|
|
reader_->GetVariableToDataTypeMap()));
|
|
}
|
|
}
|
|
|
|
bool CheckpointReader::HasTensor(const string& name) const {
|
|
if (reader_ != nullptr) {
|
|
return reader_->HasTensor(name, nullptr, nullptr);
|
|
}
|
|
return v2_reader_->Contains(name);
|
|
}
|
|
|
|
const TensorSliceReader::VarToShapeMap&
|
|
CheckpointReader::GetVariableToShapeMap() const {
|
|
CHECK(var_to_shape_map_);
|
|
return *var_to_shape_map_;
|
|
}
|
|
|
|
const TensorSliceReader::VarToDataTypeMap&
|
|
CheckpointReader::GetVariableToDataTypeMap() const {
|
|
CHECK(var_to_data_type_map_);
|
|
return *var_to_data_type_map_;
|
|
}
|
|
|
|
const string CheckpointReader::DebugString() const {
|
|
if (reader_ != nullptr) return reader_->DebugString();
|
|
return v2_reader_->DebugString();
|
|
}
|
|
|
|
void CheckpointReader::GetTensor(
|
|
const string& name, std::unique_ptr<tensorflow::Tensor>* out_tensor,
|
|
TF_Status* out_status) const {
|
|
Status status;
|
|
if (reader_ != nullptr) {
|
|
status = reader_->GetTensor(name, out_tensor);
|
|
} else {
|
|
tensorflow::DataType dtype;
|
|
tensorflow::TensorShape shape;
|
|
status = v2_reader_->LookupDtypeAndShape(name, &dtype, &shape);
|
|
if (status.ok()) {
|
|
out_tensor->reset(new Tensor(dtype, shape));
|
|
status = v2_reader_->Lookup(name, out_tensor->get());
|
|
if (!status.ok()) out_tensor->reset();
|
|
}
|
|
}
|
|
if (!status.ok()) {
|
|
Set_TF_Status_from_Status(out_status, status);
|
|
}
|
|
}
|
|
|
|
std::pair<std::unique_ptr<TensorSliceReader::VarToShapeMap>,
|
|
std::unique_ptr<TensorSliceReader::VarToDataTypeMap>>
|
|
CheckpointReader::BuildV2VarMaps() {
|
|
CHECK(v2_reader_ != nullptr);
|
|
CHECK(v2_reader_->status().ok());
|
|
|
|
// First pass: filters out the entries of the slices.
|
|
std::unordered_set<string> filtered_keys;
|
|
BundleEntryProto entry;
|
|
v2_reader_->Seek(kHeaderEntryKey);
|
|
for (v2_reader_->Next(); v2_reader_->Valid(); v2_reader_->Next()) {
|
|
CHECK(entry.ParseFromArray(v2_reader_->value().data(),
|
|
v2_reader_->value().size()))
|
|
<< entry.InitializationErrorString();
|
|
for (int i = 0; i < entry.slices_size(); ++i) {
|
|
const auto& slice_proto = entry.slices(i);
|
|
CHECK(filtered_keys
|
|
.insert(EncodeTensorNameSlice(
|
|
string(v2_reader_->key()) /* full var's name */,
|
|
TensorSlice(slice_proto)))
|
|
.second);
|
|
}
|
|
}
|
|
|
|
// Second pass: adds the entries, ignoring the filtered keys.
|
|
std::unique_ptr<TensorSliceReader::VarToShapeMap> var_to_shape_map(
|
|
new TensorSliceReader::VarToShapeMap);
|
|
std::unique_ptr<TensorSliceReader::VarToDataTypeMap> var_to_data_type_map(
|
|
new TensorSliceReader::VarToDataTypeMap);
|
|
v2_reader_->Seek(kHeaderEntryKey);
|
|
for (v2_reader_->Next(); v2_reader_->Valid(); v2_reader_->Next()) {
|
|
if (filtered_keys.count(string(v2_reader_->key())) > 0) continue;
|
|
CHECK(entry.ParseFromArray(v2_reader_->value().data(),
|
|
v2_reader_->value().size()))
|
|
<< entry.InitializationErrorString();
|
|
string key(v2_reader_->key());
|
|
(*var_to_shape_map)[key] = TensorShape(entry.shape());
|
|
(*var_to_data_type_map)[key] = DataType(entry.dtype());
|
|
}
|
|
// The returned pointers are owned by the caller.
|
|
return std::make_pair(std::move(var_to_shape_map),
|
|
std::move(var_to_data_type_map));
|
|
}
|
|
|
|
} // namespace checkpoint
|
|
} // namespace tensorflow
|