Issue 133038
Summary [LICM] LICM breaks `initializes` attribute
Labels miscompilation, loopoptim, generated by fuzzer
Assignees
Reporter dtcxzyw
    Reproducer: https://godbolt.org/z/evTP9P17j
```
; bin/opt -passes=licm reduced.ll -S
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-f80:128-n8:16:32:64-S128"
target triple = "x86_64-unknown-linux-gnu"

define void @_ZN25test7425_51072c957f3e023d3fn317hab8ab70cb1ea6b04E(ptr noalias writable align 2 dereferenceable(12) initializes((0, 12)) %0) {
  %2 = alloca [56 x i8], align 8
  %3 = getelementptr i8, ptr %0, i64 8
  %4 = getelementptr i8, ptr %0, i64 10
  br label %5

5: ; preds = %10, %1
  %6 = getelementptr i8, ptr %2, i64 16
  %7 = getelementptr i8, ptr %2, i64 16
  store i16 0, ptr %3, align 2
  store i16 0, ptr %4, align 2
  %8 = load <4 x i32>, ptr %6, align 8
  store <4 x i32> %8, ptr %7, align 8
  br label %9

9: ; preds = %5
  switch i8 1, label %11 [
    i8 1, label %11
 i8 0, label %10
  ]

10:                                               ; preds = %9
  store <4 x i16> zeroinitializer, ptr %0, align 2
  br label %5

11:                                               ; preds = %9, %9
 ret void
}

define void @_ZN25test7425_51072c957f3e023d4main17hb2c14928af806df2E() {
  %1 = alloca [64 x i8], align 8
  call void @_ZN25test7425_51072c957f3e023d3fn317hab8ab70cb1ea6b04E(ptr %1)
  ret void
}

```
After LICM, the first memory access to `%0` is `load <4 x i16>, ptr %0, align 1`, which breaks the `initializes` attribute:
> Initialization of memory means the first memory access is a non-volatile, non-atomic write.
> The initializes attribute does not imply writeonly since initializes allows reading from the pointer after writing.

```
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-i128:128-f80:128-n8:16:32:64-S128"
target triple = "x86_64-unknown-linux-gnu"

define void @_ZN25test7425_51072c957f3e023d3fn317hab8ab70cb1ea6b04E(ptr noalias writable align 2 dereferenceable(12) initializes((0, 12)) %0) {
  %2 = alloca [56 x i8], align 8
  %3 = getelementptr i8, ptr %0, i64 8
  %4 = getelementptr i8, ptr %0, i64 10
  %5 = getelementptr i8, ptr %2, i64 16
  %6 = getelementptr i8, ptr %2, i64 16
  %.promoted = load <4 x i32>, ptr %5, align 8
  %.promoted1 = load <4 x i16>, ptr %0, align 1
  br label %7

7: ; preds = %11, %1
  %8 = phi <4 x i16> [ zeroinitializer, %11 ], [ %.promoted1, %1 ]
  %9 = phi <4 x i32> [ %9, %11 ], [ %.promoted, %1 ]
  br label %10

10: ; preds = %7
  switch i8 1, label %12 [
    i8 1, label %12
    i8 0, label %11
  ]

11: ; preds = %10
  br label %7

12: ; preds = %10, %10
  %.lcssa2 = phi <4 x i16> [ %8, %10 ], [ %8, %10 ]
  %.lcssa = phi <4 x i32> [ %9, %10 ], [ %9, %10 ]
  store i16 0, ptr %3, align 2
  store i16 0, ptr %4, align 2
  store <4 x i32> %.lcssa, ptr %5, align 8
  store <4 x i16> %.lcssa2, ptr %0, align 1
  ret void
}
```

Should we clarify the LangRef wording on `initializes` to allow loads before the initialization is complete, if they are not externally observable?

This case is generated by rustlantis + llubi.
Original rust mir program: https://gist.github.com/dtcxzyw/fcf0905a562f8734428e0b216198e0c0

cc @haopliu @nikic 

_______________________________________________
llvm-bugs mailing list
llvm-bugs@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-bugs

Reply via email to