1030b8cb1SEduard Zingerman // NOTE: Assertions have been autogenerated by utils/update_cc_test_checks.py UTC_ARGS: --version 2
2030b8cb1SEduard Zingerman // REQUIRES: bpf-registered-target
3*a312dd68SFangrui Song // RUN: %clang_cc1 -triple bpfel -disable-llvm-passes -emit-llvm -o - %s | FileCheck %s
4030b8cb1SEduard Zingerman
5030b8cb1SEduard Zingerman // Check that call to preserve.static.offset is generated when bitfield
6030b8cb1SEduard Zingerman // from a struct marked with __attribute__((preserve_static_offset)) is
7030b8cb1SEduard Zingerman // accessed.
8030b8cb1SEduard Zingerman
9030b8cb1SEduard Zingerman #define __ctx __attribute__((preserve_static_offset))
10030b8cb1SEduard Zingerman
11030b8cb1SEduard Zingerman struct foo {
12030b8cb1SEduard Zingerman unsigned a:1;
13030b8cb1SEduard Zingerman } __ctx;
14030b8cb1SEduard Zingerman
15030b8cb1SEduard Zingerman // CHECK-LABEL: define dso_local void @lvalue_bitfield
16030b8cb1SEduard Zingerman // CHECK-SAME: (ptr noundef [[P:%.*]]) #[[ATTR0:[0-9]+]] {
17030b8cb1SEduard Zingerman // CHECK-NEXT: entry:
18030b8cb1SEduard Zingerman // CHECK-NEXT: [[P_ADDR:%.*]] = alloca ptr, align 8
19030b8cb1SEduard Zingerman // CHECK-NEXT: store ptr [[P]], ptr [[P_ADDR]], align 8
20030b8cb1SEduard Zingerman // CHECK-NEXT: [[TMP0:%.*]] = load ptr, ptr [[P_ADDR]], align 8
21030b8cb1SEduard Zingerman // CHECK-NEXT: [[TMP1:%.*]] = call ptr @llvm.preserve.static.offset(ptr [[TMP0]])
22030b8cb1SEduard Zingerman // CHECK-NEXT: [[BF_LOAD:%.*]] = load i8, ptr [[TMP1]], align 4
23030b8cb1SEduard Zingerman // CHECK-NEXT: [[BF_CLEAR:%.*]] = and i8 [[BF_LOAD]], -2
24030b8cb1SEduard Zingerman // CHECK-NEXT: [[BF_SET:%.*]] = or i8 [[BF_CLEAR]], 1
25030b8cb1SEduard Zingerman // CHECK-NEXT: store i8 [[BF_SET]], ptr [[TMP1]], align 4
26030b8cb1SEduard Zingerman // CHECK-NEXT: ret void
27030b8cb1SEduard Zingerman //
lvalue_bitfield(struct foo * p)28030b8cb1SEduard Zingerman void lvalue_bitfield(struct foo *p) {
29030b8cb1SEduard Zingerman p->a = 1;
30030b8cb1SEduard Zingerman }
31