2012-08-04 18:06:17 -04:00
|
|
|
/*******************************************************************************
|
|
|
|
|
2013-08-04 14:24:59 -04:00
|
|
|
Copyright(C) Jonas 'Sortie' Termansen 2011, 2012, 2014.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
2013-07-10 09:26:01 -04:00
|
|
|
This file is part of Sortix.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
2013-07-10 09:26:01 -04:00
|
|
|
Sortix is free software: you can redistribute it and/or modify it under the
|
|
|
|
terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation, either version 3 of the License, or (at your option) any later
|
|
|
|
version.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
2013-07-10 09:26:01 -04:00
|
|
|
Sortix is distributed in the hope that 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.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
2013-07-10 09:26:01 -04:00
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
|
|
Sortix. If not, see <http://www.gnu.org/licenses/>.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
2013-07-10 09:26:01 -04:00
|
|
|
x86-family/float.cpp
|
|
|
|
Handles saving and restoration of floating point numbers.
|
2012-08-04 18:06:17 -04:00
|
|
|
|
|
|
|
*******************************************************************************/
|
|
|
|
|
2013-10-26 20:42:10 -04:00
|
|
|
#include <assert.h>
|
|
|
|
|
2013-01-09 17:30:36 -05:00
|
|
|
#include <sortix/kernel/interrupt.h>
|
2013-10-26 20:42:10 -04:00
|
|
|
#include <sortix/kernel/kernel.h>
|
2013-05-12 18:41:30 -04:00
|
|
|
#include <sortix/kernel/thread.h>
|
2013-01-09 17:30:36 -05:00
|
|
|
|
2012-08-04 18:06:17 -04:00
|
|
|
#include "float.h"
|
|
|
|
|
|
|
|
namespace Sortix {
|
|
|
|
namespace Float {
|
|
|
|
|
|
|
|
static Thread* fputhread;
|
2013-08-04 14:24:59 -04:00
|
|
|
bool fpu_is_enabled = false;
|
2012-08-04 18:06:17 -04:00
|
|
|
|
|
|
|
static inline void InitFPU()
|
|
|
|
{
|
|
|
|
asm volatile ("fninit");
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void SaveState(uint8_t* dest)
|
|
|
|
{
|
2012-09-22 08:57:20 -04:00
|
|
|
assert( (((unsigned long) dest) & (16UL-1UL)) == 0 );
|
2012-08-04 18:06:17 -04:00
|
|
|
asm volatile ("fxsave (%0)" : : "r"(dest));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void LoadState(const uint8_t* src)
|
|
|
|
{
|
2012-09-22 08:57:20 -04:00
|
|
|
assert( (((unsigned long) src) & (16UL-1UL)) == 0 );
|
2012-08-04 18:06:17 -04:00
|
|
|
asm volatile ("fxrstor (%0)" : : "r"(src));
|
|
|
|
}
|
|
|
|
|
2013-08-04 14:24:59 -04:00
|
|
|
void Yield()
|
|
|
|
{
|
|
|
|
Thread* thread = CurrentThread();
|
|
|
|
|
|
|
|
Interrupt::Disable();
|
|
|
|
|
|
|
|
bool fpu_was_enabled = fpu_is_enabled;
|
|
|
|
|
|
|
|
// The FPU contains the registers for this thread.
|
|
|
|
if ( fputhread == thread )
|
|
|
|
{
|
|
|
|
if ( !fpu_was_enabled )
|
|
|
|
EnableFPU();
|
|
|
|
SaveState(thread->fpuenvaligned);
|
|
|
|
fputhread = NULL;
|
|
|
|
DisableFPU();
|
|
|
|
}
|
|
|
|
|
|
|
|
// This thread has used the FPU once.
|
|
|
|
else if ( thread->fpuinitialized )
|
|
|
|
{
|
|
|
|
// Nothing needs to be done, the FPU is owned by another thread and the
|
|
|
|
// FPU registers are already stored in the thread structure.
|
|
|
|
}
|
|
|
|
|
|
|
|
// This thread has never used the FPU and needs its registers initialized.
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( !fpu_was_enabled )
|
|
|
|
EnableFPU();
|
|
|
|
|
|
|
|
if ( fputhread )
|
|
|
|
SaveState(fputhread->fpuenvaligned);
|
|
|
|
|
|
|
|
InitFPU();
|
|
|
|
SaveState(thread->fpuenvaligned);
|
|
|
|
|
|
|
|
if ( fputhread )
|
|
|
|
LoadState(fputhread->fpuenvaligned);
|
|
|
|
|
|
|
|
if ( !fpu_was_enabled )
|
|
|
|
DisableFPU();
|
|
|
|
}
|
|
|
|
|
|
|
|
Interrupt::Enable();
|
|
|
|
}
|
|
|
|
|
2012-08-04 18:06:17 -04:00
|
|
|
static void OnFPUAccess(CPU::InterruptRegisters* /*regs*/, void* /*user*/)
|
|
|
|
{
|
|
|
|
EnableFPU();
|
|
|
|
Thread* thread = CurrentThread();
|
|
|
|
if ( thread == fputhread )
|
|
|
|
return;
|
|
|
|
if ( fputhread )
|
|
|
|
SaveState(fputhread->fpuenvaligned);
|
|
|
|
fputhread = thread;
|
|
|
|
if ( !thread->fpuinitialized )
|
|
|
|
{
|
|
|
|
InitFPU();
|
|
|
|
thread->fpuinitialized = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
LoadState(thread->fpuenvaligned);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
fputhread = CurrentThread();
|
2012-09-22 08:57:20 -04:00
|
|
|
assert(fputhread);
|
2012-08-04 18:06:17 -04:00
|
|
|
Interrupt::RegisterHandler(7, OnFPUAccess, NULL);
|
|
|
|
}
|
|
|
|
|
2013-01-14 08:38:21 -05:00
|
|
|
void NofityTaskExit(Thread* thread)
|
2012-08-04 18:06:17 -04:00
|
|
|
{
|
2013-01-14 08:38:21 -05:00
|
|
|
if ( fputhread == thread )
|
|
|
|
fputhread = NULL;
|
2012-08-04 18:06:17 -04:00
|
|
|
DisableFPU();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Float
|
|
|
|
} // namespace Sortix
|