I have gone through some posts related to this topic but was not able to sort out my doubt completely. This might be a very naive question.
I have a header file inline.h
and two translation units main.cpp
and tran.cpp
.
Details of code are as below
#ifndef __HEADER__
#include <stdio.h>
extern inline int func1(void)
{ return 5; }
static inline int func2(void)
{ return 6; }
inline int func3(void)
{ return 7; }
#endif
#define <stdio.h>
#include <inline.h>
int main(int argc, char *argv[])
{
printf("%d\n",func1());
printf("%d\n",func2());
printf("%d\n",func3());
return 0;
}
//(note that the functions are not inline here)
#include <stdio.h>
int func1(void)
{ return 500; }
int func2(void)
{ return 600; }
int func3(void)
{ return 700; }
The above code compiles in g++, but does not compile in gcc (even if you make changes related to gcc like changing the code to .c, not using any C++ header files, etc.). The error displayed is "duplicate definition of inline function - func3".
Can you clarify why this difference is present across compilers?
Also, when you run the program (g++ compiled) by creating two separate compilation units (main.o
and tran.o
) and create an executable a.out
, the output obtained is:
500
6
700
Why does the compiler pick up the definition of the function which is not inline. Actually, since #include
is used to "add" the inline definition I had expected 5,6,7
as the output. My understanding was during compilation since the inline definition is found, the function call would be "replaced" by inline function definition.
Can you please tell me in detailed steps the process of compilation and linking which would lead us to 500,6,700
output. I can only understand the output 6.
This answer is divided into the following sections:
duplicate definition of inline function - func3
problem and why.func3
is a duplicate instead of func1
.g++
The problem can be successfully reproduced by
tran.cpp
to tran.c
gcc -o main main.c tran.c
@K71993 is actually compiling using the old gnu89 inline semantics, which is different from C99. The reason for renaming tran.cpp
to tran.c
is to tell the gcc driver to treat it as C
source instead of C++
source.
The following text is quoted from GCC Document: An Inline Function is As Fast As a Macro explains why func3
is a duplicate definition instead of func1
, since func3
(instead of func1
) is an externally visible symbol (in GNU89 inline semantics)
When an inline function is not static, then the compiler must assume that there may be calls from other source files; since a global symbol can be defined only once in any program, the function must not be defined in the other source files, so the calls therein cannot be integrated. Therefore, a non-static inline function is always compiled on its own in the usual fashion.
If you specify both inline and extern in the function definition, then the definition is used only for inlining. In no case is the function compiled on its own, not even if you refer to its address explicitly. Such an address becomes an external reference, as if you had only declared the function, and had not defined it.
If compiled with C99 standard, i.e., gcc -o main main.c tran.c -std=c99
, the linker will complain that definition of func1
is a duplicate due to the reason that extern inline in C99 is a external definition as mentioned in other posts and comments.
Please also refer to this execellent answer about semantic differents between GNU89 inline
and C99 inline.
g++
.When compiled with g++
, the source program are considered as C++
source. Since func1
, func2
and func3
are defined in multiple translation units and their definitions are different, the One Defintion Rule of C++ is violated. Since the compiler is not required to generate dignostic message when definitions spans multiple translation units, the behavior is undefined.