2009-12-16 04:14:24 +08:00
|
|
|
// RUN: %clang_cc1 -fsyntax-only -verify %s
|
2007-08-29 00:20:14 +08:00
|
|
|
|
|
|
|
void *test1(void) { return 0; }
|
|
|
|
|
2008-02-07 06:48:16 +08:00
|
|
|
void test2 (const struct {int a;} *x) {
|
2015-04-11 09:53:13 +08:00
|
|
|
// expected-note@-1 {{variable 'x' declared const here}}
|
|
|
|
|
|
|
|
x->a = 10;
|
|
|
|
// expected-error-re@-1 {{cannot assign to variable 'x' with const-qualified type 'const struct (anonymous struct at {{.*}}assign.c:5:19) *'}}
|
2008-02-07 06:48:16 +08:00
|
|
|
}
|
2008-02-10 00:59:44 +08:00
|
|
|
|
|
|
|
typedef int arr[10];
|
|
|
|
void test3() {
|
2018-02-09 17:30:42 +08:00
|
|
|
const arr b; // expected-note {{variable 'b' declared const here}}
|
|
|
|
const int b2[10]; // expected-note {{variable 'b2' declared const here}}
|
|
|
|
b[4] = 1; // expected-error {{cannot assign to variable 'b' with const-qualified type 'const arr' (aka 'int const[10]')}}
|
|
|
|
b2[4] = 1; // expected-error {{cannot assign to variable 'b2' with const-qualified type 'const int [10]'}}
|
2008-02-10 00:59:44 +08:00
|
|
|
}
|
[Sema] Disallow assigning record lvalues with nested const-qualified fields.
Summary:
According to C99 6.3.2.1p1, structs and unions with nested
const-qualified fields (that is, const-qualified fields
declared at some recursive level of the aggregate) are not
modifiable lvalues. However, Clang permits assignments of
these lvalues.
With this patch, we both prohibit the assignment of records
with const-qualified fields and emit a best-effort diagnostic.
This fixes https://bugs.llvm.org/show_bug.cgi?id=31796 .
Committing on behalf of bevinh (Bevin Hansson).
Reviewers: rtrieu, rsmith, bjope
Reviewed By: bjope
Subscribers: Ka-Ka, rogfer01, bjope, fhahn, cfe-commits
Differential Revision: https://reviews.llvm.org/D37254
llvm-svn: 313628
2017-09-19 21:10:30 +08:00
|
|
|
|
|
|
|
typedef struct I {
|
|
|
|
const int a; // expected-note 4{{nested data member 'a' declared const here}} \
|
|
|
|
expected-note 6{{data member 'a' declared const here}}
|
|
|
|
} I;
|
|
|
|
typedef struct J {
|
|
|
|
struct I i;
|
|
|
|
} J;
|
|
|
|
typedef struct K {
|
|
|
|
struct J *j;
|
|
|
|
} K;
|
|
|
|
|
|
|
|
void testI(struct I i1, struct I i2) {
|
|
|
|
i1 = i2; // expected-error {{cannot assign to variable 'i1' with const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testJ1(struct J j1, struct J j2) {
|
|
|
|
j1 = j2; // expected-error {{cannot assign to variable 'j1' with nested const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testJ2(struct J j, struct I i) {
|
|
|
|
j.i = i; // expected-error {{cannot assign to non-static data member 'i' with const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testK1(struct K k, struct J j) {
|
|
|
|
*(k.j) = j; // expected-error {{cannot assign to lvalue with nested const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testK2(struct K k, struct I i) {
|
|
|
|
k.j->i = i; // expected-error {{cannot assign to non-static data member 'i' with const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
|
|
|
|
void testI_(I i1, I i2) {
|
|
|
|
i1 = i2; // expected-error {{cannot assign to variable 'i1' with const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testJ1_(J j1, J j2) {
|
|
|
|
j1 = j2; // expected-error {{cannot assign to variable 'j1' with nested const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testJ2_(J j, I i) {
|
|
|
|
j.i = i; // expected-error {{cannot assign to non-static data member 'i' with const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testK1_(K k, J j) {
|
|
|
|
*(k.j) = j; // expected-error {{cannot assign to lvalue with nested const-qualified data member 'a'}}
|
|
|
|
}
|
|
|
|
void testK2_(K k, I i) {
|
|
|
|
k.j->i = i; // expected-error {{cannot assign to non-static data member 'i' with const-qualified data member 'a'}}
|
|
|
|
}
|