Blame sysdeps/i386/start.S

Packit 6c4009
/* Startup code compliant to the ELF i386 ABI.
Packit 6c4009
   Copyright (C) 1995-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   In addition to the permissions in the GNU Lesser General Public
Packit 6c4009
   License, the Free Software Foundation gives you unlimited
Packit 6c4009
   permission to link the compiled version of this file with other
Packit 6c4009
   programs, and to distribute those programs without any restriction
Packit 6c4009
   coming from the use of this file. (The GNU Lesser General Public
Packit 6c4009
   License restrictions do apply in other respects; for example, they
Packit 6c4009
   cover modification of the file, and distribution when not linked
Packit 6c4009
   into another program.)
Packit 6c4009
Packit 6c4009
   Note that people who make modified versions of this file are not
Packit 6c4009
   obligated to grant this special exception for their modified
Packit 6c4009
   versions; it is their choice whether to do so. The GNU Lesser
Packit 6c4009
   General Public License gives permission to release a modified
Packit 6c4009
   version without this exception; this exception also makes it
Packit 6c4009
   possible to release a modified version which carries forward this
Packit 6c4009
   exception.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
/* This is the canonical entry point, usually the first thing in the text
Packit 6c4009
   segment.  The SVR4/i386 ABI (pages 3-31, 3-32) says that when the entry
Packit 6c4009
   point runs, most registers' values are unspecified, except for:
Packit 6c4009
Packit 6c4009
   %edx		Contains a function pointer to be registered with `atexit'.
Packit 6c4009
		This is how the dynamic linker arranges to have DT_FINI
Packit 6c4009
		functions called for shared libraries that have been loaded
Packit 6c4009
		before this code runs.
Packit 6c4009
Packit 6c4009
   %esp		The stack contains the arguments and environment:
Packit 6c4009
		0(%esp)			argc
Packit 6c4009
		4(%esp)			argv[0]
Packit 6c4009
		...
Packit 6c4009
		(4*argc)(%esp)		NULL
Packit 6c4009
		(4*(argc+1))(%esp)	envp[0]
Packit 6c4009
		...
Packit 6c4009
					NULL
Packit 6c4009
*/
Packit 6c4009
Packit Service b9ccaa
#include <sysdep.h>
Packit Service b9ccaa
Packit Service b9ccaa
ENTRY (_start)
Packit Service b9ccaa
	/* Clearing frame pointer is insufficient, use CFI.  */
Packit Service b9ccaa
	cfi_undefined (eip)
Packit 6c4009
	/* Clear the frame pointer.  The ABI suggests this be done, to mark
Packit 6c4009
	   the outermost frame obviously.  */
Packit 6c4009
	xorl %ebp, %ebp
Packit 6c4009
Packit 6c4009
	/* Extract the arguments as encoded on the stack and set up
Packit 6c4009
	   the arguments for `main': argc, argv.  envp will be determined
Packit 6c4009
	   later in __libc_start_main.  */
Packit 6c4009
	popl %esi		/* Pop the argument count.  */
Packit 6c4009
	movl %esp, %ecx		/* argv starts just at the current stack top.*/
Packit 6c4009
Packit 6c4009
	/* Before pushing the arguments align the stack to a 16-byte
Packit 6c4009
	(SSE needs 16-byte alignment) boundary to avoid penalties from
Packit 6c4009
	misaligned accesses.  Thanks to Edward Seidl <seidl@janed.com>
Packit 6c4009
	for pointing this out.  */
Packit 6c4009
	andl $0xfffffff0, %esp
Packit 6c4009
	pushl %eax		/* Push garbage because we allocate
Packit 6c4009
				   28 more bytes.  */
Packit 6c4009
Packit 6c4009
	/* Provide the highest stack address to the user code (for stacks
Packit 6c4009
	   which grow downwards).  */
Packit 6c4009
	pushl %esp
Packit 6c4009
Packit 6c4009
	pushl %edx		/* Push address of the shared library
Packit 6c4009
				   termination function.  */
Packit 6c4009
Packit 6c4009
#ifdef PIC
Packit 6c4009
	/* Load PIC register.  */
Packit 6c4009
	call 1f
Packit 6c4009
	addl $_GLOBAL_OFFSET_TABLE_, %ebx
Packit 6c4009
Packit 6c4009
	/* Push address of our own entry points to .fini and .init.  */
Packit 6c4009
	leal __libc_csu_fini@GOTOFF(%ebx), %eax
Packit 6c4009
	pushl %eax
Packit 6c4009
	leal __libc_csu_init@GOTOFF(%ebx), %eax
Packit 6c4009
	pushl %eax
Packit 6c4009
Packit 6c4009
	pushl %ecx		/* Push second argument: argv.  */
Packit 6c4009
	pushl %esi		/* Push first argument: argc.  */
Packit 6c4009
Packit 6c4009
# ifdef SHARED
Packit 6c4009
	pushl main@GOT(%ebx)
Packit 6c4009
# else
Packit 6c4009
	/* Avoid relocation in static PIE since _start is called before
Packit 6c4009
	   it is relocated.  Don't use "leal main@GOTOFF(%ebx), %eax"
Packit 6c4009
	   since main may be in a shared object.  Linker will convert
Packit 6c4009
	   "movl main@GOT(%ebx), %eax" to "leal main@GOTOFF(%ebx), %eax"
Packit 6c4009
	   if main is defined locally.  */
Packit 6c4009
	movl main@GOT(%ebx), %eax
Packit 6c4009
	pushl %eax
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
	/* Call the user's main function, and exit with its value.
Packit 6c4009
	   But let the libc call main.    */
Packit 6c4009
	call __libc_start_main@PLT
Packit 6c4009
#else
Packit 6c4009
	/* Push address of our own entry points to .fini and .init.  */
Packit 6c4009
	pushl $__libc_csu_fini
Packit 6c4009
	pushl $__libc_csu_init
Packit 6c4009
Packit 6c4009
	pushl %ecx		/* Push second argument: argv.  */
Packit 6c4009
	pushl %esi		/* Push first argument: argc.  */
Packit 6c4009
Packit 6c4009
	pushl $main
Packit 6c4009
Packit 6c4009
	/* Call the user's main function, and exit with its value.
Packit 6c4009
	   But let the libc call main.    */
Packit 6c4009
	call __libc_start_main
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
	hlt			/* Crash if somehow `exit' does return.  */
Packit 6c4009
Packit 6c4009
#ifdef PIC
Packit 6c4009
1:	movl	(%esp), %ebx
Packit 6c4009
	ret
Packit 6c4009
#endif
Packit Service b9ccaa
END (_start)
Packit 6c4009
Packit 6c4009
/* To fulfill the System V/i386 ABI we need this symbol.  Yuck, it's so
Packit 6c4009
   meaningless since we don't support machines < 80386.  */
Packit 6c4009
	.section .rodata
Packit 6c4009
	.globl _fp_hw
Packit 6c4009
_fp_hw:	.long 3
Packit 6c4009
	.size _fp_hw, 4
Packit 6c4009
	.type _fp_hw,@object
Packit 6c4009
Packit 6c4009
/* Define a symbol for the first piece of initialized data.  */
Packit 6c4009
	.data
Packit 6c4009
	.globl __data_start
Packit 6c4009
__data_start:
Packit 6c4009
	.long 0
Packit 6c4009
	.weak data_start
Packit 6c4009
	data_start = __data_start