xref: /titanic_50/usr/src/lib/libc/amd64/sys/getcontext.s (revision 9a70fc3be3b1e966bf78825cdb8d509963a6f0a1)
17c478bd9Sstevel@tonic-gate/*
27c478bd9Sstevel@tonic-gate * CDDL HEADER START
37c478bd9Sstevel@tonic-gate *
47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the
58cd45542Sraf * Common Development and Distribution License (the "License").
68cd45542Sraf * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate *
87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate * and limitations under the License.
127c478bd9Sstevel@tonic-gate *
137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate *
197c478bd9Sstevel@tonic-gate * CDDL HEADER END
207c478bd9Sstevel@tonic-gate */
218cd45542Sraf
227c478bd9Sstevel@tonic-gate/*
238cd45542Sraf * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
247c478bd9Sstevel@tonic-gate * Use is subject to license terms.
257c478bd9Sstevel@tonic-gate */
267c478bd9Sstevel@tonic-gate
27*9a70fc3bSMark J. Nelson	.file	"getcontext.s"
287c478bd9Sstevel@tonic-gate
297c478bd9Sstevel@tonic-gate#include <sys/asm_linkage.h>
307c478bd9Sstevel@tonic-gate
317c478bd9Sstevel@tonic-gate	ANSI_PRAGMA_WEAK(getcontext,function)
327c478bd9Sstevel@tonic-gate	ANSI_PRAGMA_WEAK(swapcontext,function)
337c478bd9Sstevel@tonic-gate
347c478bd9Sstevel@tonic-gate#include "SYS.h"
357c478bd9Sstevel@tonic-gate#include <../assym.h>
367c478bd9Sstevel@tonic-gate
377c478bd9Sstevel@tonic-gate/*
387c478bd9Sstevel@tonic-gate * getcontext() is written in assembler since it has to capture the correct
397c478bd9Sstevel@tonic-gate * machine state of the calle.
407c478bd9Sstevel@tonic-gate *
417c478bd9Sstevel@tonic-gate * As swapcontext() is actually equivalent to getcontext() + setcontext(),
427c478bd9Sstevel@tonic-gate * swapcontext() shares the most code with getcontext().
437c478bd9Sstevel@tonic-gate */
447c478bd9Sstevel@tonic-gate
457c478bd9Sstevel@tonic-gate#define	GETCONTEXT_IMPL(offset)						\
467c478bd9Sstevel@tonic-gate	pushq	%rdi;		/* preserve the ucontext_t pointer */	\
478cd45542Sraf	call	__getcontext;						\
487c478bd9Sstevel@tonic-gate				/* call getcontext: syscall */		\
497c478bd9Sstevel@tonic-gate	popq	%rdx;							\
507c478bd9Sstevel@tonic-gate	andl	%eax, %eax;	/* if (error_return_from_syscall) */	\
517c478bd9Sstevel@tonic-gate	je	1f;							\
527c478bd9Sstevel@tonic-gate	addq	$offset, %rsp;						\
537c478bd9Sstevel@tonic-gate	ret;			/*	then just return */		\
547c478bd9Sstevel@tonic-gate1:									\
557c478bd9Sstevel@tonic-gate	/*								\
567c478bd9Sstevel@tonic-gate	 * fix up %rsp and %rip						\
577c478bd9Sstevel@tonic-gate	 */								\
587c478bd9Sstevel@tonic-gate	addq	$UC_MCONTEXT_GREGS, %rdx;				\
597c478bd9Sstevel@tonic-gate				/* &ucp->uc_mcontext.gregs */		\
607c478bd9Sstevel@tonic-gate	movq	offset+0(%rsp), %rax;					\
617c478bd9Sstevel@tonic-gate				/* read return PC from stack */		\
627c478bd9Sstevel@tonic-gate	movq	%rax, RIP_OFF (%rdx);					\
637c478bd9Sstevel@tonic-gate				/* store ret PC in EIP of env var */	\
647c478bd9Sstevel@tonic-gate	leaq	offset+8(%rsp), %rax;					\
657c478bd9Sstevel@tonic-gate				/* get caller's sp at time of call */	\
667c478bd9Sstevel@tonic-gate	movq	%rax, RSP_OFF (%rdx);					\
677c478bd9Sstevel@tonic-gate				/* store the sp into UESP of env var */	\
687c478bd9Sstevel@tonic-gate	xorq	%rax, %rax;	/* return 0 */				\
697c478bd9Sstevel@tonic-gate	movq	%rax, RAX_OFF (%rdx);					\
707c478bd9Sstevel@tonic-gate				/* getcontext returns 0 after setcontext */
717c478bd9Sstevel@tonic-gate
727c478bd9Sstevel@tonic-gate/*
737c478bd9Sstevel@tonic-gate * getcontext(ucontext_t *ucp)
747c478bd9Sstevel@tonic-gate */
757c478bd9Sstevel@tonic-gate
767c478bd9Sstevel@tonic-gate	ENTRY(getcontext)
777c478bd9Sstevel@tonic-gate	GETCONTEXT_IMPL(0)
787c478bd9Sstevel@tonic-gate	ret
797c478bd9Sstevel@tonic-gate	SET_SIZE(getcontext)
807c478bd9Sstevel@tonic-gate
817c478bd9Sstevel@tonic-gate/*
827c478bd9Sstevel@tonic-gate * swapcontext(ucontext_t *oucp, const ucontext_t *ucp)
837c478bd9Sstevel@tonic-gate */
847c478bd9Sstevel@tonic-gate
857c478bd9Sstevel@tonic-gate	ENTRY(swapcontext)
867c478bd9Sstevel@tonic-gate	pushq	%rsi			/* preserve the 2nd argument */
877c478bd9Sstevel@tonic-gate
887c478bd9Sstevel@tonic-gate	GETCONTEXT_IMPL(8)
897c478bd9Sstevel@tonic-gate
907c478bd9Sstevel@tonic-gate	/* call setcontext */
917c478bd9Sstevel@tonic-gate	popq	%rdi
928cd45542Sraf	call	setcontext
937c478bd9Sstevel@tonic-gate	ret
947c478bd9Sstevel@tonic-gate	SET_SIZE(swapcontext)
95