// Copyright (c) 2020 Google LLC
//
// 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 SOURCE_REDUCE_REMOVE_STRUCT_MEMBER_REDUCTION_OPPORTUNITY_H_
#define SOURCE_REDUCE_REMOVE_STRUCT_MEMBER_REDUCTION_OPPORTUNITY_H_

#include "source/opt/instruction.h"
#include "source/reduce/reduction_opportunity.h"

namespace spvtools {
namespace reduce {

// An opportunity for removing a member from a struct type, adjusting all uses
// of the struct accordingly.
class RemoveStructMemberReductionOpportunity : public ReductionOpportunity {
 public:
  // Constructs a reduction opportunity from the struct type |struct_type|, for
  // removal of member |member_index|.
  RemoveStructMemberReductionOpportunity(opt::Instruction* struct_type,
                                         uint32_t member_index)
      : struct_type_(struct_type),
        member_index_(member_index),
        original_number_of_members_(struct_type->NumInOperands()) {}

  // Opportunities to remove fields from a common struct type mutually
  // invalidate each other.  We guard against this by requiring that the struct
  // still has the number of members it had when the opportunity was created.
  bool PreconditionHolds() override;

 protected:
  void Apply() override;

 private:
  // |composite_access_instruction| is an instruction that accesses a composite
  // id using either a series of literal indices (e.g. in the case of
  // OpCompositeInsert) or a series of index ids (e.g. in the case of
  // OpAccessChain).
  //
  // This function adjusts the indices that are used by
  // |composite_access_instruction| to that whenever an index is accessing a
  // member of |struct_type_|, it is decremented if the member is beyond
  // |member_index_|, to account for the removal of the |member_index_|-th
  // member.
  //
  // |composite_type_id| is the id of the composite type that the series of
  // indices is to be applied to.
  //
  // |first_index_input_operand| specifies the first input operand that is an
  // index.
  //
  // |literal_indices| specifies whether indices are given as literals (true),
  // or as ids (false).
  //
  // If id-based indexing is used, this function will add a constant for
  // |member_index_| - 1 to the module if needed.
  void AdjustAccessedIndices(
      uint32_t composite_type_id, uint32_t first_index_input_operand,
      bool literal_indices, opt::IRContext* context,
      opt::Instruction* composite_access_instruction) const;

  // The struct type from which a member is to be removed.
  opt::Instruction* struct_type_;

  uint32_t member_index_;

  uint32_t original_number_of_members_;
};

}  // namespace reduce
}  // namespace spvtools

#endif  //   SOURCE_REDUCE_REMOVE_STRUCT_MEMBER_REDUCTION_OPPORTUNITY_H_
