Coverage Report

Created: 2023-03-01 07:33

/src/spirv-tools/source/opt/ccp_pass.h
Line
Count
Source
1
// Copyright (c) 2017 Google Inc.
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
#ifndef SOURCE_OPT_CCP_PASS_H_
16
#define SOURCE_OPT_CCP_PASS_H_
17
18
#include <memory>
19
#include <unordered_map>
20
21
#include "source/opt/constants.h"
22
#include "source/opt/function.h"
23
#include "source/opt/ir_context.h"
24
#include "source/opt/mem_pass.h"
25
#include "source/opt/module.h"
26
#include "source/opt/propagator.h"
27
28
namespace spvtools {
29
namespace opt {
30
31
class CCPPass : public MemPass {
32
 public:
33
22.4k
  CCPPass() = default;
34
35
11.9k
  const char* name() const override { return "ccp"; }
36
  Status Process() override;
37
38
5.94k
  IRContext::Analysis GetPreservedAnalyses() override {
39
5.94k
    return IRContext::kAnalysisDefUse |
40
5.94k
           IRContext::kAnalysisInstrToBlockMapping |
41
5.94k
           IRContext::kAnalysisDecorations | IRContext::kAnalysisCombinators |
42
5.94k
           IRContext::kAnalysisCFG | IRContext::kAnalysisDominatorAnalysis |
43
5.94k
           IRContext::kAnalysisNameMap | IRContext::kAnalysisConstants |
44
5.94k
           IRContext::kAnalysisTypes;
45
5.94k
  }
46
47
 private:
48
  // Initializes the pass.
49
  void Initialize();
50
51
  // Runs constant propagation on the given function |fp|. Returns true if any
52
  // constants were propagated and the IR modified.
53
  bool PropagateConstants(Function* fp);
54
55
  // Visits a single instruction |instr|.  If the instruction is a conditional
56
  // branch that always jumps to the same basic block, it sets the destination
57
  // block in |dest_bb|.
58
  SSAPropagator::PropStatus VisitInstruction(Instruction* instr,
59
                                             BasicBlock** dest_bb);
60
61
  // Visits an OpPhi instruction |phi|. This applies the meet operator for the
62
  // CCP lattice. Essentially, if all the operands in |phi| have the same
63
  // constant value C, the result for |phi| gets assigned the value C.
64
  SSAPropagator::PropStatus VisitPhi(Instruction* phi);
65
66
  // Visits an SSA assignment instruction |instr|.  If the RHS of |instr| folds
67
  // into a constant value C, then the LHS of |instr| is assigned the value C in
68
  // |values_|.
69
  SSAPropagator::PropStatus VisitAssignment(Instruction* instr);
70
71
  // Visits a branch instruction |instr|. If the branch is conditional
72
  // (OpBranchConditional or OpSwitch), and the value of its selector is known,
73
  // |dest_bb| will be set to the corresponding destination block. Unconditional
74
  // branches always set |dest_bb| to the single destination block.
75
  SSAPropagator::PropStatus VisitBranch(Instruction* instr,
76
                                        BasicBlock** dest_bb) const;
77
78
  // Replaces all operands used in |fp| with the corresponding constant values
79
  // in |values_|.  Returns true if any operands were replaced, and false
80
  // otherwise.
81
  bool ReplaceValues();
82
83
  // Marks |instr| as varying by registering a varying value for its result
84
  // into the |values_| table. Returns SSAPropagator::kVarying.
85
  SSAPropagator::PropStatus MarkInstructionVarying(Instruction* instr);
86
87
  // Returns true if |id| is the special SSA id that corresponds to a varying
88
  // value.
89
  bool IsVaryingValue(uint32_t id) const;
90
91
  // Constant manager for the parent IR context.  Used to record new constants
92
  // generated during propagation.
93
  analysis::ConstantManager* const_mgr_;
94
95
  // Returns a new value for |instr| by computing the meet operation between
96
  // its existing value and |val2|.
97
  //
98
  // Given two values val1 and val2, the meet operation in the constant
99
  // lattice uses the following rules:
100
  //
101
  // meet(val1, UNDEFINED) = val1
102
  // meet(val1, VARYING)   = VARYING
103
  // meet(val1, val2)      = val1     if val1 == val2
104
  // meet(val1, val2)      = VARYING  if val1 != val2
105
  //
106
  // When two different values meet, the result is always varying because CCP
107
  // does not allow lateral transitions in the lattice.  This prevents
108
  // infinite cycles during propagation.
109
  uint32_t ComputeLatticeMeet(Instruction* instr, uint32_t val2);
110
111
  // Constant value table.  Each entry <id, const_decl_id> in this map
112
  // represents the compile-time constant value for |id| as declared by
113
  // |const_decl_id|. Each |const_decl_id| in this table is an OpConstant
114
  // declaration for the current module.
115
  //
116
  // Additionally, this table keeps track of SSA IDs with varying values. If an
117
  // SSA ID is found to have a varying value, it will have an entry in this
118
  // table that maps to the special SSA id kVaryingSSAId.  These values are
119
  // never replaced in the IR, they are used by CCP during propagation.
120
  std::unordered_map<uint32_t, uint32_t> values_;
121
122
  // Propagator engine used.
123
  std::unique_ptr<SSAPropagator> propagator_;
124
125
  // Value for the module's ID bound before running CCP. Used to detect whether
126
  // propagation created new instructions.
127
  uint32_t original_id_bound_;
128
};
129
130
}  // namespace opt
131
}  // namespace spvtools
132
133
#endif  // SOURCE_OPT_CCP_PASS_H_