mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-25 14:06:06 +00:00

Closes #119360. This bug occurs when passing a VLA to `va_arg`. Since the return value is inferred to be an array, it triggers `ScalarExprEmitter::VisitCastExpr`, which converts it to a pointer and subsequently calls `CodeGenFunction::EmitAggExpr`. At this point, because the inferred type is an `AggExpr` instead of a `ScalarExpr`, `ScalarExprEmitter::VisitVAArgExpr` is not invoked, and as a result, `CodeGenFunction::EmitVariablyModifiedType` is also not called, leading to the size of the VLA not being retrieved. The solution is to move the call to `CodeGenFunction::EmitVariablyModifiedType` into `CodeGenFunction::EmitVAArg`, ensuring that the size of the VLA is correctly obtained regardless of whether the expression is an `AggExpr` or a `ScalarExpr`.
129 lines
5.1 KiB
C
129 lines
5.1 KiB
C
// REQUIRES: xcore-registered-target
|
|
// RUN: %clang_cc1 -triple xcore -verify %s
|
|
_Static_assert(sizeof(long long) == 8, "sizeof long long is wrong");
|
|
_Static_assert(_Alignof(long long) == 4, "alignof long long is wrong");
|
|
|
|
_Static_assert(sizeof(double) == 8, "sizeof double is wrong");
|
|
_Static_assert(_Alignof(double) == 4, "alignof double is wrong");
|
|
|
|
// RUN: %clang_cc1 -triple xcore-unknown-unknown -fno-signed-char -fno-common -emit-llvm -o - %s | FileCheck %s
|
|
|
|
// CHECK: target triple = "xcore-unknown-unknown"
|
|
|
|
// CHECK: @cgx = external constant i32, section ".cp.rodata"
|
|
extern const int cgx;
|
|
int fcgx() { return cgx;}
|
|
// CHECK: @g1 ={{.*}} global i32 0, align 4
|
|
int g1;
|
|
// CHECK: @cg1 ={{.*}} constant i32 0, section ".cp.rodata", align 4
|
|
const int cg1;
|
|
|
|
#include <stdarg.h>
|
|
struct x { int a[5]; };
|
|
void f(void*);
|
|
void testva (int n, ...) {
|
|
// CHECK-LABEL: testva
|
|
va_list ap;
|
|
va_start(ap,n);
|
|
// CHECK: [[AP:%[a-z0-9]+]] = alloca ptr, align 4
|
|
// CHECK: [[V5:%[a-z0-9]+]] = alloca %struct.x, align 4
|
|
// CHECK: [[TMP:%[a-z0-9]+]] = alloca [4 x i32], align 4
|
|
// CHECK: call void @llvm.va_start.p0(ptr [[AP]])
|
|
|
|
char* v1 = va_arg (ap, char*);
|
|
f(v1);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 4
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: [[V1:%[a-z0-9]+]] = load ptr, ptr [[I]]
|
|
// CHECK: store ptr [[V1]], ptr [[V:%[a-z0-9]+]], align 4
|
|
// CHECK: [[V2:%[a-z0-9]+]] = load ptr, ptr [[V]], align 4
|
|
// CHECK: call void @f(ptr noundef [[V2]])
|
|
|
|
char v2 = va_arg (ap, char); // expected-warning{{second argument to 'va_arg' is of promotable type 'char'}}
|
|
f(&v2);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 4
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: [[V1:%[a-z0-9]+]] = load i8, ptr [[I]]
|
|
// CHECK: store i8 [[V1]], ptr [[V:%[a-z0-9]+]], align 1
|
|
// CHECK: call void @f(ptr noundef [[V]])
|
|
|
|
int v3 = va_arg (ap, int);
|
|
f(&v3);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 4
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: [[V1:%[a-z0-9]+]] = load i32, ptr [[I]]
|
|
// CHECK: store i32 [[V1]], ptr [[V:%[a-z0-9]+]], align 4
|
|
// CHECK: call void @f(ptr noundef [[V]])
|
|
|
|
long long int v4 = va_arg (ap, long long int);
|
|
f(&v4);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 8
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: [[V1:%[a-z0-9]+]] = load i64, ptr [[I]]
|
|
// CHECK: store i64 [[V1]], ptr [[V:%[a-z0-9]+]], align 4
|
|
// CHECK: call void @f(ptr noundef [[V]])
|
|
|
|
struct x v5 = va_arg (ap, struct x); // typical aggregate type
|
|
f(&v5);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[P:%[a-z0-9]+]] = load ptr, ptr [[I]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 4
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: call void @llvm.memcpy.p0.p0.i32(ptr align 4 [[V5]], ptr align 4 [[P]], i32 20, i1 false)
|
|
// CHECK: call void @f(ptr noundef [[V5]])
|
|
|
|
// an unusual aggregate type
|
|
int* v6 = va_arg (ap, int[4]); // expected-warning{{second argument to 'va_arg' is of array type 'int[4]'}}
|
|
f(v6);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[P:%[a-z0-9]+]] = load ptr, ptr [[I]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 4
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: call void @llvm.memcpy.p0.p0.i32(ptr align 4 [[TMP]], ptr align 4 [[P]], i32 16, i1 false)
|
|
// CHECK: [[V2:%[a-z0-9]+]] = getelementptr inbounds [4 x i32], ptr [[TMP]], i32 0, i32 0
|
|
// CHECK: store ptr [[V2]], ptr [[V:%[a-z0-9]+]], align 4
|
|
// CHECK: [[V3:%[a-z0-9]+]] = load ptr, ptr [[V]], align 4
|
|
// CHECK: call void @f(ptr noundef [[V3]])
|
|
|
|
double v7 = va_arg (ap, double);
|
|
f(&v7);
|
|
// CHECK: [[I:%[a-z0-9]+]] = load ptr, ptr [[AP]]
|
|
// CHECK: [[IN:%[a-z0-9]+]] = getelementptr inbounds i8, ptr [[I]], i32 8
|
|
// CHECK: store ptr [[IN]], ptr [[AP]]
|
|
// CHECK: [[V1:%[a-z0-9]+]] = load double, ptr [[I]]
|
|
// CHECK: store double [[V1]], ptr [[V:%[a-z0-9]+]], align 4
|
|
// CHECK: call void @f(ptr noundef [[V]])
|
|
}
|
|
|
|
void testbuiltin (void) {
|
|
// CHECK-LABEL: testbuiltin
|
|
// CHECK: call i32 @llvm.xcore.getid()
|
|
// CHECK: call i32 @llvm.xcore.getps(i32 {{%[a-z0-9]+}})
|
|
// CHECK: call i32 @llvm.xcore.bitrev(i32 {{%[a-z0-9]+}})
|
|
// CHECK: call void @llvm.xcore.setps(i32 {{%[a-z0-9]+}}, i32 {{%[a-z0-9]+}})
|
|
volatile int i = __builtin_getid();
|
|
volatile unsigned int ui = __builtin_getps(i);
|
|
ui = __builtin_bitrev(ui);
|
|
__builtin_setps(i,ui);
|
|
|
|
// CHECK: store volatile i32 0, ptr {{%[a-z0-9]+}}, align 4
|
|
// CHECK: store volatile i32 1, ptr {{%[a-z0-9]+}}, align 4
|
|
// CHECK: store volatile i32 -1, ptr {{%[a-z0-9]+}}, align 4
|
|
volatile int res;
|
|
res = __builtin_eh_return_data_regno(0);
|
|
res = __builtin_eh_return_data_regno(1);
|
|
res = __builtin_eh_return_data_regno(2);
|
|
}
|
|
|
|
// CHECK-LABEL: define{{.*}} zeroext i8 @testchar()
|
|
// CHECK: ret i8 -1
|
|
char testchar (void) {
|
|
return (char)-1;
|
|
}
|
|
|
|
// CHECK-NOT: frame-pointer
|