I tried the following versions of codes with both GCC and Clang/LLVM:
Version 1
#include <stdio.h>
main() {
work();
return 0;
}
work() {
int b;
printf("b: %d \n", b);
}
Version 2
#include <stdio.h>
main() {
work();
return 0;
}
work() {
int a = 1;
int b;
printf("a: %d b: %d \n", a, b);
}
Version 3
#include <stdio.h>
void work() {
int a = 1;
int b;
printf("a: %d b: %d \n", a, b);
}
int main(int argc, char** argv) {
work();
return 0;
}
For GCC,
Version 1 outputs b: 0
.
Version 2 outputs a: 1 b: 4195728
.
Version 3 outputs a: 1 b: 1742650088
.
For Clang
(LLVM),
Version 1 outputs b: 0
.
Version 2 outputs a: 1 b: -1643302816
.
Version 3 outputs a: 1 b: 0
.
I ran the same codes on several machines many times. The ones produced "0"
always produced "0", that is, in these cases, b
was initialized.
Questions:
What makes the GCC and Clang always produce "0" in version 1?
What makes Clang always produce "0" in Version 3?
Reading a variable that is never initialized and for which the address is never taken has undefined behavior. That means that a compiler implementor can chose whatever pleases. So if you'd really interested ask the implementors directly, not us. But as Pascal suggests, I don't think that they will be much interested in your question, and I doubt also that you might find anything interesting about that in their documentation. But since these are open source compilers you may look into the sources to see what they are doing.
In essence you can't expect anything sensible from your code.
If you are seriously be interested in your compilers and what they have to tell you, switch on at least some warnings for your code (use
-Wall
). The first two versions that you have are not even C according to modern standards.