129 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			129 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
// 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.
 | 
						|
 | 
						|
#ifndef SOURCE_FUZZ_TRANSFORMATION_ADD_FUNCTION_H_
 | 
						|
#define SOURCE_FUZZ_TRANSFORMATION_ADD_FUNCTION_H_
 | 
						|
 | 
						|
#include "source/fuzz/protobufs/spirvfuzz_protobufs.h"
 | 
						|
#include "source/fuzz/transformation.h"
 | 
						|
#include "source/fuzz/transformation_context.h"
 | 
						|
#include "source/opt/ir_context.h"
 | 
						|
 | 
						|
namespace spvtools {
 | 
						|
namespace fuzz {
 | 
						|
 | 
						|
class TransformationAddFunction : public Transformation {
 | 
						|
 public:
 | 
						|
  explicit TransformationAddFunction(
 | 
						|
      protobufs::TransformationAddFunction message);
 | 
						|
 | 
						|
  // Creates a transformation to add a non live-safe function.
 | 
						|
  explicit TransformationAddFunction(
 | 
						|
      const std::vector<protobufs::Instruction>& instructions);
 | 
						|
 | 
						|
  // Creates a transformation to add a live-safe function.
 | 
						|
  TransformationAddFunction(
 | 
						|
      const std::vector<protobufs::Instruction>& instructions,
 | 
						|
      uint32_t loop_limiter_variable_id, uint32_t loop_limit_constant_id,
 | 
						|
      const std::vector<protobufs::LoopLimiterInfo>& loop_limiters,
 | 
						|
      uint32_t kill_unreachable_return_value_id,
 | 
						|
      const std::vector<protobufs::AccessChainClampingInfo>&
 | 
						|
          access_chain_clampers);
 | 
						|
 | 
						|
  // - |message_.instruction| must correspond to a sufficiently well-formed
 | 
						|
  //   sequence of instructions that a function can be created from them
 | 
						|
  // - If |message_.is_livesafe| holds then |message_| must contain suitable
 | 
						|
  //   ingredients to make the function livesafe, and the function must only
 | 
						|
  //   invoke other livesafe functions
 | 
						|
  // - Adding the created function to the module must lead to a valid module.
 | 
						|
  bool IsApplicable(
 | 
						|
      opt::IRContext* ir_context,
 | 
						|
      const TransformationContext& transformation_context) const override;
 | 
						|
 | 
						|
  // Adds the function defined by |message_.instruction| to the module, making
 | 
						|
  // it livesafe if |message_.is_livesafe| holds.
 | 
						|
  void Apply(opt::IRContext* ir_context,
 | 
						|
             TransformationContext* transformation_context) const override;
 | 
						|
 | 
						|
  std::unordered_set<uint32_t> GetFreshIds() const override;
 | 
						|
 | 
						|
  protobufs::Transformation ToMessage() const override;
 | 
						|
 | 
						|
  // Helper method that, given composite type |composite_type_inst|, returns the
 | 
						|
  // type of the sub-object at index |index_id|, which is required to be in-
 | 
						|
  // bounds.
 | 
						|
  static opt::Instruction* FollowCompositeIndex(
 | 
						|
      opt::IRContext* ir_context, const opt::Instruction& composite_type_inst,
 | 
						|
      uint32_t index_id);
 | 
						|
 | 
						|
  // Returns id of the back-edge block, given the corresponding
 | 
						|
  // |loop_header_block_id|. |loop_header_block_id| must be the id of a loop
 | 
						|
  // header block. Returns 0 if the loop has no back-edge block.
 | 
						|
  static uint32_t GetBackEdgeBlockId(opt::IRContext* ir_context,
 | 
						|
                                     uint32_t loop_header_block_id);
 | 
						|
 | 
						|
  // Attempts to create a function from the series of instructions in
 | 
						|
  // |message_.instruction| and add it to |ir_context|.
 | 
						|
  //
 | 
						|
  // Returns false if adding the function is not possible due to the messages
 | 
						|
  // not respecting the basic structure of a function, e.g. if there is no
 | 
						|
  // OpFunction instruction or no blocks; in this case |ir_context| is left in
 | 
						|
  // an indeterminate state.
 | 
						|
  //
 | 
						|
  // Otherwise returns true.  Whether |ir_context| is valid after addition of
 | 
						|
  // the function depends on the contents of |message_.instruction|.
 | 
						|
  //
 | 
						|
  // Intended usage:
 | 
						|
  // - Perform a dry run of this method on a clone of a module, and use
 | 
						|
  //   the validator to check whether the resulting module is valid.  Working
 | 
						|
  //   on a clone means it does not matter if the function fails to be cleanly
 | 
						|
  //   added, or leads to an invalid module.
 | 
						|
  // - If the dry run succeeds, run the method on the real module of interest,
 | 
						|
  //   to add the function.
 | 
						|
  bool TryToAddFunction(opt::IRContext* ir_context) const;
 | 
						|
 | 
						|
 private:
 | 
						|
  // Should only be called if |message_.is_livesafe| holds.  Attempts to make
 | 
						|
  // the function livesafe (see FactFunctionIsLivesafe for a definition).
 | 
						|
  // Returns false if this is not possible, due to |message_| or |ir_context|
 | 
						|
  // not containing sufficient ingredients (such as types and fresh ids) to add
 | 
						|
  // the instrumentation necessary to make the function livesafe.
 | 
						|
  bool TryToMakeFunctionLivesafe(
 | 
						|
      opt::IRContext* ir_context,
 | 
						|
      const TransformationContext& transformation_context) const;
 | 
						|
 | 
						|
  // A helper for TryToMakeFunctionLivesafe that tries to add loop-limiting
 | 
						|
  // logic.
 | 
						|
  bool TryToAddLoopLimiters(opt::IRContext* ir_context,
 | 
						|
                            opt::Function* added_function) const;
 | 
						|
 | 
						|
  // A helper for TryToMakeFunctionLivesafe that tries to replace OpKill and
 | 
						|
  // OpUnreachable instructions into return instructions.
 | 
						|
  bool TryToTurnKillOrUnreachableIntoReturn(
 | 
						|
      opt::IRContext* ir_context, opt::Function* added_function,
 | 
						|
      opt::Instruction* kill_or_unreachable_inst) const;
 | 
						|
 | 
						|
  // A helper for TryToMakeFunctionLivesafe that tries to clamp access chain
 | 
						|
  // indices so that they are guaranteed to be in-bounds.
 | 
						|
  bool TryToClampAccessChainIndices(opt::IRContext* ir_context,
 | 
						|
                                    opt::Instruction* access_chain_inst) const;
 | 
						|
 | 
						|
  protobufs::TransformationAddFunction message_;
 | 
						|
};
 | 
						|
 | 
						|
}  // namespace fuzz
 | 
						|
}  // namespace spvtools
 | 
						|
 | 
						|
#endif  // SOURCE_FUZZ_TRANSFORMATION_ADD_FUNCTION_H_
 |