Blame sysdeps/x86_64/start.S

Packit 6c4009
/* Startup code compliant to the ELF x86-64 ABI.
Packit 6c4009
   Copyright (C) 2001-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Andreas Jaeger <aj@suse.de>, 2001.
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
   %rdx		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
   %rsp		The stack contains the arguments and environment:
Packit 6c4009
		0(%rsp)				argc
Packit 6c4009
		LP_SIZE(%rsp)			argv[0]
Packit 6c4009
		...
Packit 6c4009
		(LP_SIZE*argc)(%rsp)		NULL
Packit 6c4009
		(LP_SIZE*(argc+1))(%rsp)	envp[0]
Packit 6c4009
		...
Packit 6c4009
						NULL
Packit 6c4009
*/
Packit 6c4009
Packit 6c4009
#include <sysdep.h>
Packit 6c4009
Packit 6c4009
ENTRY (_start)
Packit 6c4009
	/* Clearing frame pointer is insufficient, use CFI.  */
Packit 6c4009
	cfi_undefined (rip)
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 __libc_start_main (int (*main) (int, char **, char **),
Packit 6c4009
		   int argc, char *argv,
Packit 6c4009
		   void (*init) (void), void (*fini) (void),
Packit 6c4009
		   void (*rtld_fini) (void), void *stack_end).
Packit 6c4009
	   The arguments are passed via registers and on the stack:
Packit 6c4009
	main:		%rdi
Packit 6c4009
	argc:		%rsi
Packit 6c4009
	argv:		%rdx
Packit 6c4009
	init:		%rcx
Packit 6c4009
	fini:		%r8
Packit 6c4009
	rtld_fini:	%r9
Packit 6c4009
	stack_end:	stack.	*/
Packit 6c4009
Packit 6c4009
	mov %RDX_LP, %R9_LP	/* Address of the shared library termination
Packit 6c4009
				   function.  */
Packit 6c4009
#ifdef __ILP32__
Packit 6c4009
	mov (%rsp), %esi	/* Simulate popping 4-byte argument count.  */
Packit 6c4009
	add $4, %esp
Packit 6c4009
#else
Packit 6c4009
	popq %rsi		/* Pop the argument count.  */
Packit 6c4009
#endif
Packit 6c4009
	/* argv starts just at the current stack top.  */
Packit 6c4009
	mov %RSP_LP, %RDX_LP
Packit 6c4009
	/* Align the stack to a 16 byte boundary to follow the ABI.  */
Packit 6c4009
	and  $~15, %RSP_LP
Packit 6c4009
Packit 6c4009
	/* Push garbage because we push 8 more bytes.  */
Packit 6c4009
	pushq %rax
Packit 6c4009
Packit 6c4009
	/* Provide the highest stack address to the user code (for stacks
Packit 6c4009
	   which grow downwards).  */
Packit 6c4009
	pushq %rsp
Packit 6c4009
Packit 6c4009
#ifdef PIC
Packit 6c4009
	/* Pass address of our own entry points to .fini and .init.  */
Packit 6c4009
	mov __libc_csu_fini@GOTPCREL(%rip), %R8_LP
Packit 6c4009
	mov __libc_csu_init@GOTPCREL(%rip), %RCX_LP
Packit 6c4009
Packit 6c4009
	mov main@GOTPCREL(%rip), %RDI_LP
Packit 6c4009
#else
Packit 6c4009
	/* Pass address of our own entry points to .fini and .init.  */
Packit 6c4009
	mov $__libc_csu_fini, %R8_LP
Packit 6c4009
	mov $__libc_csu_init, %RCX_LP
Packit 6c4009
Packit 6c4009
	mov $main, %RDI_LP
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.  Since __libc_start_main in
Packit 6c4009
	   libc.so is called very early, lazy binding isn't relevant
Packit 6c4009
	   here.  Use indirect branch via GOT to avoid extra branch
Packit 6c4009
	   to PLT slot.  In case of static executable, ld in binutils
Packit 6c4009
	   2.26 or above can convert indirect branch into direct
Packit 6c4009
	   branch.  */
Packit 6c4009
	call *__libc_start_main@GOTPCREL(%rip)
Packit 6c4009
Packit 6c4009
	hlt			/* Crash if somehow `exit' does return.	 */
Packit 6c4009
END (_start)
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