1 /* Copyright 2018 The TensorFlow Authors. All Rights Reserved. 2 3 Licensed under the Apache License, Version 2.0 (the "License"); 4 you may not use this file except in compliance with the License. 5 You may obtain a copy of the License at 6 7 http://www.apache.org/licenses/LICENSE-2.0 8 9 Unless required by applicable law or agreed to in writing, software 10 distributed under the License is distributed on an "AS IS" BASIS, 11 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 See the License for the specific language governing permissions and 13 limitations under the License. 14 ==============================================================================*/ 15 16 #ifndef TENSORFLOW_COMPILER_XLA_SERVICE_HLO_INPUT_OUTPUT_ALIAS_CONFIG_H_ 17 #define TENSORFLOW_COMPILER_XLA_SERVICE_HLO_INPUT_OUTPUT_ALIAS_CONFIG_H_ 18 19 #include <utility> 20 21 #include "absl/container/flat_hash_set.h" 22 #include "absl/types/optional.h" 23 #include "tensorflow/compiler/xla/service/hlo.pb.h" 24 #include "tensorflow/compiler/xla/shape_tree.h" 25 #include "tensorflow/compiler/xla/shape_util.h" 26 27 namespace xla { 28 29 class HloModule; 30 31 // This class specifies the alias map from output index to parameter number and 32 // parameter index in the entry computation. 33 class HloInputOutputAliasConfig { 34 public: 35 // The kind of aliases which can be set. A kUserAlias is one setup at 36 // compilation time by the user, and has to be respected. A kSystemAlias one 37 // might be setup by the compiler, if it decides it is convenient to do so. 38 enum AliasKind { 39 kNoAlias, 40 kUserAlias, 41 kSystemAlias, 42 }; 43 44 // Defines the alias information for a given output buffer. A given output 45 // buffer shape index can refer only to one parameter+index. 46 struct Alias { AliasAlias47 Alias(AliasKind kind, int64 parameter_number, ShapeIndex parameter_index) 48 : kind(kind), 49 parameter_number(parameter_number), 50 parameter_index(std::move(parameter_index)) {} 51 52 AliasKind kind; 53 int64 parameter_number; 54 ShapeIndex parameter_index; 55 }; 56 57 HloInputOutputAliasConfig() = default; 58 HloInputOutputAliasConfig(Shape output_shape)59 explicit HloInputOutputAliasConfig(Shape output_shape) 60 : alias_(std::move(output_shape)) {} 61 62 virtual ~HloInputOutputAliasConfig() = default; 63 64 // Sets up alias config from `output_index` to `param_index` at 65 // `param_number`. 66 Status SetUpAlias(const ShapeIndex& output_index, int64 param_number, 67 const ShapeIndex& param_index, AliasKind kind); 68 69 // Returns the kind of alias for the given parameter number and parameter 70 // index. If no alias exists, AliasKind::kNoAlias is returned. 71 AliasKind ParameterAliasKind(int64 param_number, 72 const ShapeIndex& param_index) const; 73 74 // Returns true if the given parameter is aliased with one of the output 75 // buffers. ParameterHasAlias(int64 param_number,const ShapeIndex & param_index)76 bool ParameterHasAlias(int64 param_number, 77 const ShapeIndex& param_index) const { 78 return ParameterAliasKind(param_number, param_index) != AliasKind::kNoAlias; 79 } 80 81 // Checks whether the provided output index has already been aliased. 82 bool OutputHasAlias(const ShapeIndex& output_index) const; 83 84 // (De)Serializes an HloInputOutputAliasConfig to/from an 85 // HloInputOutputAliasProto. 86 HloInputOutputAliasProto ToProto() const; 87 88 static StatusOr<HloInputOutputAliasConfig> CreateFromProto( 89 Shape output_shape, const HloInputOutputAliasProto& proto); 90 91 // Returns the output index that the given parameter and parameter index is 92 // aliased with. A nullopt is returned if there is no output that is aliased 93 // with the parameter number and index. 94 absl::optional<ShapeIndex> GetAliasedOutput( 95 int64 param_number, const ShapeIndex& param_index) const; 96 97 // Returns the number of parameter and index of the parameter buffer that the 98 // given output buffer index is aliased with. A nullopt is returned if there 99 // is no parameter is aliased with the specific output. 100 absl::optional<Alias> GetAliasedParameter( 101 const ShapeIndex& output_index) const; 102 103 using AliasFn = 104 std::function<void(const ShapeIndex& output_index, const Alias&)>; 105 106 // Iterates through each aliased output and input. 107 void ForEachAlias(AliasFn fn) const; 108 109 using AliasFnWithStatus = 110 std::function<Status(const ShapeIndex& output_index, const Alias&)>; 111 112 // Verifies that the given config is valid for the given module. 113 // Specifically, the config's input and output should be in-bound and size of 114 // the aliased buffers should match. 115 Status Verify(const HloModule& module, 116 std::function<int64(const Shape&)> size_func_) const; 117 118 Status ForEachAliasWithStatus(AliasFnWithStatus fn) const; 119 120 // Returns the shape of the output of the alias config. 121 const Shape& shape() const; 122 123 string ToString() const; 124 125 private: 126 // A ShapeTree which indicates the list of buffers that's expected to be 127 // aliased. The key on this shape tree represents the output index. The value 128 // is an Alias data structure which defines the input parameter coordinates. 129 // If the value is nullopt, it means there is no parameter aliasing for this 130 // output. 131 ShapeTree<absl::optional<Alias>> alias_; 132 }; 133 134 std::ostream& operator<<(std::ostream& out, 135 const HloInputOutputAliasConfig& config); 136 137 } // namespace xla 138 139 #endif // TENSORFLOW_COMPILER_XLA_SERVICE_HLO_INPUT_OUTPUT_ALIAS_CONFIG_H_ 140