Subversion Repositories shark

Rev

Rev 648 | Rev 1618 | Go to most recent revision | Blame | Compare with Previous | 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.
 */


/*      Xlib initialization code        */

#include <ll/i386/mem.h>
#include <ll/i386/cons.h>
#include <ll/i386/mb-info.h>
#include <ll/i386/error.h>
#include <ll/i386/pit.h>
#include <ll/i386/pic.h>

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

FILE(X-Init);

extern DWORD ll_irq_table[256];

#ifdef __VIRCSW__
int activeInt = 0;
#endif

/* Assembly external routines!      */
/* Setup the TR register of the 80386, to initialize context switch */

extern void init_TR(WORD v);
TSS main_tss;

/* Architecture definition */
LL_ARCH ll_arch;

/* The following stuff is in llCxA.Asm/S    */

static void dummyfun(int i)
{
#if 0
  if (i < 32) {
    cputs("Unhandled Exc occured!!!\n");
  } else {
    cputs("Unhandled Int occured!!!\n");
  }
#else
  message("Unhandled Exc or Int %d occured!!!\n", i);
#endif
  halt();
}

void l1_int_bind(int i, void *f)
{
  ll_irq_table[i] = (DWORD)f;
}

void *l1_init(void)
{
  register int i;
  struct ll_cpuInfo cpuInfo;
  extern unsigned char X86_apic;
  extern unsigned char X86_tsc;
  extern BYTE X86_fpu;
  LIN_ADDR b;
 
  for(i = 0; i < 256; i++) {
    ll_irq_table[i] = (DWORD)dummyfun;
  }
 
  X86_get_CPU(&cpuInfo);
  X86_get_FPU();
  ll_arch.x86.arch = __LL_ARCH__;
  ll_arch.x86.cpu = cpuInfo.X86_cpu;
  ll_arch.x86.fpu = X86_fpu;
  memcpy(&(ll_arch.x86.vendor), &(cpuInfo.X86_vendor_1), 12);
 
  X86_apic = (cpuInfo.X86_StandardFeature>>9) & 1;
  X86_tsc  = (cpuInfo.X86_StandardFeature>>4) & 1;
 
  /* TODO! Need to map featuresXXX & Signature onto ll_arch!  */
  /* TODO! Need to check for CPU bugs!!           */
 
#ifdef __LL_DEBUG__
  message("LL Architecture: %s\n", __LL_ARCH__);
  message("CPU : %u\nFPU : %u\n", cpuInfo.X86_cpu, X86_fpu);
  message("Signature : 0x%lx\nVendor: %s\n", cpuInfo.X86_signature,
          ll_arch.x86.vendor);
  message("Features #1: 0x%lx\n", cpuInfo.X86_IntelFeature_1);
  message("Features #2: 0x%lx\n", cpuInfo.X86_IntelFeature_2);
  message("Features #3: 0x%lx\n", cpuInfo.X86_StandardFeature);
  message("Has APIC: %s\n", X86_apic);
  message("Has TSC: %s\n", X86_tsc);
#endif /* __LL_DEBUG__ */
 
  IDT_init();

  /* Init coprocessor & assign it to main() */
  /* OK... Now I know the sense of all this... :
     We need a initial value for the FPU context (to be used for creating
     new FPU contexts, as init value)...
     ... And we get it in this strange way!!!!
  */

  reset_fpu();
  init_fpu();
 
  /* Init PIC controllers & unmask timer */
  PIC_init();
 
  /* Set the TR initial value */
  b = (LIN_ADDR)(&main_tss);
  GDT_place(MAIN_SEL, (DWORD)b, sizeof(TSS), FREE_TSS386, GRAN_16);
  init_TR(MAIN_SEL);
 
  return mbi_address();
}


void l1_end(void)
{
  outp(0x21,0xFF);
  outp(0xA1,0xFF);
  /* Back to DOS settings */
  PIC_end();
  /* Reset the timer chip according DOS specification */
  /* Mode: Binary/Mode 3/16 bit Time_const/Counter 0 */
#if 0
  outp(0x43,0x36);
  /* Time_const = 65536; write 0 in CTR */
  outp(0x40,0);
  outp(0x40,0);
#endif
  pit_init(0, TMR_MD3, 0);    /* Timer 0, Mode 3, Time constant 0 */
  if(ll_arch.x86.cpu > 4) {    
    pit_init(1, TMR_MD2, 18);
  } else {
    pit_init(2, TMR_MD0, 0);
    outp(0x61, 0);              /* Stop channel 2 */
  }
}