Blame sysdeps/unix/sysv/linux/mips/vfork.S

Packit 6c4009
/* Copyright (C) 2005-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
   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
/* vfork() is just a special case of clone().  */
Packit 6c4009
Packit 6c4009
#include <sys/asm.h>
Packit 6c4009
#include <sysdep.h>
Packit 6c4009
#include <asm/unistd.h>
Packit 6c4009
#include <sgidefs.h>
Packit 6c4009
#include <tls.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* int vfork() */
Packit 6c4009
Packit 6c4009
	.text
Packit 6c4009
	.set		nomips16
Packit 6c4009
LOCALSZ= 1
Packit 6c4009
FRAMESZ= (((NARGSAVE+LOCALSZ)*SZREG)+ALSZ)&ALMASK
Packit 6c4009
GPOFF= FRAMESZ-(1*SZREG)
Packit 6c4009
NESTED(__libc_vfork,FRAMESZ,sp)
Packit 6c4009
#ifdef __PIC__
Packit 6c4009
	SETUP_GP
Packit 6c4009
#endif
Packit 6c4009
	PTR_SUBU sp, FRAMESZ
Packit 6c4009
	cfi_adjust_cfa_offset (FRAMESZ)
Packit 6c4009
	SETUP_GP64_REG (a5, __libc_vfork)
Packit 6c4009
#ifdef __PIC__
Packit 6c4009
	SAVE_GP (GPOFF)
Packit 6c4009
#endif
Packit 6c4009
#ifdef PROF
Packit 6c4009
# if (_MIPS_SIM != _ABIO32)
Packit 6c4009
	PTR_S		a5, GPOFF(sp)
Packit 6c4009
# endif
Packit 6c4009
	.set		noat
Packit 6c4009
	move		$1, ra
Packit 6c4009
# if (_MIPS_SIM == _ABIO32)
Packit 6c4009
	subu		sp,sp,8
Packit 6c4009
# endif
Packit 6c4009
	jal		_mcount
Packit 6c4009
	.set		at
Packit 6c4009
# if (_MIPS_SIM != _ABIO32)
Packit 6c4009
	PTR_L		a5, GPOFF(sp)
Packit 6c4009
# endif
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
	PTR_ADDU	sp, FRAMESZ
Packit 6c4009
	cfi_adjust_cfa_offset (-FRAMESZ)
Packit 6c4009
Packit 6c4009
	li		a0, 0x4112	/* CLONE_VM | CLONE_VFORK | SIGCHLD */
Packit 6c4009
	move		a1, sp
Packit 6c4009
Packit 6c4009
	/* Do the system call */
Packit 6c4009
	li		v0,__NR_clone
Packit 6c4009
	syscall
Packit 6c4009
Packit 6c4009
	cfi_remember_state
Packit 6c4009
	bnez		a3,L(error)
Packit 6c4009
Packit 6c4009
	/* Successful return from the parent or child.  */
Packit 6c4009
	RESTORE_GP64_REG
Packit 6c4009
	ret
Packit 6c4009
Packit 6c4009
	/* Something bad happened -- no child created.  */
Packit 6c4009
L(error):
Packit 6c4009
	cfi_restore_state
Packit 6c4009
#ifdef __PIC__
Packit 6c4009
	PTR_LA		t9, __syscall_error
Packit 6c4009
	RESTORE_GP64_REG
Packit 6c4009
	jr		t9
Packit 6c4009
#else
Packit 6c4009
	RESTORE_GP64_REG
Packit 6c4009
	j		__syscall_error
Packit 6c4009
#endif
Packit 6c4009
	END(__libc_vfork)
Packit 6c4009
Packit 6c4009
#if IS_IN (libc)
Packit 6c4009
weak_alias (__libc_vfork, vfork)
Packit 6c4009
strong_alias (__libc_vfork, __vfork)
Packit 6c4009
libc_hidden_def (__vfork)
Packit 6c4009
#endif