Consider following C program (see live demo here).
const int main = 195;
I know that in the real world no programmer writes code like this, because it serves no useful purpose and doesn't make any sense. But when I remove the const
keyword from above the program it immediately results in a segmentation fault. Why? I am eager to know the reason behind this.
GCC 4.8.2 gives following warning when compiling it.
warning: 'main' is usually a function [-Wmain]
const int main = 195; ^
Why does the presence and absence of const
keyword make a difference here in the behavior of the program?
Observe how the value 195 corresponds to the ret
(return from function) instruction on 8086 compatibles. This definition of main
thus behaves as if you defined it as int main() {}
when executed.
On some platforms, const
data is loaded into an executable but not writeable memory region whereas mutable data (i.e. data not qualified const
) is loaded into a writeable but not executable memory region. For this reason, the program “works” when you declare main
as const
but not when you leave off the const
qualifier.
Traditionally, binaries contained three segments:
text
segment is (if supported by the architecture) write-protected and executable, and contains executable code, variables of static storage duration qualified const
, and string literalsdata
segment is writeable and cannot be executed. It contains variables not qualified const
with static storage duration and (at runtime) objects with allocated storage durationbss
segment is similar to the data
segment but is initialized to all zeroes. It contains variables of static storage duration not qualified const
that have been declared without an initializerstack
segment is not present in the binary and contains variables with automatic storage durationRemoving the const
qualifier from the variable main
causes it to be moved from the text
to the data
segment, which isn't executable, causing the segmentation violation you observe.
Modern platforms often have further segments (e.g. a rodata
segment for data that is neither writeable nor executable) so please don't take this as an accurate description of your platform without consulting platform-specific documentation.
Please understand that not making main
a function is usually incorrect, although technically a platform could allow main
to be declared as a variable, cf. ISO 9899:2011 §5.1.2.2.1 ¶1, emphasis mine:
1 The function called at program startup is named
main
. The implementation declares no prototype for this function. It shall be defined with a return type ofint
and with no parameters (...) or with two parameters (...) or equivalent; or in some other implementation-defined manner.
In C, main
at global scope is almost always a function.
To use main
as a variable at global scope makes the behaviour of the program undefined.
(It just might be the case that when you write const
the compiler optimises out the variable to a constant and so your program behaviour is different. But the program behaviour is still undefined).
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With