data:image/s3,"s3://crabby-images/5e3ae/5e3aec325ca2bca13222a113c714389ba308a9ed" alt=""
Some dark corners of C
.pdfdata:image/s3,"s3://crabby-images/d8e27/d8e273e8bce9bab4ac0e4972a6faa23be1c5c1ff" alt=""
Some dark corners of C
-std=c99 -pedantic -Wall -Warn-me-harder
Rob Kendrick <rob.kendrick@codethink.co.uk>
Stolen from Rob Kendrick <rjek+dark corners@rjek.com>
data:image/s3,"s3://crabby-images/d8ada/d8ada4e26dd9a8e4e56d8c59219544061f11cd16" alt=""
Dark corner of C
Here it is!
data:image/s3,"s3://crabby-images/0e0cd/0e0cdef4af564e3f4a5c9026775cbb909265d997" alt=""
Shadowy escape
int |
x = 42; |
int |
x = 42; |
int func() { |
int func() { |
||
int x = 3840; |
int x = 3840; |
||
{ |
|
{ |
|
|
return x; |
|
extern int x; |
} |
|
|
return x; |
} |
|
} |
|
|
|
} |
|
Returns 3840 |
Returns 42 |
data:image/s3,"s3://crabby-images/1aaca/1aaca10d4c8f0d1757ff527f9c5c3e4c6e37b688" alt=""
Bizarre keyword reuse
void foo(int x[static 10]);
Passed array must be at least 10 entries.
foo.c:8:2: warning: array argument is too small; contains 5 elements, callee requires at least 10 [-Warray-bounds]
void foo(char x[static 1]);
Free compile-time non-NULL check!
foo.c:8:2: warning: null passed to a callee which requires a non-null argument [- Wnonnull]
data:image/s3,"s3://crabby-images/01e71/01e71ba8c19ba7dbeba7a28e0c768d64e904ecf9" alt=""
Warning: Don't breathe this
int x = 'FOO!';
Yes, single quotes.
Yes, that's right, it makes demons fly out of your nose!
(On my system, 1,179,602,721.)
If you do this, I will find you.
data:image/s3,"s3://crabby-images/e2262/e2262ddafe8381f1a1d4f82dce8d343c1502ef78" alt=""
[ ] is just + in disguise
assert(spong[x] == x[spong]); assert(spong[2] == 2[spong]);
d = "0123456789abcdef"[n]; d = n["0123456789abcdef"];
data:image/s3,"s3://crabby-images/b5bd9/b5bd9cabb6be104bbc0d0b9f699d7975fda77d0a" alt=""
Pointer aliasing
void foo1(int *x, int *y, int *z) {
*x += *z; |
Compiler |
|
*y += *z; |
||
doesn't |
||
} |
||
know that *z |
||
movl (%rdx), %eax |
||
won't also be |
||
addl %eax, (%rdi) |
||
*x, so loads |
||
movl (%rdx), %eax |
it twice :( |
|
addl %eax, (%rsi) |
|
|
ret |
|
data:image/s3,"s3://crabby-images/02b55/02b55f649ffe5f8319244df6e87cf735cdc5ec4d" alt=""
Pointer aliasing: Fixed?
void foo2(int *x, int *y, int *z) {
int w = *z; *x += w;
*y += w;
}
movl (%rdx), %eax addl %eax, (%rdi) addl %eax, (%rsi) ret
Only one load, at the cost of beauty.
data:image/s3,"s3://crabby-images/b677c/b677cc33bef0b611a93748355d1282a55e3e8b10" alt=""
Pointer aliasing: Fixed!
void foo3(int *x, int *y, |
|
|
int * |
z) { |
|
*x += *z; |
|
|
*y += *z; |
'restrict' |
|
} |
||
means we |
||
movl (%rdx), %eax |
||
promise not |
||
addl %eax, (%rdi) |
||
to mess |
||
addl %eax, (%rsi) |
||
about. |
||
ret |
||
|
data:image/s3,"s3://crabby-images/c5e87/c5e877328cd8731b213cba35203668e908d08513" alt=""
Counting up ...
int fact1(int n) { int i, fact = 1; for (i = 1; i <= n;
i++) fact *= i;
return fact;
}
fact1: |
|
movl |
$1, %eax |
testl |
%edi, %edi |
jle |
.exit |
xorl |
%ecx, %ecx |
.loop: |
|
incl |
%ecx |
imull |
%ecx, %eax |
cmpl |
%ecx, %edi |
jne |
.loop |
.exit |
|
ret |
|