Static linking:
Executable built with libraries at build time. Every time we create a new executable depending on platform.
Dynamic linking:
Linking with libraries at run time. Easy designing, change of code, update etc. These files are first loaded in memory and then linked. Dynamic code is shared between multiple applications.
Important library for C programming is libc.
The library file created with extension .a or .so .
In UNIX environment file name having .a extension called as static library and for .so as dynamic library.
In windows environment dynamic library as .dll extension and static as .libs extension.
Creating a Static and dynamic library in LINUX:
Tool for creating library is ar (library compressed tool).
create source files like first.c and second.c
create relocatable executable files like
gcc -c first.c -o first.o
gcc -c second.c -o second.o
ar rcs lmine.a first.o second.o
Note: rcs stands replace create symbol(variable,function name etc).
Relocatable binary is created in two forms.
position dependent.
position independent.
Above created static library is position dependent.
Position independent code can be created using
gcc -c -fpic first.c -o first.o
gcc -c -fpic second.c -o second.o
gcc -shared -o libmine.so first.o second.o
files which showed in green color required permision for load.
/usr/lib is a default libary repository.
/usr/include is a default header repository.
while using the static or dynamic library we have to create a header file which contains the function prototype.
gcc -I./ app.c -o app
search for include files in current directory.
for using library
gcc app.o -o app -lmine.a
it search in default directory /usr/lib
for searching in current directory.
gcc -c app.o -o app ./mine.a
here we can see, it is having static library but it is also linking some dynamic library called libc etc.
for linking statically give -static option during creation of binary executable.
when we find the object dump we get more information how it linked.
objdump -D app | more
linker creates procedure address table, It contains dynamically linked function and address of it.
Dynamic or Shared libraries can be used by an application as a load time library or run time library. Load time library mainly loaded during init.
load time libraries are those which are loaded during process initialization and remains resident through out the process execution time.
libraries are called as run time when they are loaded by functionality.
it mainly avoid direct reference to library.
for example the source is in mandl.c
void (*ptr) ( void);
void *libptr=dlopen("./mylib.so",RTLD_NOW);
ptr=dlsym(libptr,"func");
ptr(void);
dlclose(libptr);
gcc mandl.c -o mandl -ldl
RTLD_LAZY is used when it takes much time in execution.
Binary Image:
Binary image are of 3 types.
1) Linkable( relocatable).
2) Loadable( shared library with position independent).
3) Executable( functionality + runtime ).
functionality further divided into instruction and data.
each part of binary image is loaded into each block of memory(stack, BSS, Code and data).
To read the header information of binary file objdump or readelf is used.
.text contains all instructions.
.data contains only initialized global variable, static variables etc.
.bss uninitialized global data.
.rodata contains read only data. No write operation performed in this.
other sections are linker specific.
each .o file has section header table.
linker combines all section header table into single program header table with matching the same flags.
The header table in executable binary is called program header table.
loader mainly use this header table to load into process address space.
Various flags in sections.
PROGBITS: code,data,debug information.
SYMTAB and DYNSYM: Symbol table.
STRTAB: String table.
etc.
How a binary executable loaded into memory.
- Think that we want to execute program1.For execution we have to write ./program1
- The shell receives this request to run a program, Checks for the existence of file.
- It verifies the Application Binary Interface(ABI). To know ABI type readelf -a
- It every thing is fine it call loader to load the program.
- Loader checks the program header table part of elf header.
- Using the kernel memory allocator it allocates memory block in user space and maps the segments in executable image.
- Calls the kernel process manager to carry process registration.
- It allocates structures to keep track of information i.e TCB ( task control block) named as task_struct.
- A valid process id is given.
- The process control block is en queued to run queue. ( depends on different queue different run queues are used).
- Depending on the scheduler it selects PCB from run queue.
- During runtime it adds a section named as stack.
- Booting of application starts with an init routine. Initializer is responsible for intilising of stack segment and loading of shared objects or libraries. Link loader is resposible for loading all shared libraries.
- Before loading it resolves all symbols of program header table. During loading of shared libraries symbol entries in PL table are updated with absolute address.
- after initialization it calls main program.
- Process detach is responsible for detaching shared libraries from process. It also releases the stack segment assigned to it.