blob: e4942bd76a6b6cb3e77d0cb060c38a06f5969e22 [file] [log] [blame]
/* Copyright 2017 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/compiler/xla/service/gpu/gpu_executable.h"
#include <set>
#include <utility>
#include <vector>
#include "absl/container/flat_hash_map.h"
#include "absl/memory/memory.h"
#include "tensorflow/compiler/xla/map_util.h"
#include "tensorflow/compiler/xla/service/gpu/buffer_allocations.h"
#include "tensorflow/compiler/xla/service/gpu/gpu_debug_info_manager.h"
#include "tensorflow/compiler/xla/service/gpu/gpu_types.h"
#include "tensorflow/compiler/xla/service/gpu/hlo_execution_profiler.h"
#include "tensorflow/compiler/xla/service/hlo_instruction.h"
#include "tensorflow/compiler/xla/service/llvm_ir/buffer_assignment_util.h"
#include "tensorflow/compiler/xla/service/logical_buffer.h"
#include "tensorflow/compiler/xla/service/shaped_buffer.h"
#include "tensorflow/compiler/xla/service/transfer_manager.h"
#include "tensorflow/compiler/xla/shape_tree.h"
#include "tensorflow/compiler/xla/shape_util.h"
#include "tensorflow/compiler/xla/status_macros.h"
#include "tensorflow/compiler/xla/util.h"
#include "tensorflow/core/platform/logging.h"
#include "tensorflow/core/platform/tracing.h"
#include "tensorflow/core/profiler/lib/traceme.h"
#include "tensorflow/stream_executor/platform.h"
namespace xla {
namespace gpu {
namespace {
using tensorflow::tracing::ScopedAnnotation;
} // namespace
// Implementation note: HLO profiling is always enabled for GPU executables,
// since we can use timers around thunks.
GpuExecutable::GpuExecutable(
const string& text, const std::vector<uint8>& binary,
GpuVersion gpu_version, std::unique_ptr<const ThunkSchedule> thunk_schedule,
std::shared_ptr<HloModule> hlo_module,
std::shared_ptr<const BufferAssignment> assignment,
std::unique_ptr<HloProfilePrinterData> hlo_profile_printer_data,
std::unique_ptr<HloProfileIndexMap> hlo_profile_index_map)
: Executable(std::move(hlo_module), std::move(hlo_profile_printer_data),
std::move(hlo_profile_index_map)),
text_(text),
binary_(binary),
gpu_version_(gpu_version),
thunk_schedule_(std::move(thunk_schedule)),
assignment_(std::move(assignment)) {
CHECK(has_module() && assignment_);
GpuDebugInfoManager::Get()->RegisterModule(module().name(), shared_module(),
assignment_);
ComputeThunkAnnotations();
}
GpuExecutable::~GpuExecutable() {
CHECK(has_module() && assignment_);
GpuDebugInfoManager::Get()->UnregisterModule(module().name(), shared_module(),
assignment_);
}
void GpuExecutable::ComputeThunkAnnotations() {
CanonicalNameMap canonical_name_map;
for (Thunk* thunk : thunk_schedule_->TotalOrder()) {
const HloInstruction* hlo = thunk->hlo_instruction();
CHECK(hlo);
thunk_annotations_[thunk] = absl::StrFormat(
"%s:#tf_op=%s,hlo_op=%s,hlo_module=%s#",
hlo->ToStringWithCanonicalNameMap(HloPrintOptions::Canonical(),
&canonical_name_map),
hlo->metadata().op_name(), hlo->name(), hlo->GetModule()->name());
}
}
Status GpuExecutable::CheckCompatibilityWithServiceExecutableRunOptions(
const ServiceExecutableRunOptions* run_options) {
se::Stream* main_stream = run_options->stream();
stream_executor::PlatformKind platform_kind =
main_stream->parent()->platform_kind();
if (platform_kind == stream_executor::PlatformKind::kROCm) {
int stream_isa_version;
main_stream->parent()->GetDeviceDescription().rocm_amdgpu_isa_version(
&stream_isa_version);
GpuVersion amd_isa_version = stream_isa_version;
TF_RET_CHECK(amd_isa_version == gpu_version_)
<< "AMDGPU GCN ISA version mismatch; expected {"
<< absl::get<int>(gpu_version_) << ", but was " << stream_isa_version;
} else if (platform_kind == stream_executor::PlatformKind::kCuda) {
std::pair<int, int> stream_compute_compatibility;
main_stream->parent()->GetDeviceDescription().cuda_compute_capability(
&stream_compute_compatibility.first,
&stream_compute_compatibility.second);
GpuVersion nvdia_compute_compatibility = stream_compute_compatibility;
TF_RET_CHECK(nvdia_compute_compatibility == gpu_version_)
<< "Compute capability mismatch; expected {"
<< absl::get<std::pair<int, int>>(gpu_version_).first << ", "
<< absl::get<std::pair<int, int>>(gpu_version_).second << "}, but was {"
<< stream_compute_compatibility.first << ", "
<< stream_compute_compatibility.second << "}";
} else {
return InternalError("Unknown platform: %d", platform_kind);
}
return Status::OK();
}
Status GpuExecutable::ExecuteThunks(
const ServiceExecutableRunOptions* run_options,
const BufferAllocations& buffer_allocations, bool block_host_until_done,
HloExecutionProfile* hlo_execution_profile) {
TF_RETURN_IF_ERROR(
CheckCompatibilityWithServiceExecutableRunOptions(run_options));
GpuDebugInfoManager::Get()->OnModuleStart(module().name());
auto cleanup = MakeCleanup(
[&]() { GpuDebugInfoManager::Get()->OnModuleStop(module().name()); });
se::Stream* main_stream = run_options->stream();
se::StreamExecutor* executor = main_stream->parent();
bool do_profile = hlo_execution_profile != nullptr;
if (do_profile) {
LOG(WARNING) << "PROFILING: profiling is enabled";
}
// Stream 0 indicates `main_stream` and substreams start from stream 1.
std::vector<StreamPool::Ptr> sub_streams;
sub_streams.reserve(thunk_schedule_->StreamCount() - 1);
while (sub_streams.size() + 1 < thunk_schedule_->StreamCount()) {
sub_streams.emplace_back();
TF_ASSIGN_OR_RETURN(sub_streams.back(),
run_options->BorrowStream(executor->device_ordinal()));
}
HloExecutionProfiler profiler(do_profile, hlo_execution_profile, main_stream,
sub_streams, hlo_module_->entry_computation());
uint64 start_micros = tensorflow::Env::Default()->NowMicros();
tensorflow::profiler::TraceMe hlo_module_activity(
[&] { return absl::StrCat(hlo_module_->name(), ":XLA GPU module"); },
tensorflow::profiler::TraceMeLevel::kInfo);
std::map<const Thunk*, std::unique_ptr<se::Event>> thunk_to_finish_event;
bool scoped_annotation_enabled = ScopedAnnotation::IsEnabled();
for (Thunk* thunk : thunk_schedule_->TotalOrder()) {
// Annotate execution of this op if tracing was enabled when we started
// running this module. If tracing is enabled *while* we're running the
// module, we won't get any data, but that's probably an OK trade-off.
absl::optional<ScopedAnnotation> op_annotation;
CHECK(thunk->hlo_instruction());
if (scoped_annotation_enabled) {
op_annotation.emplace(FindOrDie(thunk_annotations_, thunk));
}
TF_RETURN_IF_ERROR(thunk->Initialize(*this, executor));
int32 stream_no =
thunk_schedule_->StreamNumberForHlo(*thunk->hlo_instruction());
se::Stream* stream =
(stream_no == 0 ? main_stream : sub_streams[stream_no - 1].get());
for (const Thunk* dependency : thunk_schedule_->DependsOn(thunk)) {
stream->ThenWaitFor(FindOrDie(thunk_to_finish_event, dependency).get());
}
VLOG(2) << "Executing the thunk for "
<< thunk->hlo_instruction()->ToString() << " on stream "
<< stream_no;
Thunk::ExecuteParams thunk_params{
&buffer_allocations, stream, run_options->run_options().run_id(),
&profiler, run_options->run_options().device_assignment()};
TF_RETURN_IF_ERROR(thunk->ExecuteOnStream(thunk_params));
if (thunk_schedule_->Depended(thunk)) {
auto finish_event = absl::make_unique<se::Event>(main_stream->parent());
finish_event->Init();
stream->ThenRecordEvent(finish_event.get());
thunk_to_finish_event[thunk] = std::move(finish_event);
}
}
main_stream->ThenWaitFor(&sub_streams);
// Make sure kernels are completed before deallocating temporary buffers.
// TODO(b/30100571): we could potentially postpone deallocating the temp
// buffers until a different computation is executed.
if (block_host_until_done) {
Status block_status = main_stream->BlockHostUntilDone();
if (!block_status.ok()) {
return InternalError(
"Failed to complete all kernels launched on stream %p: %s",
main_stream, block_status.error_message());
}
}
profiler.FinishExecution();
uint64 end_micros = tensorflow::Env::Default()->NowMicros();
{
tensorflow::mutex_lock lock(mutex_);
const double nanoseconds = (end_micros - start_micros) * 1000.0;
execution_profile_.set_compute_time_ns(std::max(nanoseconds, 1.0));
// If hlo profiling was disabled then the cycle count is left empty.
if (do_profile) {
execution_profile_.set_compute_cycle_count(
hlo_execution_profile->total_cycles_executed(
*module().entry_computation()));
}
}
return Status::OK();
}
StatusOr<const GpuExecutable::BufferAllocToDeviceMemoryMap*>
GpuExecutable::ResolveConstantGlobals(se::StreamExecutor* executor) {
tensorflow::mutex_lock lock(module_handle_mutex_);
auto it = module_globals_.find(executor);
if (it != module_globals_.end()) {
return &it->second;
}
se::MultiModuleLoaderSpec module_spec;
if (!binary().empty()) {
module_spec.AddCudaCubinInMemory(binary());
}
module_spec.AddCudaPtxInMemory(text().c_str());
absl::flat_hash_map<int64, se::DeviceMemoryBase> globals;
se::ModuleHandle module_handle;
executor->LoadModule(module_spec, &module_handle);
for (BufferAllocation::Index i = 0; i < assignment_->Allocations().size();
++i) {
const BufferAllocation& allocation = assignment_->GetAllocation(i);
if (allocation.is_constant()) {
TF_ASSIGN_OR_RETURN(
se::DeviceMemoryBase global,
executor->GetUntypedSymbol(
llvm_ir::ConstantBufferAllocationToGlobalName(allocation),
module_handle));
VLOG(3) << "Resolved global "
<< llvm_ir::ConstantBufferAllocationToGlobalName(allocation)
<< " to " << global.opaque();
InsertOrDie(&globals, i, global);
const Literal& literal =
llvm_ir::LiteralForConstantAllocation(allocation);
CHECK(literal.shape().IsArray());
if (!ShouldEmitLiteralInLlvmIr(literal)) {
VLOG(3) << "H2D memcpy for constant with shape "
<< ShapeUtil::HumanString(literal.shape());
TF_RETURN_IF_ERROR(executor->SynchronousMemcpyH2D(
literal.untyped_data(), allocation.size(), &global));
}
}
}
module_handles_.emplace(executor,
se::ScopedModuleHandle(executor, module_handle));
return &module_globals_.emplace(executor, std::move(globals)).first->second;
}
StatusOr<ScopedShapedBuffer> GpuExecutable::Execute(
const ServiceExecutableRunOptions* run_options,
absl::Span<const ShapedBuffer* const> arguments,
HloExecutionProfile* hlo_execution_profile, bool block_host_until_done) {
se::DeviceMemoryAllocator* memory_allocator = run_options->allocator();
if (GetRootValueSet().IsAmbiguous()) {
return Unimplemented("Points-to set of root instruction is ambiguous");
}
BufferAllocations::Builder buffer_allocations_builder;
se::StreamExecutor* executor = run_options->stream()->parent();
const GpuExecutable::BufferAllocToDeviceMemoryMap* globals;
{
tensorflow::profiler::TraceMe hlo_module_activity(
[&] { return std::string("Resolve constant globals"); },
tensorflow::profiler::TraceMeLevel::kInfo);
TF_ASSIGN_OR_RETURN(globals, ResolveConstantGlobals(executor));
}
std::unique_ptr<BufferAllocations> buffer_allocations;
{
tensorflow::profiler::TraceMe hlo_module_activity(
[&] { return std::string("Build buffer allocations"); },
tensorflow::profiler::TraceMeLevel::kInfo);
for (BufferAllocation::Index i = 0; i < assignment_->Allocations().size();
++i) {
const BufferAllocation& allocation = assignment_->GetAllocation(i);
if (allocation.is_entry_computation_parameter()) {
auto param_no = allocation.parameter_number();
se::DeviceMemoryBase buffer =
arguments[param_no]->buffer(allocation.param_shape_index());
// All top-level buffers and sub-buffers must have an explicit, non-null
// pointer, except for zero-sized buffers, which may be null.
if (buffer.is_null() && buffer.size() > 0) {
return FailedPrecondition(
"Cannot run XLA computation because pointer to (sub-)buffer at "
"index %s of parameter %d was null. All pointers to "
"(sub-)buffers must not be null, unless the (sub-)buffer has "
"zero elements.",
allocation.param_shape_index().ToString(), param_no);
}
buffer_allocations_builder.RegisterBuffer(i, buffer);
}
if (allocation.is_constant()) {
buffer_allocations_builder.RegisterBuffer(i, FindOrDie(*globals, i));
}
}
TF_ASSIGN_OR_RETURN(
buffer_allocations,
buffer_allocations_builder.Build(
assignment_.get(), executor->device_ordinal(), memory_allocator));
}
TF_RETURN_IF_ERROR(ExecuteThunks(run_options, *buffer_allocations,
block_host_until_done,
hlo_execution_profile));
HloInstruction* root = hlo_module_->entry_computation()->root_instruction();
auto device_ordinal = executor->device_ordinal();
ScopedShapedBuffer shaped_buffer(root->shape(), root->shape(),
memory_allocator, device_ordinal);
// Copy DeviceMemoryBase values which contain the array(s) of the result into
// the respective location in ShapedBuffer.
std::set<se::DeviceMemoryBase> buffers_in_result;
TF_RETURN_IF_ERROR(shaped_buffer.buffers().ForEachMutableElementWithStatus(
[&buffer_allocations, &buffers_in_result, this](
const ShapeIndex& index, se::DeviceMemoryBase* device_memory) {
const auto& sources = this->GetRootValueSet().element(index);
// The points-to set is unambiguous so the set should be a
// singleton. That is, we know exactly which instruction
// produced the array at this element.
CHECK_EQ(1, sources.values().size());
auto src_hlo = sources.values()[0]->instruction();
VLOG(4) << "Looking at: " << sources.values()[0];
// The source instruction should have a non-parameter buffer
// assigned.
TF_ASSIGN_OR_RETURN(const BufferAllocation::Slice slice,
this->assignment_->GetUniqueSlice(
src_hlo, sources.values()[0]->index()));
se::DeviceMemoryBase src_base =
buffer_allocations->GetDeviceAddress(slice.index());
CHECK(!src_base.is_null() || src_base.size() == 0);
if (!slice.allocation()->is_entry_computation_parameter()) {
// If the buffer coming out of the result is from a parameter, it
// means the caller aliased some parameter buffer to an output one
// (via the HloInputOutputAliasConfig API). If that is the case, the
// caller will receive a partially complete scoped shaped buffer,
// which they will have to fill up on return.
// Unfortunately the interface to the execute APIs are ShapedBuffer
// pointer based, which assumes caller ownership, and hence a buffer
// coming from there cannot be part of the new ScopedShapedBuffer we
// create for the result (which assumes ownership).
*device_memory = src_base;
} else {
const HloInputOutputAliasConfig& input_output_alias =
module().input_output_alias_config();
auto output_alias = input_output_alias.GetAliasedOutput(
slice.allocation()->parameter_number(),
slice.allocation()->param_shape_index());
CHECK(output_alias)
<< "Ouput buffer is coming from parameter "
<< slice.allocation()->parameter_number() << " at index "
<< slice.allocation()->param_shape_index()
<< ", but no alias exists";
CHECK_EQ(*output_alias, index);
}
buffers_in_result.insert(src_base);
return Status::OK();
}));
TF_RETURN_IF_ERROR(buffer_allocations->TearDown(buffers_in_result));
return std::move(shaped_buffer);
}
StatusOr<ScopedShapedBuffer> GpuExecutable::ExecuteOnStream(
const ServiceExecutableRunOptions* run_options,
absl::Span<const ShapedBuffer* const> arguments,
HloExecutionProfile* hlo_execution_profile) {
// TODO(b/134086343): ExecuteOnStream should not be async according to the
// documentation, instead ExecuteAsyncOnStream should be used.
return Execute(run_options, arguments, hlo_execution_profile,
/*block_host_until_done=*/
!run_options->allocator()->AllowsAsynchronousDeallocation());
}
StatusOr<ScopedShapedBuffer> GpuExecutable::ExecuteAsyncOnStream(
const ServiceExecutableRunOptions* run_options,
absl::Span<const ShapedBuffer* const> arguments) {
se::DeviceMemoryAllocator* memory_allocator = run_options->allocator();
// Force synchronous execution if the allocator requires it.
bool block_host_until_done =
!memory_allocator->AllowsAsynchronousDeallocation();
return Execute(run_options, arguments, nullptr, block_host_until_done);
}
const InstructionValueSet& GpuExecutable::GetRootValueSet() const {
return assignment_->dataflow_analysis().GetInstructionValueSet(
module().entry_computation()->root_instruction());
}
} // namespace gpu
} // namespace xla