The C++ build pipeline
DECLARATIONS AND DEFINITIONS IN three EXAMPLES
1.
2.
3.
DEclarations
Definitions
int main()
{
sqr(3);
}
int sqr(int x)
{
return x * x;
}
main.cpp
int sqr(int x)
{
return x * x;
}
int main()
{
sqr(3);
}
main.cpp
main.cpp
int sqr(int);
int main()
{
sqr(3);
}
int sqr(int x)
{
return x * x;
}
HOW TO USE MULTIPLE FILES IN C++?
MULTIPLE FILES
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
int main()
{
sqr(3);
cube(2);
}
main.cpp
1.
2.
HOW TO USE MULTIPLE FILES IN C++?
MULTIPLE FILES
int sqr(int);
int cube(int);
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
int main()
{
sqr(3);
cube(2);
}
main.cpp
1.
2.
HOW TO USE MULTIPLE FILES IN C++?
MULTIPLE FILES
int sqr(int);
int cube(int);
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
2.
#include "math.h"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
3.
int sqr(int);
int cube(int);
math.h
HOW TO USE MULTIPLE FILES IN C++?
MULTIPLE FILES
Header Files
Source Files
(DEclarations)
(Definitions)
#include "math.h"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int);
int cube(int);
math.h
HOW TO USE MULTIPLE FILES IN C++?
MULTIPLE FILES
Header Files
Source Files
(DEclarations)
(Definitions)
math.h io.h utility.h error.h net.h log.h
math.cpp io.cpp utility.cpp error.cpp net.cpp log.cpp
main.cpp
THE one definition rule - the reason for headers
O
R
D
#include "math.cpp"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
THE one definition rule - the reason for headers
O
R
D
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
THE one definition rule - the reason for headers
O
R
D
#include "math.cpp"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int);
int cube(int);
math.h
THE one definition rule - the reason for headers
O
R
D
#include "math.h"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int);
int cube(int);
math.h
THE one definition rule - the reason for headers
O
R
D
int sqr(int);
int cube(int);
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int x)
{
return x * x;
}
int cube(int x)
{
return x * x * x;
}
math.cpp
int sqr(int);
int cube(int);
math.h
int sqr(int);
int cube(int);
shapes.h
namespaces
#include "math.h"
#include "shapes.h"
int main()
{
sqr(3);
cube(2);
}
main.cpp
int sqr(int);
int cube(int);
math.h
int sqr(int);
int cube(int);
shapes.h
...
math.cpp
...
shapes.cpp
namespaces
#include "math.h"
#include "shapes.h"
int main()
{
math_sqr(3);
shapes_cube(2);
}
main.cpp
int math_sqr(int);
int math_cube(int);
math.h
int shapes_sqr(int);
int shapes_cube(int);
shapes.h
...
math.cpp
...
shapes.cpp
namespaces
#include "math.h"
#include "shapes.h"
int main()
{
math::sqr(3);
shapes::cube(2);
}
main.cpp
namespace math {
int sqr(int);
int cube(int);
}
math.h
namespace shapes {
int sqr(int);
int cube(int);
}
shapes.h
...
math.cpp
...
shapes.cpp
namespaces
#include "math.h"
#include "shapes.h"
int main()
{
math::sqr(3);
shapes::cube(2);
}
main.cpp
namespace math {
int sqr(int);
int cube(int);
}
math.h
namespace shapes {
int sqr(int);
int cube(int);
}
shapes.h
...
math.cpp
...
shapes.cpp
namespace std {
istream cin;
ostream cout;
...
}
iostream
namespaces
#include "math.h"
#include "shapes.h"
using namespace math;
int main()
{
sqr(3);
shapes::cube(2);
}
main.cpp
namespace math {
int sqr(int);
int cube(int);
}
math.h
namespace shapes {
int sqr(int);
int cube(int);
}
shapes.h
...
math.cpp
...
shapes.cpp
namespace std {
istream cin;
ostream cout;
...
}
iostream
namespaces
#include "math.h"
#include "shapes.h"
int main()
{
using namespace math;
sqr(3);
shapes::cube(2);
}
int func(int x)
{
math::sqr(3);
}
main.cpp
namespace math {
int sqr(int);
int cube(int);
}
math.h
namespace shapes {
int sqr(int);
int cube(int);
}
shapes.h
...
math.cpp
...
shapes.cpp
namespace std {
istream cin;
ostream cout;
...
}
iostream
The C++ build pipeline
What is
?
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
The C++ build pipeline
What is
?
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
Source Files
Preprocessed Code
Object
Files
Static
Libraries
Executable
File
Dynamic
Libraries
Library
Header
Files
Header Files
HOW IS A C++ program STRUCTURED?
Source Files
Header Files
Internal
Compile-time
Runtime
Resource Files
Textures, Audio
Config Files
...
Static Libraries
Dynamic Libraries
Library Header Files
EXternal
Compile-time
Runtime
Resource Files
Textures, Audio
...
The C++ build pipeline
What is
?
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
Preprocessor
#ifndef HEADER_GUARD
#define HEADER_GUARD
#include <type_traits>
#include "myheader.h"
#define PI 3.1415926
#define ERROR_MSG "Exception access violation"
#define ADD(ARG1, ARG2) ((ARG1) + (ARG2))
#define STRINGIFY(ARG1) #ARG1
#define CONCAT(ARG1, ARG2) ARG1 ## ARG2
#define FWD(...) std::forward<decltype(__VA_ARGS__)>(__VA_ARGS__)
#ifdef __APPLE__
#include "apple_specific.h"
#elif __linux__
#include "linux_specific.h"
#elif _WIN32
#include "windows_specific.h"
#else
#error Unsupported platform
#endif
#line 42
#endif // HEADER_GUARD
- Header guards
- Includes
- Constants
- Compile-time computation
- Source code transformation
- Cross-platform compatibility
- Platform detection
- Conditional compilation
- Reporting compilation errors
- Code generation support
Preprocessor
#include <iostream>
#define SQR(x) x * x
int main()
{
std::cout << SQR(2);
}
Preprocessor
#include <iostream>
#define SQR(x) x * x
int main()
{
std::cout << 2 * 2;
}
Preprocessor
#include <iostream>
#define SQR(x) x * x
int main()
{
std::cout << SQR(2 + 3);
}
Preprocessor
#include <iostream>
#define SQR(x) x * x
int main()
{
std::cout << 2 + 3 * 2 + 3;
}
Preprocessor
#include <iostream>
#define SQR(x) x * x
int main()
{
std::cout << 2 + 3 * 2 + 3;
//std::cout << (2 + 3) * (2 + 3);
}
Preprocessor
#include <iostream>
#define SQR(x) ((x) * (x))
int main()
{
std::cout << SQR(2 + 3);
}
Preprocessor
#include <iostream>
#define SQR(x) ((x) * (x))
int main()
{
std::cout << ((2 + 3) * (2 + 3));
}
Preprocessor
#include <iostream>
#define SQR(x) ((x) * (x))
int main()
{
int x = 3;
std::cout << SQR(++x);
}
Preprocessor
#include <iostream>
#define SQR(x) ((x) * (x))
int main()
{
int x = 3;
std::cout << ((++x) * (++x));
}
Preprocessor
#include <iostream>
#define SQR(x) ((x) * (x))
int main()
{
int x = 3;
std::cout << ((++x) * (++x));
//std::cout << 20;
}
Preprocessor
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
#ifndef _GLIBCXX_IOSTREAM
#define _GLIBCXX_IOSTREAM 1
#pragma GCC system_header
#include <bits/c++config.h>
#include <ostream>
#include <istream>
namespace std
{
extern istream cin;
extern ostream cout;
extern ostream cerr;
extern ostream clog;
#ifdef _GLIBCXX_USE_WCHAR_T
extern wistream wcin;
extern wostream wcout;
extern wostream wcerr;
extern wostream wclog;
#endif
static ios_base::Init __ioinit;
} // namespace std
#endif /* _GLIBCXX_IOSTREAM */
Preprocessor
#ifndef _GLIBCXX_IOSTREAM
#define _GLIBCXX_IOSTREAM 1
#pragma GCC system_header
#include <bits/c++config.h>
#include <ostream>
#include <istream>
namespace std
{
extern istream cin;
extern ostream cout;
extern ostream cerr;
extern ostream clog;
#ifdef _GLIBCXX_USE_WCHAR_T
extern wistream wcin;
extern wostream wcout;
extern wostream wcerr;
extern wostream wclog;
#endif
static ios_base::Init __ioinit;
} // namespace std
#endif /* _GLIBCXX_IOSTREAM */
int main()
{
std::cout << "Hello World!\n";
}
#ifndef _GLIBCXX_IOSTREAM
#define _GLIBCXX_IOSTREAM 1
#pragma GCC system_header
#include <bits/c++config.h>
#include <ostream>
#include <istream>
namespace std
{
extern istream cin;
extern ostream cout;
extern ostream cerr;
extern ostream clog;
#ifdef _GLIBCXX_USE_WCHAR_T
extern wistream wcin;
extern wostream wcout;
extern wostream wcerr;
extern wostream wclog;
#endif
static ios_base::Init __ioinit;
} // namespace std
#endif /* _GLIBCXX_IOSTREAM */
Compiler
Preprocessor
...
int main()
{
std::cout << "Hello World!\n";
}
Compiler
The C++ build pipeline
What is
?
...
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Compiler
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Compiler
main: # @main
push rax
mov edi, offset _ZSt4cout
mov esi, offset .L.str
mov edx, 13
call _ZSt16__ostream_insertIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_PKS3_l
xor eax, eax
pop rcx
ret
_GLOBAL__sub_I_example.cpp: # @_GLOBAL__sub_I_example.cpp
push rax
mov edi, offset _ZStL8__ioinit
call _ZNSt8ios_base4InitC1Ev
mov edi, offset _ZNSt8ios_base4InitD1Ev
mov esi, offset _ZStL8__ioinit
mov edx, offset __dso_handle
pop rax
jmp __cxa_atexit # TAILCALL
.L.str:
.asciz "Hello World!\n"
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Compiler
$ ./a.out
Hello World!
$
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Compiler
$ ./a.out
Hello World!
$
=
=
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Compiler
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
$ ./a.out
Hello World!
$
Compiler
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
ff 25 da 2f 00 00
68 01 00 00 00
e9 df 00 00 00
ff 25 d2 2f 00 00
...
main: # @main
push rax
mov edi, offset _ZSt4cout
mov esi, offset .L.str
mov edx, 13
call _ZSt16__ostream_insertIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_PKS3_l
xor eax, eax
pop rcx
ret
_GLOBAL__sub_I_example.cpp: # @_GLOBAL__sub_I_example.cpp
push rax
mov edi, offset _ZStL8__ioinit
call _ZNSt8ios_base4InitC1Ev
mov edi, offset _ZNSt8ios_base4InitD1Ev
mov esi, offset _ZStL8__ioinit
mov edx, offset __dso_handle
pop rax
jmp __cxa_atexit # TAILCALL
.L.str:
.asciz "Hello World!\n"
$ ./a.out
Hello World!
$
Compiler
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
ff 25 da 2f 00 00
68 01 00 00 00
e9 df 00 00 00
ff 25 d2 2f 00 00
...
main: # @main
push rax
mov edi, offset _ZSt4cout
mov esi, offset .L.str
mov edx, 13
call _ZSt16__ostream_insertIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_PKS3_l
xor eax, eax
pop rcx
ret
_GLOBAL__sub_I_example.cpp: # @_GLOBAL__sub_I_example.cpp
push rax
mov edi, offset _ZStL8__ioinit
call _ZNSt8ios_base4InitC1Ev
mov edi, offset _ZNSt8ios_base4InitD1Ev
mov esi, offset _ZStL8__ioinit
mov edx, offset __dso_handle
pop rax
jmp __cxa_atexit # TAILCALL
.L.str:
.asciz "Hello World!\n"
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Assembler
$ ./a.out
Hello World!
$
Compiler
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
ff 25 da 2f 00 00
68 01 00 00 00
e9 df 00 00 00
ff 25 d2 2f 00 00
...
main: # @main
push rax
mov edi, offset _ZSt4cout
mov esi, offset .L.str
mov edx, 13
call _ZSt16__ostream_insertIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_PKS3_l
xor eax, eax
pop rcx
ret
_GLOBAL__sub_I_example.cpp: # @_GLOBAL__sub_I_example.cpp
push rax
mov edi, offset _ZStL8__ioinit
call _ZNSt8ios_base4InitC1Ev
mov edi, offset _ZNSt8ios_base4InitD1Ev
mov esi, offset _ZStL8__ioinit
mov edx, offset __dso_handle
pop rax
jmp __cxa_atexit # TAILCALL
.L.str:
.asciz "Hello World!\n"
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Assembler
Linker
Compiler
01010111011010101010101010101010101010100100101110100000010101010101000001111101010110101010101010100001011111001010111011010101010101010101010101010100100101110100001010110101010101010100001011111001010111011010101010101010101010101010100100101110100001011010100001011111010100111000011100101110101001000011011001010111110010011101010111101001000101011110101101010101010101101010101010110111110100011110100101110000101100101011100101001110
ff 25 da 2f 00 00
68 01 00 00 00
e9 df 00 00 00
ff 25 d2 2f 00 00
...
main: # @main
push rax
mov edi, offset _ZSt4cout
mov esi, offset .L.str
mov edx, 13
call _ZSt16__ostream_insertIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_PKS3_l
xor eax, eax
pop rcx
ret
_GLOBAL__sub_I_example.cpp: # @_GLOBAL__sub_I_example.cpp
push rax
mov edi, offset _ZStL8__ioinit
call _ZNSt8ios_base4InitC1Ev
mov edi, offset _ZNSt8ios_base4InitD1Ev
mov esi, offset _ZStL8__ioinit
mov edx, offset __dso_handle
pop rax
jmp __cxa_atexit # TAILCALL
.L.str:
.asciz "Hello World!\n"
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Assembler
Linker
The C++ build pipeline
What is
?
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
Linker
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Linker
The C++ build pipeline
#include <iostream>
int main()
{
std::cout << "Hello World!\n";
}
$ ./a.out
Hello World!
$
Compiler
Linker
Assembler
Preprocessor
Source Files
Preprocessed Code
Object
Files
Static
Libraries
Executable
File
Dynamic
Libraries
Library
Header
Files
Header Files
The C++ Build Pipeline
By Jan Bielak
The C++ Build Pipeline
A presentation about the C++ build pipeline. It is presented here: https://www.youtube.com/watch?v=Ev8DrjtBtNg .
- 690