Jlm
Load.hpp
Go to the documentation of this file.
1 /*
2  * Copyright 2017 Nico Reißmann <nico.reissmann@gmail.com>
3  * See COPYING for terms of redistribution.
4  */
5 
6 #ifndef JLM_LLVM_IR_OPERATORS_LOAD_HPP
7 #define JLM_LLVM_IR_OPERATORS_LOAD_HPP
8 
9 #include <jlm/llvm/ir/tac.hpp>
10 #include <jlm/llvm/ir/types.hpp>
11 #include <jlm/rvsdg/graph.hpp>
13 
14 #include <optional>
15 
16 namespace jlm::llvm
17 {
18 
26 {
27 protected:
29  const std::vector<std::shared_ptr<const rvsdg::Type>> & operandTypes,
30  const std::vector<std::shared_ptr<const rvsdg::Type>> & resultTypes,
31  const size_t numMemoryStates,
32  const size_t alignment)
33  : SimpleOperation(operandTypes, resultTypes),
35  Alignment_(alignment)
36  {
37  JLM_ASSERT(!operandTypes.empty() && !resultTypes.empty());
38 
39  auto & addressType = *operandTypes[0];
40  JLM_ASSERT(is<PointerType>(addressType));
41 
42  auto & loadedType = *resultTypes[0];
43  JLM_ASSERT(loadedType.Kind() == rvsdg::TypeKind::Value);
44 
45  JLM_ASSERT(operandTypes.size() == resultTypes.size());
46  for (size_t n = 1; n < operandTypes.size(); n++)
47  {
48  auto & operandType = *operandTypes[n];
49  auto & resultType = *resultTypes[n];
50  JLM_ASSERT(operandType == resultType);
51  JLM_ASSERT(operandType.Kind() == rvsdg::TypeKind::State);
52  }
53  }
54 
55 public:
56  [[nodiscard]] size_t
57  GetAlignment() const noexcept
58  {
59  return Alignment_;
60  }
61 
62  [[nodiscard]] std::shared_ptr<const rvsdg::Type>
63  GetLoadedType() const noexcept
64  {
65  return result(0);
66  }
67 
68  [[nodiscard]] size_t
69  NumMemoryStates() const noexcept
70  {
71  return NumMemoryStates_;
72  }
73 
74  [[nodiscard]] static rvsdg::Input &
75  AddressInput(const rvsdg::Node & node) noexcept
76  {
77  JLM_ASSERT(is<LoadOperation>(&node));
78  const auto input = node.input(0);
79  JLM_ASSERT(is<PointerType>(input->Type()));
80  return *input;
81  }
82 
83  [[nodiscard]] static rvsdg::Output &
85  {
86  JLM_ASSERT(is<LoadOperation>(&node));
87  const auto output = node.output(0);
88  JLM_ASSERT(output->Type()->Kind() == rvsdg::TypeKind::Value);
89  return *output;
90  }
91 
100  [[nodiscard]] static size_t
101  numMemoryStates(const rvsdg::SimpleNode & node) noexcept
102  {
103  const auto loadOperation = util::assertedCast<const LoadOperation>(&node.GetOperation());
104  return loadOperation->NumMemoryStates_;
105  }
106 
115  [[nodiscard]] static rvsdg::Node::OutputIteratorRange
116  MemoryStateOutputs(const rvsdg::Node & node) noexcept
117  {
118  const auto loadOperation = util::assertedCast<const LoadOperation>(&node.GetOperation());
119  if (loadOperation->NumMemoryStates_ == 0)
120  {
121  return { rvsdg::Output::Iterator(nullptr), rvsdg::Output::Iterator(nullptr) };
122  }
123 
124  const auto firstMemoryStateOutput =
125  node.output(loadOperation->nresults() - loadOperation->NumMemoryStates_);
126  JLM_ASSERT(is<MemoryStateType>(firstMemoryStateOutput->Type()));
127  return { rvsdg::Output::Iterator(firstMemoryStateOutput), rvsdg::Output::Iterator(nullptr) };
128  }
129 
138  [[nodiscard]] static rvsdg::Node::InputIteratorRange
139  MemoryStateInputs(const rvsdg::Node & node) noexcept
140  {
141  const auto loadOperation = util::assertedCast<const LoadOperation>(&node.GetOperation());
142  if (loadOperation->NumMemoryStates_ == 0)
143  {
144  return { rvsdg::Input::Iterator(nullptr), rvsdg::Input::Iterator(nullptr) };
145  }
146 
147  const auto firstMemoryStateOutput =
148  node.input(loadOperation->narguments() - loadOperation->NumMemoryStates_);
149  JLM_ASSERT(is<MemoryStateType>(firstMemoryStateOutput->Type()));
150  return { rvsdg::Input::Iterator(firstMemoryStateOutput), rvsdg::Input::Iterator(nullptr) };
151  }
152 
156  [[nodiscard]] static rvsdg::Input &
158  {
159  JLM_ASSERT(is<MemoryStateType>(output.Type()));
160  auto [loadNode, loadOperation] = rvsdg::TryGetSimpleNodeAndOptionalOp<LoadOperation>(output);
161  JLM_ASSERT(loadOperation);
162  JLM_ASSERT(loadNode->ninputs() == loadNode->noutputs());
163  const auto input = loadNode->input(output.index());
164  JLM_ASSERT(is<MemoryStateType>(input->Type()));
165  return *input;
166  }
167 
171  [[nodiscard]] static rvsdg::Output &
173  {
174  JLM_ASSERT(is<MemoryStateType>(input.Type()));
175  auto [loadNode, loadOperation] = rvsdg::TryGetSimpleNodeAndOptionalOp<LoadOperation>(input);
176  JLM_ASSERT(loadOperation);
177  JLM_ASSERT(loadNode->ninputs() == loadNode->noutputs());
178  const auto output = loadNode->output(input.index());
179  JLM_ASSERT(is<MemoryStateType>(output->Type()));
180  return *output;
181  }
182 
183 private:
185  size_t Alignment_;
186 };
187 
200 {
201 public:
202  ~LoadVolatileOperation() noexcept override;
203 
205  std::shared_ptr<const rvsdg::Type> loadedType,
206  size_t numMemoryStates,
207  size_t alignment)
208  : LoadOperation(
210  CreateResultTypes(std::move(loadedType), numMemoryStates),
212  alignment)
213  {}
214 
215  bool
216  operator==(const Operation & other) const noexcept override;
217 
218  [[nodiscard]] std::string
219  debug_string() const override;
220 
221  [[nodiscard]] std::unique_ptr<Operation>
222  copy() const override;
223 
224  [[nodiscard]] static rvsdg::Input &
225  IOStateInput(const rvsdg::Node & node) noexcept
226  {
227  JLM_ASSERT(is<LoadVolatileOperation>(&node));
228  const auto input = node.input(1);
229  JLM_ASSERT(is<IOStateType>(input->Type()));
230  return *input;
231  }
232 
233  [[nodiscard]] static rvsdg::Output &
235  {
236  JLM_ASSERT(is<LoadVolatileOperation>(&node));
237  const auto output = node.output(1);
238  JLM_ASSERT(is<IOStateType>(output->Type()));
239  return *output;
240  }
241 
242  static std::unique_ptr<llvm::ThreeAddressCode>
244  const Variable * address,
245  const Variable * iOState,
246  const Variable * memoryState,
247  std::shared_ptr<const rvsdg::Type> loadedType,
248  size_t alignment)
249  {
250  auto operation = std::make_unique<LoadVolatileOperation>(std::move(loadedType), 1, alignment);
251  return ThreeAddressCode::create(std::move(operation), { address, iOState, memoryState });
252  }
253 
254  static rvsdg::SimpleNode &
255  CreateNode(
256  rvsdg::Region & region,
257  std::unique_ptr<LoadVolatileOperation> loadOperation,
258  const std::vector<rvsdg::Output *> & operands);
259 
260  static rvsdg::SimpleNode &
262  rvsdg::Output & address,
263  rvsdg::Output & iOState,
264  const std::vector<rvsdg::Output *> & memoryStates,
265  std::shared_ptr<const rvsdg::Type> loadedType,
266  size_t alignment)
267  {
268  std::vector operands({ &address, &iOState });
269  operands.insert(operands.end(), memoryStates.begin(), memoryStates.end());
270 
271  auto operation = std::make_unique<LoadVolatileOperation>(
272  std::move(loadedType),
273  memoryStates.size(),
274  alignment);
275  return CreateNode(*address.region(), std::move(operation), operands);
276  }
277 
278 private:
279  static std::vector<std::shared_ptr<const rvsdg::Type>>
281  {
282  std::vector<std::shared_ptr<const rvsdg::Type>> types(
284  std::vector<std::shared_ptr<const rvsdg::Type>> states(
287  types.insert(types.end(), states.begin(), states.end());
288  return types;
289  }
290 
291  static std::vector<std::shared_ptr<const rvsdg::Type>>
292  CreateResultTypes(std::shared_ptr<const rvsdg::Type> loadedType, size_t numMemoryStates)
293  {
294  std::vector<std::shared_ptr<const rvsdg::Type>> types(
295  { std::move(loadedType), IOStateType::Create() });
296  std::vector<std::shared_ptr<const rvsdg::Type>> states(
299  types.insert(types.end(), states.begin(), states.end());
300  return types;
301  }
302 };
303 
310 {
311 public:
312  ~LoadNonVolatileOperation() noexcept override;
313 
315  std::shared_ptr<const rvsdg::Type> loadedType,
316  size_t numMemoryStates,
317  size_t alignment)
318  : LoadOperation(
320  CreateResultTypes(std::move(loadedType), numMemoryStates),
322  alignment)
323  {}
324 
325  bool
326  operator==(const Operation & other) const noexcept override;
327 
328  [[nodiscard]] std::string
329  debug_string() const override;
330 
331  [[nodiscard]] std::unique_ptr<Operation>
332  copy() const override;
333 
352  static std::optional<std::vector<rvsdg::Output *>>
354  const LoadNonVolatileOperation & operation,
355  const std::vector<rvsdg::Output *> & operands);
356 
373  static std::optional<std::vector<rvsdg::Output *>>
375  const LoadNonVolatileOperation & operation,
376  const std::vector<rvsdg::Output *> & operands);
377 
397  static std::optional<std::vector<rvsdg::Output *>>
399  const LoadNonVolatileOperation & operation,
400  const std::vector<rvsdg::Output *> & operands);
401 
415  static std::optional<std::vector<rvsdg::Output *>>
417  const LoadNonVolatileOperation & operation,
418  const std::vector<rvsdg::Output *> & operands);
419 
438  static std::optional<std::vector<rvsdg::Output *>>
440  const LoadNonVolatileOperation & operation,
441  const std::vector<rvsdg::Output *> & operands);
442 
443  static std::unique_ptr<llvm::ThreeAddressCode>
445  const Variable * address,
446  const Variable * state,
447  std::shared_ptr<const rvsdg::Type> loadedType,
448  size_t alignment)
449  {
450  auto operation =
451  std::make_unique<LoadNonVolatileOperation>(std::move(loadedType), 1, alignment);
452  return ThreeAddressCode::create(std::move(operation), { address, state });
453  }
454 
455  static std::vector<rvsdg::Output *>
457  rvsdg::Output * address,
458  const std::vector<rvsdg::Output *> & memoryStates,
459  std::shared_ptr<const rvsdg::Type> loadedType,
460  const size_t alignment)
461  {
462  return rvsdg::outputs(&CreateNode(*address, memoryStates, std::move(loadedType), alignment));
463  }
464 
465  static rvsdg::SimpleNode &
467  rvsdg::Region & region,
468  std::unique_ptr<LoadNonVolatileOperation> loadOperation,
469  const std::vector<rvsdg::Output *> & operands)
470  {
471  return rvsdg::SimpleNode::Create(region, std::move(loadOperation), operands);
472  }
473 
474  static std::vector<rvsdg::Output *>
476  rvsdg::Region & region,
477  std::unique_ptr<LoadNonVolatileOperation> loadOperation,
478  const std::vector<rvsdg::Output *> & operands)
479  {
480  return outputs(&CreateNode(region, std::move(loadOperation), operands));
481  }
482 
483  static rvsdg::SimpleNode &
485  rvsdg::Output & address,
486  const std::vector<rvsdg::Output *> & memoryStates,
487  std::shared_ptr<const rvsdg::Type> loadedType,
488  size_t alignment)
489  {
490  std::vector operands({ &address });
491  operands.insert(operands.end(), memoryStates.begin(), memoryStates.end());
492 
493  auto operation = std::make_unique<LoadNonVolatileOperation>(
494  std::move(loadedType),
495  memoryStates.size(),
496  alignment);
497  return CreateNode(*address.region(), std::move(operation), operands);
498  }
499 
500 private:
501  static std::vector<std::shared_ptr<const rvsdg::Type>>
503  {
504  std::vector<std::shared_ptr<const rvsdg::Type>> types(1, PointerType::Create());
505  std::vector<std::shared_ptr<const rvsdg::Type>> states(
508  types.insert(types.end(), states.begin(), states.end());
509  return types;
510  }
511 
512  static std::vector<std::shared_ptr<const rvsdg::Type>>
513  CreateResultTypes(std::shared_ptr<const rvsdg::Type> loadedType, size_t numMemoryStates)
514  {
515  std::vector<std::shared_ptr<const rvsdg::Type>> types(1, std::move(loadedType));
516  std::vector<std::shared_ptr<const rvsdg::Type>> states(
519  types.insert(types.end(), states.begin(), states.end());
520  return types;
521  }
522 };
523 
524 }
525 
526 #endif
static std::shared_ptr< const IOStateType > Create()
Definition: types.cpp:343
static std::vector< std::shared_ptr< const rvsdg::Type > > CreateOperandTypes(size_t numMemoryStates)
Definition: Load.hpp:502
static rvsdg::SimpleNode & CreateNode(rvsdg::Region &region, std::unique_ptr< LoadNonVolatileOperation > loadOperation, const std::vector< rvsdg::Output * > &operands)
Definition: Load.hpp:466
static std::optional< std::vector< rvsdg::Output * > > NormalizeIOBarrierAllocaAddress(const LoadNonVolatileOperation &operation, const std::vector< rvsdg::Output * > &operands)
Redirect the address operand of the LoadNonVolatileOperation from an IOBarrierOperation when it can b...
Definition: Load.cpp:338
static std::vector< rvsdg::Output * > Create(rvsdg::Output *address, const std::vector< rvsdg::Output * > &memoryStates, std::shared_ptr< const rvsdg::Type > loadedType, const size_t alignment)
Definition: Load.hpp:456
static rvsdg::SimpleNode & CreateNode(rvsdg::Output &address, const std::vector< rvsdg::Output * > &memoryStates, std::shared_ptr< const rvsdg::Type > loadedType, size_t alignment)
Definition: Load.hpp:484
std::unique_ptr< Operation > copy() const override
Definition: Load.cpp:35
static std::optional< std::vector< rvsdg::Output * > > NormalizeLoadStoreState(const LoadNonVolatileOperation &operation, const std::vector< rvsdg::Output * > &operands)
If the producer of a load's address is an alloca operation, then we can remove all state edges origin...
Definition: Load.cpp:316
~LoadNonVolatileOperation() noexcept override
static std::optional< std::vector< rvsdg::Output * > > NormalizeDuplicateStates(const LoadNonVolatileOperation &operation, const std::vector< rvsdg::Output * > &operands)
Remove duplicated state operands.
Definition: Load.cpp:327
static std::vector< std::shared_ptr< const rvsdg::Type > > CreateResultTypes(std::shared_ptr< const rvsdg::Type > loadedType, size_t numMemoryStates)
Definition: Load.hpp:513
static std::vector< rvsdg::Output * > Create(rvsdg::Region &region, std::unique_ptr< LoadNonVolatileOperation > loadOperation, const std::vector< rvsdg::Output * > &operands)
Definition: Load.hpp:475
static std::unique_ptr< llvm::ThreeAddressCode > Create(const Variable *address, const Variable *state, std::shared_ptr< const rvsdg::Type > loadedType, size_t alignment)
Definition: Load.hpp:444
static std::optional< std::vector< rvsdg::Output * > > NormalizeLoadAlloca(const LoadNonVolatileOperation &operation, const std::vector< rvsdg::Output * > &operands)
If the producer of a load's address is an alloca operation, then we can remove all state edges origin...
Definition: Load.cpp:305
std::string debug_string() const override
Definition: Load.cpp:29
bool operator==(const Operation &other) const noexcept override
Definition: Load.cpp:20
static std::optional< std::vector< rvsdg::Output * > > NormalizeLoadStore(const LoadNonVolatileOperation &operation, const std::vector< rvsdg::Output * > &operands)
Forwards the value from a store operation.
Definition: Load.cpp:244
size_t NumMemoryStates() const noexcept
Definition: Load.hpp:69
std::shared_ptr< const rvsdg::Type > GetLoadedType() const noexcept
Definition: Load.hpp:63
static rvsdg::Input & AddressInput(const rvsdg::Node &node) noexcept
Definition: Load.hpp:75
static rvsdg::Output & LoadedValueOutput(const rvsdg::Node &node)
Definition: Load.hpp:84
LoadOperation(const std::vector< std::shared_ptr< const rvsdg::Type >> &operandTypes, const std::vector< std::shared_ptr< const rvsdg::Type >> &resultTypes, const size_t numMemoryStates, const size_t alignment)
Definition: Load.hpp:28
static size_t numMemoryStates(const rvsdg::SimpleNode &node) noexcept
Definition: Load.hpp:101
static rvsdg::Node::OutputIteratorRange MemoryStateOutputs(const rvsdg::Node &node) noexcept
Definition: Load.hpp:116
static rvsdg::Output & mapMemoryStateInputToOutput(const rvsdg::Input &input)
Definition: Load.hpp:172
static rvsdg::Node::InputIteratorRange MemoryStateInputs(const rvsdg::Node &node) noexcept
Definition: Load.hpp:139
size_t GetAlignment() const noexcept
Definition: Load.hpp:57
static rvsdg::Input & MapMemoryStateOutputToInput(const rvsdg::Output &output)
Definition: Load.hpp:157
std::string debug_string() const override
Definition: Load.cpp:376
~LoadVolatileOperation() noexcept override
static rvsdg::Input & IOStateInput(const rvsdg::Node &node) noexcept
Definition: Load.hpp:225
static rvsdg::SimpleNode & CreateNode(rvsdg::Output &address, rvsdg::Output &iOState, const std::vector< rvsdg::Output * > &memoryStates, std::shared_ptr< const rvsdg::Type > loadedType, size_t alignment)
Definition: Load.hpp:261
bool operator==(const Operation &other) const noexcept override
Definition: Load.cpp:367
static std::vector< std::shared_ptr< const rvsdg::Type > > CreateResultTypes(std::shared_ptr< const rvsdg::Type > loadedType, size_t numMemoryStates)
Definition: Load.hpp:292
static rvsdg::SimpleNode & CreateNode(rvsdg::Region &region, std::unique_ptr< LoadVolatileOperation > loadOperation, const std::vector< rvsdg::Output * > &operands)
Definition: Load.cpp:388
static std::unique_ptr< llvm::ThreeAddressCode > Create(const Variable *address, const Variable *iOState, const Variable *memoryState, std::shared_ptr< const rvsdg::Type > loadedType, size_t alignment)
Definition: Load.hpp:243
std::unique_ptr< Operation > copy() const override
Definition: Load.cpp:382
static std::vector< std::shared_ptr< const rvsdg::Type > > CreateOperandTypes(size_t numMemoryStates)
Definition: Load.hpp:280
static rvsdg::Output & IOStateOutput(const rvsdg::Node &node)
Definition: Load.hpp:234
static std::shared_ptr< const MemoryStateType > Create()
Definition: types.cpp:379
static std::shared_ptr< const PointerType > Create()
Definition: types.cpp:45
static std::unique_ptr< llvm::ThreeAddressCode > create(std::unique_ptr< rvsdg::SimpleOperation > operation, const std::vector< const Variable * > &operands)
Definition: tac.hpp:135
size_t index() const noexcept
Definition: node.hpp:52
const std::shared_ptr< const rvsdg::Type > & Type() const noexcept
Definition: node.hpp:67
NodeOutput * output(size_t index) const noexcept
Definition: node.hpp:650
rvsdg::Region * region() const noexcept
Definition: node.cpp:151
const std::shared_ptr< const rvsdg::Type > & Type() const noexcept
Definition: node.hpp:366
size_t index() const noexcept
Definition: node.hpp:274
Represent acyclic RVSDG subgraphs.
Definition: region.hpp:213
static SimpleNode & Create(Region &region, std::unique_ptr< Operation > operation, const std::vector< rvsdg::Output * > &operands)
Definition: simple-node.hpp:49
const std::shared_ptr< const rvsdg::Type > & result(size_t index) const noexcept
Definition: operation.cpp:36
SimpleOperation(std::vector< std::shared_ptr< const jlm::rvsdg::Type >> operands, std::vector< std::shared_ptr< const jlm::rvsdg::Type >> results)
Definition: operation.hpp:61
#define JLM_ASSERT(x)
Definition: common.hpp:16
Global memory state passed between functions.
@ State
Designate a state type.
@ Value
Designate a value type.
static std::vector< jlm::rvsdg::Output * > operands(const Node *node)
Definition: node.hpp:1049
static std::vector< jlm::rvsdg::Output * > outputs(const Node *node)
Definition: node.hpp:1058