2005-04-16 15:20:36 -07:00
/*
* Kevin D . Kissell , kevink @ mips and Carsten Langgaard , carstenl @ mips . com
* Copyright ( C ) 2000 MIPS Technologies , Inc . All rights reserved .
*
* This program is free software ; you can distribute it and / or modify it
* under the terms of the GNU General Public License ( Version 2 ) as
* published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License
* for more details .
*
* You should have received a copy of the GNU General Public License along
* with this program ; if not , write to the Free Software Foundation , Inc . ,
* 59 Temple Place - Suite 330 , Boston MA 02111 - 1307 , USA .
*
* Routines corresponding to Linux kernel FP context
* manipulation primitives for the Algorithmics MIPS
* FPU Emulator
*/
# include <linux/sched.h>
# include <asm/processor.h>
# include <asm/signal.h>
# include <asm/uaccess.h>
2008-07-15 19:57:31 +03:00
# include <asm/fpu.h>
2005-04-16 15:20:36 -07:00
# include <asm/fpu_emulator.h>
# define SIGNALLING_NAN 0x7ff800007ff80000LL
void fpu_emulator_init_fpu ( void )
{
static int first = 1 ;
int i ;
if ( first ) {
first = 0 ;
printk ( " Algorithmics/MIPS FPU Emulator v1.5 \n " ) ;
}
2006-05-16 01:26:03 +09:00
current - > thread . fpu . fcr31 = 0 ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < 32 ; i + + ) {
2006-05-16 01:26:03 +09:00
current - > thread . fpu . fpr [ i ] = SIGNALLING_NAN ;
2005-04-16 15:20:36 -07:00
}
}
/*
* Emulator context save / restore to / from a signal context
* presumed to be on the user stack , and therefore accessed
* with appropriate macros from uaccess . h
*/
2007-03-10 01:07:45 +09:00
int fpu_emulator_save_context ( struct sigcontext __user * sc )
2005-04-16 15:20:36 -07:00
{
int i ;
int err = 0 ;
for ( i = 0 ; i < 32 ; i + + ) {
err | =
2006-05-16 01:26:03 +09:00
__put_user ( current - > thread . fpu . fpr [ i ] , & sc - > sc_fpregs [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 01:26:03 +09:00
err | = __put_user ( current - > thread . fpu . fcr31 , & sc - > sc_fpc_csr ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
2007-03-10 01:07:45 +09:00
int fpu_emulator_restore_context ( struct sigcontext __user * sc )
2005-04-16 15:20:36 -07:00
{
int i ;
int err = 0 ;
for ( i = 0 ; i < 32 ; i + + ) {
err | =
2006-05-16 01:26:03 +09:00
__get_user ( current - > thread . fpu . fpr [ i ] , & sc - > sc_fpregs [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 01:26:03 +09:00
err | = __get_user ( current - > thread . fpu . fcr31 , & sc - > sc_fpc_csr ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
2005-09-03 15:56:16 -07:00
# ifdef CONFIG_64BIT
2005-04-16 15:20:36 -07:00
/*
* This is the o32 version
*/
2007-03-10 01:07:45 +09:00
int fpu_emulator_save_context32 ( struct sigcontext32 __user * sc )
2005-04-16 15:20:36 -07:00
{
int i ;
int err = 0 ;
for ( i = 0 ; i < 32 ; i + = 2 ) {
err | =
2006-05-16 01:26:03 +09:00
__put_user ( current - > thread . fpu . fpr [ i ] , & sc - > sc_fpregs [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 01:26:03 +09:00
err | = __put_user ( current - > thread . fpu . fcr31 , & sc - > sc_fpc_csr ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
2007-03-10 01:07:45 +09:00
int fpu_emulator_restore_context32 ( struct sigcontext32 __user * sc )
2005-04-16 15:20:36 -07:00
{
int i ;
int err = 0 ;
for ( i = 0 ; i < 32 ; i + = 2 ) {
err | =
2006-05-16 01:26:03 +09:00
__get_user ( current - > thread . fpu . fpr [ i ] , & sc - > sc_fpregs [ i ] ) ;
2005-04-16 15:20:36 -07:00
}
2006-05-16 01:26:03 +09:00
err | = __get_user ( current - > thread . fpu . fcr31 , & sc - > sc_fpc_csr ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
# endif