/* This file is part of the sample code and exercises * used by the class "Advanced Programming in the UNIX * Environment" taught by Jan Schaumann * at Stevens Institute of * Technology. * * This file is in the public domain. * * You don't have to, but if you feel like * acknowledging where you got this code, you may * reference me by name, email address, or point * people to the course website: * https://stevens.netmeister.org/631/ */ /* This program allows us to visualize the layout of a * process in memory, printing the addresses of * various parts of the program, including components * in the text, data, and bss segments as well as on * the heap and the stack. * * This program can also illustrate a stack overflow * if compiled with '-DSTACKOVERFLOW'. */ #include #include #include #include #define ARRAY_SIZE 16 #define MALLOC_SIZE 32 char array[ARRAY_SIZE]; char *string = "a string"; char *string2; int num = 10; int num2; extern char **environ; void func(int); void func2(const char *); int main(int argc, char **argv, char **envp) { int vars; char *ptr; char func_array[ARRAY_SIZE]; vars = 0; char **tmp = envp; while (*tmp++) { vars++; } (void)printf("High address (args and env):\n"); (void)printf("----------------------------\n"); (void)printf("envp[%d] at : 0x%12lX\n", vars, (unsigned long)&envp[vars]); (void)printf("environ[%d] at : 0x%12lX\n", vars, (unsigned long)&environ[vars]); (void)printf("envp[0] at : 0x%12lX\n", (unsigned long)envp); (void)printf("environ[0] at : 0x%12lX\n", (unsigned long)environ); (void)printf("last arg at : 0x%12lX\n", (unsigned long)&argv[argc]); (void)printf("first arg at : 0x%12lX\n", (unsigned long)&argv[0]); (void)printf("\n"); (void)printf("Stack:\n"); (void)printf("------\n"); (void)printf("First variable inside main at : 0x%12lX\n", (unsigned long)&vars); (void)printf("func_array[] ends at : 0x%12lX\n", (unsigned long)&func_array[ARRAY_SIZE]); (void)printf("func_array[] (like 'array[]', but on stack) begins at : 0x%12lX\n", (unsigned long)&func_array[0]); (void)printf("argc at : 0x%12lX\n", (unsigned long)&argc); (void)printf("argv at : 0x%12lX\n", (unsigned long)&argv); (void)printf("envp at : 0x%12lX\n", (unsigned long)&envp); func2("from main"); func(0); (void)printf("\n"); (void)printf("Heap:\n"); (void)printf("-----\n"); if ((ptr = malloc(MALLOC_SIZE)) == NULL) { err(EXIT_FAILURE, "unable to allocate memory"); /* NOTREACHED */ } (void)printf("malloced area ends at : 0x%12lX\n", (unsigned long)ptr+MALLOC_SIZE); (void)printf("malloced area begins at : 0x%12lX\n", (unsigned long)ptr); free(ptr); (void)printf("\n"); (void)printf("Uninitialized Data (BSS):\n"); (void)printf("-------------------------\n"); (void)printf("array[] ends at : 0x%12lX\n", (unsigned long)&array[ARRAY_SIZE]); (void)printf("array[] (uninitialized, fixed-size char * on BSS) from : 0x%12lX\n", (unsigned long)&array[0]); (void)printf("num2 (uninitialized global int) at : 0x%12lX\n", (unsigned long)&num2); (void)printf("string2 (uninitialized global char *) at : 0x%12lX\n", (unsigned long)&string2); (void)printf("extern **environ at : 0x%12lX\n", (unsigned long)&environ); (void)printf("\n"); (void)printf("Initialized Data:\n"); (void)printf("-----------------\n"); (void)printf("num (initialized global int) at : 0x%12lX\n", (unsigned long)&num); (void)printf("string (initialized global char *) at : 0x%12lX\n", (unsigned long)&string); (void)printf("\n"); (void)printf("Text Segment:\n"); (void)printf("-------------\n"); (void)printf("func2 (function) at : 0x%12lX\n", (unsigned long)&func2); (void)printf("func (function) at : 0x%12lX\n", (unsigned long)&func); (void)printf("main (function) at : 0x%12lX\n", (unsigned long)&main); (void)printf("\n"); return EXIT_SUCCESS; } void func(int recurse) { int fint; char *msg = "from func"; /* Change this value to 0 and note how * the location of where it is stored * changes from the Data to BSS segment. */ static int n = 1; (void)printf("func frame at : 0x%12lX\n", (unsigned long)&fint); if (recurse) { msg = "recursive"; } (void)printf("static int n within func at : 0x%12lX\n", (unsigned long)&n); printf("func (called %5d times): frame at : 0x%12lX\n", n, (unsigned long)&fint); n++; func2(msg); } void func2(const char *how) { int fint; (void)printf("func2 (%s): frame at : 0x%12lX\n", how, (unsigned long)&fint); #ifdef STACKOVERFLOW func(1); #endif }