Starting code
We used the simple hello world code for this lab:
#include <stdio.h> int main() { printf("Hello World!\n"); }
We compiled this code using the GCC compiler with the following arguments.
-g # enable debugging information -O0 # do not optimize (that's a capital letter and then the digit zero) -fno-builtin # do not use builtin function optimizations
In the objdump you could see that <main> calls to <printf> to print Hello World.
1. Add the compiler option -static. Note and explain the change in size, section headers, and the function call.
When you add the option -static, the compiler imports the entire library when it only needs one function. This causes the file size to be much larger.
2. Remove the compiler option -fno-builtin. Note and explain the change in the function call.
<printf> replaced with <puts>
3. Remove the compiler option -g. Note and explain the change in size, section headers, and disassembly output.
When you enter the command objdump --source you normally get the written code together with the assembler code. This is handy when you want to see which assembler code belongs to which C code. When you remove the option -g you remove all the debug information. This also means when you enter the command objdump --source, you do not get the C code in the file because this is seen as debug information.
4. Add additional arguments to the printf() function in your program. Note which register each argument is placed in.
In the AARCH64 architecture 7 arguents get saved in registers, the overflow gets pushed on the stack.
3. Remove the compiler option -g. Note and explain the change in size, section headers, and disassembly output.
When you enter the command objdump --source you normally get the written code together with the assembler code. This is handy when you want to see which assembler code belongs to which C code. When you remove the option -g you remove all the debug information. This also means when you enter the command objdump --source, you do not get the C code in the file because this is seen as debug information.
4. Add additional arguments to the printf() function in your program. Note which register each argument is placed in.
In the AARCH64 architecture 7 arguents get saved in registers, the overflow gets pushed on the stack.
In the INTEL architecture only 5 arguments get stored in registerd, and the overflow also gets pushed on the stack.
5. Move the printf() call to a separate function named output(), and call that function from main(). Explain the changes in the object code.
5.1
Our task was to move the printf() function to a seperate function output().
We compiled this with the same arguments stated above. The objdump now shows:
<main> calls <output> and that calls <printf>
<output> is the same as <main> without our output() function.
5.2
We now compiled the following code to inspect differences when you add parameters.
In the objdump you see that it loads the given parameter to the x0 register.
The output function in the <main> adds a pointer to the parameter to register x0 before calling <output>.
<printf> in the <output> then takes the x0 parameter and puts it in the x1 register to put it as argument 2 for printf.
File size increased slightly with each change.
6. Remove -O0 and add -O3 to the gcc options. Note and explain the difference in the compiled code.These options have to do with optimizations. The compiler gets 'smarter' and deletes lines of code it doesn't need. For example, in O0 it sets up the stack but it never uses the stack after that. These lines are deleted in the O3 option. Another example is that after the <printf> is done, it doesn't return to main. It goes back to whatever called <main>.
5. Move the printf() call to a separate function named output(), and call that function from main(). Explain the changes in the object code.
5.1
Our task was to move the printf() function to a seperate function output().
#include <stdio.h> void output(){ printf("Hello World!\n"); } int main() { output(); }
We compiled this with the same arguments stated above. The objdump now shows:
<main> calls <output> and that calls <printf>
<output> is the same as <main> without our output() function.
5.2
We now compiled the following code to inspect differences when you add parameters.
#include <stdio.h> void output(char a[]){ printf(a); } int main() { output("Hello World!\n"); }
In the objdump you see that it loads the given parameter to the x0 register.
The output function in the <main> adds a pointer to the parameter to register x0 before calling <output>.
<printf> in the <output> then takes the x0 parameter and puts it in the x1 register to put it as argument 2 for printf.
File size increased slightly with each change.
6. Remove -O0 and add -O3 to the gcc options. Note and explain the difference in the compiled code.These options have to do with optimizations. The compiler gets 'smarter' and deletes lines of code it doesn't need. For example, in O0 it sets up the stack but it never uses the stack after that. These lines are deleted in the O3 option. Another example is that after the <printf> is done, it doesn't return to main. It goes back to whatever called <main>.
No comments:
Post a Comment