Subversion Repositories shark

Rev

Blame | Last modification | View Log | RSS feed


/* Project:     OSLib
 * Description: The OS Construction Kit
 * Date:                1.6.2000
 * Idea by:             Luca Abeni & Gerardo Lamastra
 *
 * OSLib is an SO project aimed at developing a common, easy-to-use
 * low-level infrastructure for developing OS kernels and Embedded
 * Applications; it partially derives from the HARTIK project but it
 * currently is independently developed.
 *
 * OSLib is distributed under GPL License, and some of its code has
 * been derived from the Linux kernel source; also some important
 * ideas come from studying the DJGPP go32 extender.
 *
 * We acknowledge the Linux Community, Free Software Foundation,
 * D.J. Delorie and all the other developers who believe in the
 * freedom of software and ideas.
 *
 * For legalese, check out the included GPL license.
 */


/*
   FPU Context switch & management functions!
   Generic 32 bit module
*/


#include <ll/i386/hw-data.h>
#include <ll/i386/hw-instr.h>
#include <ll/i386/mem.h>

#include <ll/i386/tss-ctx.h>

FILE(FPU);

extern TSS TSS_table[];

BYTE LL_FPU_savearea[FPU_CONTEXT_SIZE]; /* Global FPU scratch SaveArea */
#ifdef __FPU_DEBUG__
long int ndp_called = 0, ndp_switched = 0;
#endif

/* FPU context management */
static CONTEXT LL_has_FPU = TSSMain;

/* As the 8086 does not have an hardware mechanism to support task      */
/* switch, also the FPU context switch is implemented via software.     */
/* When a preemption occurs, if the task is marked as a MATH task, the  */
/* preemption routine will save/restore the FPU context.                */
/* The hook is called whenever a FPU context switch is necessarty       */

void ll_FPU_hook(void)
{
    clts();
#ifdef __FPU_DEBUG__
    ndp_called++;
#endif
    if (LL_has_FPU == TSSsel2index(get_TR()))
        return;
#ifdef __FPU_DEBUG__
    ndp_switched++;
#endif

#if 0
    LL_FPU_save();
    memcpy(TSS_table[LL_has_FPU].ctx_FPU, LL_FPU_savearea,
           FPU_CONTEXT_SIZE);
#else
    save_fpu(&(TSS_table[LL_has_FPU]));
#endif

    LL_has_FPU = TSSsel2index(get_TR());

#if 1
    memcpy(LL_FPU_savearea, TSS_table[LL_has_FPU].ctx_FPU,
           FPU_CONTEXT_SIZE);
    LL_FPU_restore();
#else
    restore_fpu(&(TSS_table[LL_has_FPU]));
#endif
    return;
}

CONTEXT LL_FPU_get_task(void)
{
    return (LL_has_FPU);
}