/* Copyright 2017 The OpenXLA Authors.

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 XLA_SERVICE_COMPILE_ONLY_SERVICE_H_
#define XLA_SERVICE_COMPILE_ONLY_SERVICE_H_

#include "absl/status/statusor.h"
#include "xla/service/backend.h"
#include "xla/service/compiler.h"
#include "xla/service/service.h"
#include "xla/stream_executor/stream_executor.h"
#include "xla/xla_data.pb.h"

namespace xla {

// An XLA Service specialization for ahead-of-time compilation.  This only
// instantiates a Compiler object for the relevant platform; it does not
// instantiate or require an execution backend.
class CompileOnlyService : public Service {
 public:
  // Factory for creating a CompileOnlyService. The parameter platform is the
  // platform that the service should target. If platform is null then the
  // default platform is used.
  static absl::StatusOr<std::unique_ptr<CompileOnlyService>> NewService(
      se::Platform* platform);
  static absl::StatusOr<std::unique_ptr<CompileOnlyService>> NewService(
      const ServiceOptions& options);

  // A description of a xla computation to compile using CompileAheadOfTime.
  struct AotXlaComputationInstance {
    HloModuleProto computation;
    std::vector<const Shape*> argument_layouts;
    Shape result_layout;
  };

  // Compiles a list of xla computations for ahead-of-time execution.  This is
  // intended for use in static compilation.  See
  // |CompileOnlyClient::CompileAheadOfTime| for additional details.
  absl::StatusOr<std::vector<std::unique_ptr<AotCompilationResult>>>
  CompileAheadOfTime(absl::Span<const AotXlaComputationInstance> computations,
                     const AotCompilationOptions& options,
                     std::unique_ptr<AotCompilationMetadata>* metadata);

  absl::StatusOr<std::vector<DeviceHandle>> GetDeviceHandles(
      int64_t device_count) override {
    return Unimplemented("CompileOnlyService does not support devices.");
  }

  absl::StatusOr<std::unique_ptr<GlobalData>> TransferToServer(
      const LiteralSlice& literal_slice,
      const DeviceHandle* device_handle) override {
    return Unimplemented(
        "CompileOnlyService does not support device data transfers.");
  }

  absl::Status TransferToInfeed(const LiteralSlice& literal, int64_t replica_id,
                                const DeviceHandle* device_handle) override {
    return Unimplemented(
        "CompileOnlyService does not support device data transfers.");
  }

 private:
  explicit CompileOnlyService(const ServiceOptions& options,
                              std::unique_ptr<Compiler> compiler);
  CompileOnlyService(const CompileOnlyService&) = delete;
  void operator=(const CompileOnlyService&) = delete;

  // The compiler for the target platform.  This is included in place of
  // the Service::execute_backend_'s compiler, since execute_backend_ is a
  // nullptr in CompileOnlyService.
  std::unique_ptr<Compiler> compiler_;
};

}  // namespace xla

#endif  // XLA_SERVICE_COMPILE_ONLY_SERVICE_H_
