// Copyright (c) 2019 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.

#include "source/fuzz/uniform_buffer_element_descriptor.h"

#include <algorithm>

namespace spvtools {
namespace fuzz {

protobufs::UniformBufferElementDescriptor MakeUniformBufferElementDescriptor(
    uint32_t descriptor_set, uint32_t binding,
    std::vector<uint32_t>&& indices) {
  protobufs::UniformBufferElementDescriptor result;
  result.set_descriptor_set(descriptor_set);
  result.set_binding(binding);
  for (auto index : indices) {
    result.add_index(index);
  }
  return result;
}

bool UniformBufferElementDescriptorEquals::operator()(
    const protobufs::UniformBufferElementDescriptor* first,
    const protobufs::UniformBufferElementDescriptor* second) const {
  return first->descriptor_set() == second->descriptor_set() &&
         first->binding() == second->binding() &&
         first->index().size() == second->index().size() &&
         std::equal(first->index().begin(), first->index().end(),
                    second->index().begin());
}

opt::Instruction* FindUniformVariable(
    const protobufs::UniformBufferElementDescriptor&
        uniform_buffer_element_descriptor,
    opt::IRContext* context, bool check_unique) {
  opt::Instruction* result = nullptr;

  for (auto& inst : context->types_values()) {
    // Consider all global variables with uniform storage class.
    if (inst.opcode() != SpvOpVariable) {
      continue;
    }
    if (inst.GetSingleWordInOperand(0) != SpvStorageClassUniform) {
      continue;
    }

    // Determine whether the variable is decorated with a descriptor set
    // matching that in |uniform_buffer_element|.
    bool descriptor_set_matches = false;
    context->get_decoration_mgr()->ForEachDecoration(
        inst.result_id(), SpvDecorationDescriptorSet,
        [&descriptor_set_matches, &uniform_buffer_element_descriptor](
            const opt::Instruction& decoration_inst) {
          const uint32_t kDescriptorSetOperandIndex = 2;
          if (decoration_inst.GetSingleWordInOperand(
                  kDescriptorSetOperandIndex) ==
              uniform_buffer_element_descriptor.descriptor_set()) {
            descriptor_set_matches = true;
          }
        });
    if (!descriptor_set_matches) {
      // Descriptor set does not match.
      continue;
    }

    // Determine whether the variable is decorated with a binding matching that
    // in |uniform_buffer_element|.
    bool binding_matches = false;
    context->get_decoration_mgr()->ForEachDecoration(
        inst.result_id(), SpvDecorationBinding,
        [&binding_matches, &uniform_buffer_element_descriptor](
            const opt::Instruction& decoration_inst) {
          const uint32_t kBindingOperandIndex = 2;
          if (decoration_inst.GetSingleWordInOperand(kBindingOperandIndex) ==
              uniform_buffer_element_descriptor.binding()) {
            binding_matches = true;
          }
        });
    if (!binding_matches) {
      // Binding does not match.
      continue;
    }

    // This instruction is a uniform variable with the right descriptor set and
    // binding.
    if (!check_unique) {
      // If we aren't checking uniqueness, return it.
      return &inst;
    }

    if (result) {
      // More than one uniform variable is decorated with the given descriptor
      // set and binding. This means the fact is ambiguous.
      return nullptr;
    }
    result = &inst;
  }

  // We get here either if no match was found, or if |check_unique| holds and
  // exactly one match was found.
  assert(result == nullptr || check_unique);
  return result;
}

}  // namespace fuzz
}  // namespace spvtools
