/* Copyright (c) 2016 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 "paddle/fluid/operators/detail/sendrecvop_utils.h" #include "google/protobuf/io/coded_stream.h" #include "google/protobuf/io/zero_copy_stream.h" #include "paddle/fluid/framework/data_type.h" #include "paddle/fluid/operators/detail/bytebuffer_stream.h" #include "paddle/fluid/operators/detail/proto_encoder_helper.h" namespace paddle { namespace operators { namespace detail { void SerializeToMessage(const std::string& name, const framework::Variable* var, const platform::DeviceContext& ctx, sendrecv::VariableMessage* msg) { msg->set_varname(name); std::ostringstream oss; switch (framework::ToVarType(var->Type())) { case framework::proto::VarType_Type_LOD_TENSOR: msg->set_type(sendrecv::VarType::LOD_TENSOR); framework::SerializeToStream(oss, var->Get(), ctx); break; case framework::proto::VarType_Type_SELECTED_ROWS: msg->set_type(sendrecv::VarType::SELECTED_ROWS); framework::SerializeToStream(oss, var->Get(), ctx); break; default: { PADDLE_THROW("Serialize does not support type: %s", typeid(var->Type()).name()); break; } } msg->set_serialized(oss.str()); } void DeserializeFromMessage(const sendrecv::VariableMessage& msg, const platform::DeviceContext& ctx, framework::Variable* var) { std::istringstream iss(msg.serialized()); switch (msg.type()) { case sendrecv::VarType::LOD_TENSOR: DeserializeFromStream(iss, var->GetMutable(), ctx); break; case sendrecv::VarType::SELECTED_ROWS: { DeserializeFromStream(iss, var->GetMutable(), ctx); break; } default: { PADDLE_THROW("Deserialize does not support type: %s", typeid(var->Type()).name()); break; } } } void SerializeToByteBuffer(const std::string& name, framework::Variable* var, const platform::DeviceContext& ctx, ::grpc::ByteBuffer* msg) { using VarMsg = sendrecv::VariableMessage; sendrecv::VariableMessage request; std::string header; request.AppendToString(&header); // When using GPU, need to free the copied CPU buffer // when the ByteBuffer destroies // TODO(typhoonzero): add unref here, if we have dependent // parallelism execution, need to know when to free the tensor. DestroyCallback destroy_callback = [](void* backing) {}; void* buf = malloc(1024); void* payload = nullptr; size_t payload_size; ProtoEncodeHelper e((char*)buf, 1024); e.WriteString(VarMsg::kVarnameFieldNumber, name); if (var->IsType()) { e.WriteUint64(VarMsg::kTypeFieldNumber, 0); } else if (var->IsType()) { e.WriteUint64(VarMsg::kTypeFieldNumber, 1); } switch (framework::ToVarType(var->Type())) { case framework::proto::VarType_Type_LOD_TENSOR: { auto tensor = var->Get(); e.WriteUint64(VarMsg::kDataTypeFieldNumber, framework::ToDataType(tensor.type())); for (auto& dim : framework::vectorize(tensor.dims())) { e.WriteUint64(VarMsg::kDimsFieldNumber, dim); } auto lod = tensor.lod(); // std::vector> if (lod.size() > 0) { e.WriteUint64(VarMsg::kLodLevelFieldNumber, lod.size()); for (auto& each : lod) { e.WriteVarlengthBeginning(VarMsg::kLodFieldNumber, 2 + // tag + varintlength of submessage 1 + // kLodDataFieldNumber each.size()); // auto copied from GPU for (auto& d : each) { e.WriteUint64(VarMsg::LodData::kLodDataFieldNumber, d); } } } if (platform::is_gpu_place(ctx.GetPlace())) { #ifdef PADDLE_WITH_CUDA PADDLE_ENFORCE(platform::is_gpu_place(tensor.place())); platform::CPUPlace cpu; auto& gpu_dev_ctx = static_cast(ctx); auto copy_size = tensor.memory_size(); payload = memory::Alloc(cpu, copy_size); memory::Copy(cpu, payload, boost::get(tensor.place()), reinterpret_cast(tensor.data()), copy_size, gpu_dev_ctx.stream()); ctx.Wait(); destroy_callback = [](void* backing) { platform::CPUPlace cpu; memory::Free(cpu, backing); }; #endif } else { payload = tensor.data(); } payload_size = tensor.memory_size(); e.WriteVarlengthBeginning(VarMsg::kSerializedFieldNumber, payload_size); } break; case framework::proto::VarType_Type_SELECTED_ROWS: { // TODO(typhoonzero): selectedrows implement should not use unique_ptr auto* slr = var->GetMutable(); e.WriteUint64(VarMsg::kDataTypeFieldNumber, framework::ToDataType(slr->value().type())); for (auto& dim : framework::vectorize(slr->value().dims())) { e.WriteUint64(VarMsg::kDimsFieldNumber, dim); } e.WriteUint64(VarMsg::kLodLevelFieldNumber, 0); auto* tensor = slr->mutable_value(); if (platform::is_gpu_place(ctx.GetPlace())) { #ifdef PADDLE_WITH_CUDA platform::CPUPlace cpu; auto& gpu_dev_ctx = static_cast(ctx); auto copy_size = tensor->memory_size(); payload = memory::Alloc(cpu, copy_size); memory::Copy(cpu, payload, boost::get(tensor->place()), reinterpret_cast(tensor->data()), copy_size, gpu_dev_ctx.stream()); ctx.Wait(); destroy_callback = [](void* backing) { platform::CPUPlace cpu; memory::Free(cpu, backing); }; #endif } else { payload = slr->mutable_value()->data(); } payload_size = tensor->memory_size(); e.WriteVarlengthBeginning(VarMsg::kSerializedFieldNumber, payload_size); } break; default: PADDLE_THROW("Serialize does not support type: %s", typeid(var->Type()).name()); break; } // steal reference of tensor data ::grpc::Slice slices[4]; // metadata, tensor, rows meta, rows int num_slices = 2; // only SelectedRows have rows buffer slices[0] = ::grpc::Slice(e.size()); memcpy(const_cast(slices[0].begin()), e.data(), e.size()); slices[1] = ::grpc::Slice( grpc_slice_new_with_user_data(payload, payload_size, destroy_callback, static_cast(payload)), ::grpc::Slice::STEAL_REF); if (framework::ToVarType(var->Type()) == framework::proto::VarType_Type_SELECTED_ROWS) { auto* slr = var->GetMutable(); ProtoEncodeHelper e2((char*)buf, 128); // NOTE: rows is of type int64_t size_t rows_memory_size = slr->rows().capacity() * framework::SizeOfType(typeid(int64_t)); e2.WriteVarlengthBeginning(VarMsg::kRowsFieldNumber, rows_memory_size); slices[2] = ::grpc::Slice(e2.size()); memcpy(const_cast(slices[2].begin()), e2.data(), e2.size()); slices[3] = ::grpc::Slice( grpc_slice_new_with_user_data( const_cast( reinterpret_cast(slr->rows().data())), rows_memory_size, [](void* backing) { // TODO(typhoonzero): add unref here, same as above. }, const_cast( reinterpret_cast(slr->rows().data()))), ::grpc::Slice::STEAL_REF); num_slices = 4; } ::grpc::ByteBuffer tmp(&slices[0], num_slices); msg->Swap(&tmp); } void DeserializeFromByteBuffer(const ::grpc::ByteBuffer& msg, const platform::DeviceContext& ctx, framework::Variable* var) { sendrecv::VariableMessage meta; GrpcByteBufferSource source; source.Init(msg); ::google::protobuf::io::CodedInputStream input(&source); // do zerocopy parsing PADDLE_ENFORCE(meta.ParseFromCodedStream(&input)); PADDLE_ENFORCE(input.ConsumedEntireMessage()); // dims is needed by both tensor and selectedrows std::vector vecdims; for (auto& d : meta.dims()) { vecdims.push_back(d); } framework::DDim dims = framework::make_ddim(vecdims); if (meta.type() == sendrecv::LOD_TENSOR) { auto* tensor = var->GetMutable(); tensor->Resize(dims); void* tensor_data = tensor->mutable_data( ctx.GetPlace(), paddle::operators::detail::ToTypeIndex(meta.data_type())); framework::LoD lod; for (int i = 0; i < meta.lod_level(); ++i) { framework::Vector v; for (int j = 0; j < meta.lod(i).lod_data_size(); ++j) { v.push_back(meta.lod(i).lod_data(j)); } lod.push_back(v); } tensor->set_lod(lod); // How to avoid copying and use the message buffer directly? // Maybe need to find a way to release all memory except tensor content. if (platform::is_gpu_place(ctx.GetPlace())) { #ifdef PADDLE_WITH_CUDA platform::CPUPlace cpu; auto& gpu_dev_ctx = static_cast(ctx); memory::Copy(boost::get(tensor->place()), tensor_data, cpu, reinterpret_cast(meta.serialized().data()), meta.serialized().size(), gpu_dev_ctx.stream()); ctx.Wait(); #endif } else { memcpy(tensor_data, reinterpret_cast(meta.serialized().data()), meta.serialized().size()); } } else if (meta.type() == sendrecv::SELECTED_ROWS) { auto* slr = var->GetMutable(); auto* tensor = slr->mutable_value(); int64_t* rows_data = slr->mutable_rows()->data(); tensor->Resize(dims); void* tensor_data = tensor->mutable_data( ctx.GetPlace(), paddle::operators::detail::ToTypeIndex(meta.data_type())); if (platform::is_gpu_place(ctx.GetPlace())) { #ifdef PADDLE_WITH_CUDA platform::CPUPlace cpu; auto& gpu_dev_ctx = static_cast(ctx); memory::Copy(boost::get(tensor->place()), tensor_data, cpu, reinterpret_cast(meta.serialized().data()), meta.serialized().size(), gpu_dev_ctx.stream()); ctx.Wait(); #endif } else { memcpy(tensor_data, reinterpret_cast(meta.serialized().data()), meta.serialized().size()); } // copy rows CPU data, GPU data will be copied lazly memcpy(rows_data, reinterpret_cast(meta.rows().data()), meta.rows().size()); } } } // namespace detail } // namespace operators } // namespace paddle