830 lines
20 KiB
ArmAsm
830 lines
20 KiB
ArmAsm
|
/* sections */
|
||
|
|
||
|
|
||
|
#include <minix/config.h>
|
||
|
#include <minix/const.h>
|
||
|
#include <machine/asm.h>
|
||
|
#include <machine/interrupt.h>
|
||
|
#include <machine/vm.h>
|
||
|
#include "archconst.h"
|
||
|
#include "kernel/const.h"
|
||
|
#include "sconst.h"
|
||
|
#include <machine/multiboot.h>
|
||
|
|
||
|
|
||
|
/* Easy way to make functions */
|
||
|
|
||
|
/* Make a function of the form func(arg) */
|
||
|
|
||
|
#define STACKARG 8(%ebp)
|
||
|
|
||
|
#define ARG_EAX_ACTION(FUNCTION, ACTION) ;\
|
||
|
ENTRY(FUNCTION) ;\
|
||
|
push %ebp ;\
|
||
|
mov %esp, %ebp ;\
|
||
|
mov STACKARG, %eax ;\
|
||
|
ACTION ;\
|
||
|
pop %ebp ;\
|
||
|
ret
|
||
|
|
||
|
/* Make a function of the form ret = func() */
|
||
|
#define ARG_EAX_RETURN(FUNCTION, EXPR) ;\
|
||
|
ENTRY(FUNCTION) ;\
|
||
|
push %ebp ;\
|
||
|
mov %esp, %ebp ;\
|
||
|
mov EXPR, %eax ;\
|
||
|
pop %ebp ;\
|
||
|
ret
|
||
|
|
||
|
/* Make a function of the form ret = func() */
|
||
|
#define ARG_EAX_SET(FUNCTION, DEST) ;\
|
||
|
ENTRY(FUNCTION) ;\
|
||
|
push %ebp ;\
|
||
|
mov %esp, %ebp ;\
|
||
|
mov STACKARG, %eax ;\
|
||
|
mov %eax, DEST ;\
|
||
|
jmp 0f /* a jump is required for some sets */ ;\
|
||
|
0: pop %ebp ;\
|
||
|
ret
|
||
|
|
||
|
/* Make a function of the form ret = func() */
|
||
|
#define ARG_AX_SET(FUNCTION, DEST) ;\
|
||
|
ENTRY(FUNCTION) ;\
|
||
|
push %ebp ;\
|
||
|
mov %esp, %ebp ;\
|
||
|
mov STACKARG, %eax ;\
|
||
|
mov %ax, DEST ;\
|
||
|
jmp 0f /* a jump is required for some sets */ ;\
|
||
|
0: pop %ebp ;\
|
||
|
ret
|
||
|
|
||
|
/*
|
||
|
* This file contains a number of assembly code utility routines needed by the
|
||
|
* kernel.
|
||
|
*/
|
||
|
|
||
|
ENTRY(__main)
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_insw */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void phys_insw(Port_t port, phys_bytes buf, size_t count);
|
||
|
* Input an array from an I/O port. Absolute address version of insw().
|
||
|
*/
|
||
|
/* transfer data from (disk controller) port to memory */
|
||
|
ENTRY(phys_insw)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
cld
|
||
|
push %edi
|
||
|
|
||
|
mov 8(%ebp), %edx /* port to read from */
|
||
|
mov 12(%ebp), %edi /* destination addr */
|
||
|
mov 16(%ebp), %ecx /* byte count */
|
||
|
shr $1, %ecx /* word count */
|
||
|
rep insw /* input many words */
|
||
|
pop %edi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_insb */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void phys_insb(Port_t port, phys_bytes buf, size_t count);
|
||
|
* Input an array from an I/O port. Absolute address version of insb().
|
||
|
*/
|
||
|
/* transfer data from (disk controller) port to memory byte by byte */
|
||
|
ENTRY(phys_insb)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
cld
|
||
|
push %edi
|
||
|
|
||
|
mov 8(%ebp), %edx /* port to read from */
|
||
|
mov 12(%ebp), %edi /* destination addr */
|
||
|
mov 16(%ebp), %ecx /* byte count */
|
||
|
rep insb /* input many bytes */
|
||
|
pop %edi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_outsw */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void phys_outsw(Port_t port, phys_bytes buf, size_t count);
|
||
|
* Output an array to an I/O port. Absolute address version of outsw().
|
||
|
*/
|
||
|
/* transfer data from memory to (disk controller) port */
|
||
|
ENTRY(phys_outsw)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
cld
|
||
|
push %esi
|
||
|
|
||
|
mov 8(%ebp), %edx /* port to write to */
|
||
|
mov 12(%ebp), %esi /* source addr */
|
||
|
mov 16(%ebp), %ecx /* byte count */
|
||
|
shr $1, %ecx /* word count */
|
||
|
rep outsw /* output many words */
|
||
|
pop %esi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_outsb */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void phys_outsb(Port_t port, phys_bytes buf, size_t count);
|
||
|
* Output an array to an I/O port. Absolute address version of outsb().
|
||
|
*/
|
||
|
/* transfer data from memory to (disk controller) port byte by byte */
|
||
|
ENTRY(phys_outsb)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
cld
|
||
|
push %esi
|
||
|
|
||
|
mov 8(%ebp), %edx /* port to write to */
|
||
|
mov 12(%ebp), %esi /* source addr */
|
||
|
mov 16(%ebp), %ecx /* byte count */
|
||
|
rep outsb /* output many bytes */
|
||
|
pop %esi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_copy */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC phys_bytes phys_copy(phys_bytes source, phys_bytes destination,
|
||
|
* phys_bytes bytecount);
|
||
|
* Copy a block of data from anywhere to anywhere in physical memory.
|
||
|
*/
|
||
|
/* es edi esi eip src dst len */
|
||
|
ENTRY(phys_copy)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
|
||
|
cld
|
||
|
push %esi
|
||
|
push %edi
|
||
|
|
||
|
mov 8(%ebp), %esi
|
||
|
mov 12(%ebp), %edi
|
||
|
mov 16(%ebp), %eax
|
||
|
|
||
|
cmp $10, %eax /* avoid align overhead for small counts */
|
||
|
jb pc_small
|
||
|
mov %esi, %ecx /* align source, hope target is too */
|
||
|
neg %ecx
|
||
|
and $3, %ecx /* count for alignment */
|
||
|
sub %ecx, %eax
|
||
|
|
||
|
rep movsb (%esi), (%edi)
|
||
|
mov %eax, %ecx
|
||
|
shr $2, %ecx /* count of dwords */
|
||
|
|
||
|
rep movsl (%esi), (%edi)
|
||
|
and $3, %eax
|
||
|
pc_small:
|
||
|
xchg %eax, %ecx /* remainder */
|
||
|
|
||
|
rep movsb (%esi), (%edi)
|
||
|
|
||
|
mov $0, %eax /* 0 means: no fault */
|
||
|
LABEL(phys_copy_fault) /* kernel can send us here */
|
||
|
pop %edi
|
||
|
pop %esi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
LABEL(phys_copy_fault_in_kernel) /* kernel can send us here */
|
||
|
pop %edi
|
||
|
pop %esi
|
||
|
pop %ebp
|
||
|
mov %cr2, %eax
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* copy_msg_from_user */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* int copy_msg_from_user(message * user_mbuf, message * dst);
|
||
|
*
|
||
|
* Copies a message of 64 bytes from user process space to a kernel buffer. This
|
||
|
* function assumes that the process address space is installed (cr3 loaded).
|
||
|
*
|
||
|
* This function from the callers point of view either succeeds or returns an
|
||
|
* error which gives the caller a chance to respond accordingly. In fact it
|
||
|
* either succeeds or if it generates a pagefault, general protection or other
|
||
|
* exception, the trap handler has to redirect the execution to
|
||
|
* __user_copy_msg_pointer_failure where the error is reported to the caller
|
||
|
* without resolving the pagefault. It is not kernel's problem to deal with
|
||
|
* wrong pointers from userspace and the caller should return an error to
|
||
|
* userspace as if wrong values or request were passed to the kernel
|
||
|
*/
|
||
|
ENTRY(copy_msg_from_user)
|
||
|
/* load the source pointer */
|
||
|
mov 4(%esp), %ecx
|
||
|
/* load the destination pointer */
|
||
|
mov 8(%esp), %edx
|
||
|
|
||
|
/* mov 0*4(%ecx), %eax
|
||
|
mov %eax, 0*4(%edx) */
|
||
|
mov 1*4(%ecx), %eax
|
||
|
mov %eax, 1*4(%edx)
|
||
|
mov 2*4(%ecx), %eax
|
||
|
mov %eax, 2*4(%edx)
|
||
|
mov 3*4(%ecx), %eax
|
||
|
mov %eax, 3*4(%edx)
|
||
|
mov 4*4(%ecx), %eax
|
||
|
mov %eax, 4*4(%edx)
|
||
|
mov 5*4(%ecx), %eax
|
||
|
mov %eax, 5*4(%edx)
|
||
|
mov 6*4(%ecx), %eax
|
||
|
mov %eax, 6*4(%edx)
|
||
|
mov 7*4(%ecx), %eax
|
||
|
mov %eax, 7*4(%edx)
|
||
|
mov 8*4(%ecx), %eax
|
||
|
mov %eax, 8*4(%edx)
|
||
|
|
||
|
mov 9*4(%ecx), %eax
|
||
|
mov %eax, 9*4(%edx)
|
||
|
mov 10*4(%ecx), %eax
|
||
|
mov %eax, 10*4(%edx)
|
||
|
mov 11*4(%ecx), %eax
|
||
|
mov %eax, 11*4(%edx)
|
||
|
mov 12*4(%ecx), %eax
|
||
|
mov %eax, 12*4(%edx)
|
||
|
mov 13*4(%ecx), %eax
|
||
|
mov %eax, 13*4(%edx)
|
||
|
mov 14*4(%ecx), %eax
|
||
|
mov %eax, 14*4(%edx)
|
||
|
mov 15*4(%ecx), %eax
|
||
|
mov %eax, 15*4(%edx)
|
||
|
|
||
|
LABEL(__copy_msg_from_user_end)
|
||
|
movl $0, %eax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* copy_msg_to_user */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* void copy_msg_to_user(message * src, message * user_mbuf);
|
||
|
*
|
||
|
* Copies a message of 64 bytes to user process space from a kernel buffer.
|
||
|
*
|
||
|
* All the other copy_msg_from_user() comments apply here as well!
|
||
|
*/
|
||
|
ENTRY(copy_msg_to_user)
|
||
|
/* load the source pointer */
|
||
|
mov 4(%esp), %ecx
|
||
|
/* load the destination pointer */
|
||
|
mov 8(%esp), %edx
|
||
|
|
||
|
mov 0*4(%ecx), %eax
|
||
|
mov %eax, 0*4(%edx)
|
||
|
mov 1*4(%ecx), %eax
|
||
|
mov %eax, 1*4(%edx)
|
||
|
mov 2*4(%ecx), %eax
|
||
|
mov %eax, 2*4(%edx)
|
||
|
mov 3*4(%ecx), %eax
|
||
|
mov %eax, 3*4(%edx)
|
||
|
mov 4*4(%ecx), %eax
|
||
|
mov %eax, 4*4(%edx)
|
||
|
mov 5*4(%ecx), %eax
|
||
|
mov %eax, 5*4(%edx)
|
||
|
mov 6*4(%ecx), %eax
|
||
|
mov %eax, 6*4(%edx)
|
||
|
mov 7*4(%ecx), %eax
|
||
|
mov %eax, 7*4(%edx)
|
||
|
mov 8*4(%ecx), %eax
|
||
|
mov %eax, 8*4(%edx)
|
||
|
|
||
|
|
||
|
mov 9*4(%ecx), %eax
|
||
|
mov %eax, 9*4(%edx)
|
||
|
mov 10*4(%ecx), %eax
|
||
|
mov %eax, 10*4(%edx)
|
||
|
mov 11*4(%ecx), %eax
|
||
|
mov %eax, 11*4(%edx)
|
||
|
mov 12*4(%ecx), %eax
|
||
|
mov %eax, 12*4(%edx)
|
||
|
mov 13*4(%ecx), %eax
|
||
|
mov %eax, 13*4(%edx)
|
||
|
mov 14*4(%ecx), %eax
|
||
|
mov %eax, 14*4(%edx)
|
||
|
mov 15*4(%ecx), %eax
|
||
|
mov %eax, 15*4(%edx)
|
||
|
|
||
|
LABEL(__copy_msg_to_user_end)
|
||
|
movl $0, %eax
|
||
|
ret
|
||
|
|
||
|
/*
|
||
|
* if a function from a selected set of copies from or to userspace fails, it is
|
||
|
* because of a wrong pointer supplied by the userspace. We have to clean up and
|
||
|
* and return -1 to indicated that something wrong has happend. The place it was
|
||
|
* called from has to handle this situation. The exception handler redirect us
|
||
|
* here to continue, clean up and report the error
|
||
|
*/
|
||
|
ENTRY(__user_copy_msg_pointer_failure)
|
||
|
movl $-1, %eax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* phys_memset */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void phys_memset(phys_bytes dst, unsigned long pattern,
|
||
|
* phys_bytes bytecount);
|
||
|
* Fill a block of physical memory with pattern.
|
||
|
*/
|
||
|
ENTRY(phys_memset)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
push %edi
|
||
|
cld
|
||
|
|
||
|
mov 8(%ebp), %edi
|
||
|
mov 16(%ebp), %ecx
|
||
|
mov 12(%ebp), %eax
|
||
|
shr $2, %ecx
|
||
|
rep stosl
|
||
|
|
||
|
/* Any remaining bytes? */
|
||
|
mov 16(%ebp), %ecx
|
||
|
and $3, %ecx
|
||
|
rep stosb
|
||
|
|
||
|
LABEL(memset_fault) /* kernel can send us here */
|
||
|
mov $0, %eax /* 0 means: no fault */
|
||
|
pop %edi
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
LABEL(memset_fault_in_kernel) /* kernel can send us here */
|
||
|
pop %edi
|
||
|
pop %ebp
|
||
|
mov %cr2, %eax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* x86_triplefault */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void x86_triplefault();
|
||
|
* Reset the system by loading IDT with offset 0 and interrupting.
|
||
|
*/
|
||
|
ENTRY(x86_triplefault)
|
||
|
lidt idt_zero
|
||
|
int $3 /* anything goes, the 386 will not like it */
|
||
|
.data
|
||
|
idt_zero:
|
||
|
.long 0, 0
|
||
|
.text
|
||
|
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* halt_cpu */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void halt_cpu(void);
|
||
|
* reanables interrupts and puts the cpu in the halts state. Once an interrupt
|
||
|
* is handled the execution resumes by disabling interrupts and continues
|
||
|
*/
|
||
|
ENTRY(halt_cpu)
|
||
|
sti
|
||
|
hlt /* interrupts enabled only after this instruction is executed! */
|
||
|
/*
|
||
|
* interrupt handlers make sure that the interrupts are disabled when we
|
||
|
* get here so we take only _one_ interrupt after halting the CPU
|
||
|
*/
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* poweroff_vmware_clihlt */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC void poweroff_vmware_clihlt(void);
|
||
|
* VMware detects this peculiar sequence and forces the virtual machine off
|
||
|
* when the parameter gui.exitOnCLIHLT is set to TRUE.
|
||
|
* Otherwise this sequence just hangs the CPU, requiring a power down action.
|
||
|
*/
|
||
|
ENTRY(poweroff_vmware_clihlt)
|
||
|
#ifndef NO_VMWARE_DETECTION
|
||
|
mov $1, %eax
|
||
|
cpuid
|
||
|
test $[1<<31], %ecx /* "virtualized" */
|
||
|
jz 1f /* always 0 on real hardware */
|
||
|
mov $0x40000000, %eax /* select hypervisor-use leaf */
|
||
|
cpuid
|
||
|
cmp $0x61774D56, %ebx /* ASCII "VMwa" */
|
||
|
jne 1f
|
||
|
cmp $0x4D566572, %ecx /* ASCII "reVM" */
|
||
|
jne 1f
|
||
|
cmp $0x65726177, %edx /* ASCII "ware" */
|
||
|
jne 1f
|
||
|
/* we are virtualized by some VMware product! */
|
||
|
#endif
|
||
|
cli
|
||
|
hlt
|
||
|
1: ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* read_flags */
|
||
|
/*===========================================================================*/
|
||
|
/*
|
||
|
* PUBLIC unsigned long read_cpu_flags(void);
|
||
|
* Read CPU status flags from C.
|
||
|
*/
|
||
|
ENTRY(read_cpu_flags)
|
||
|
pushf
|
||
|
mov (%esp), %eax
|
||
|
add $4, %esp
|
||
|
ret
|
||
|
|
||
|
ENTRY(read_ds)
|
||
|
mov $0, %eax
|
||
|
mov %ds, %ax
|
||
|
ret
|
||
|
|
||
|
ENTRY(read_cs)
|
||
|
mov $0, %eax
|
||
|
mov %cs, %ax
|
||
|
ret
|
||
|
|
||
|
ENTRY(read_ss)
|
||
|
mov $0, %eax
|
||
|
mov %ss, %ax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* fpu_routines */
|
||
|
/*===========================================================================*/
|
||
|
|
||
|
/* non-waiting FPU initialization */
|
||
|
ENTRY(fninit)
|
||
|
fninit
|
||
|
ret
|
||
|
|
||
|
ENTRY(clts)
|
||
|
clts
|
||
|
ret
|
||
|
|
||
|
/* store status word (non-waiting) */
|
||
|
ENTRY(fnstsw)
|
||
|
xor %eax, %eax
|
||
|
|
||
|
/* DO NOT CHANGE THE OPERAND!!! gas2ack does not handle it yet */
|
||
|
fnstsw %ax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* fxrstor */
|
||
|
/*===========================================================================*/
|
||
|
ENTRY(fxrstor)
|
||
|
mov 4(%esp), %eax
|
||
|
fxrstor (%eax)
|
||
|
ENTRY(__fxrstor_end)
|
||
|
xor %eax, %eax
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* frstor */
|
||
|
/*===========================================================================*/
|
||
|
ENTRY(frstor)
|
||
|
mov 4(%esp), %eax
|
||
|
frstor (%eax)
|
||
|
ENTRY(__frstor_end)
|
||
|
xor %eax, %eax
|
||
|
ret
|
||
|
|
||
|
/* Shared exception handler for both fxrstor and frstor. */
|
||
|
ENTRY(__frstor_failure)
|
||
|
mov $1, %eax
|
||
|
ret
|
||
|
|
||
|
/* Read/write control registers */
|
||
|
ARG_EAX_RETURN(read_cr0, %cr0);
|
||
|
ARG_EAX_RETURN(read_cr2, %cr2);
|
||
|
ARG_EAX_RETURN(read_cr3, %cr3);
|
||
|
ARG_EAX_RETURN(read_cr4, %cr4);
|
||
|
ARG_EAX_SET(write_cr4, %cr4);
|
||
|
ARG_EAX_SET(write_cr0, %cr0);
|
||
|
ARG_EAX_SET(write_cr3, %cr3);
|
||
|
|
||
|
/* Read/write various descriptor tables */
|
||
|
ARG_EAX_ACTION(x86_ltr, ltr STACKARG );
|
||
|
ARG_EAX_ACTION(x86_lidt, lidtl (%eax));
|
||
|
ARG_EAX_ACTION(x86_lgdt, lgdt (%eax));
|
||
|
ARG_EAX_ACTION(x86_lldt, lldt STACKARG);
|
||
|
ARG_EAX_ACTION(x86_sgdt, sgdt (%eax));
|
||
|
ARG_EAX_ACTION(x86_sidt, sidt (%eax));
|
||
|
|
||
|
/* Load segments */
|
||
|
ARG_AX_SET(x86_load_ds, %ds)
|
||
|
ARG_AX_SET(x86_load_es, %es)
|
||
|
ARG_AX_SET(x86_load_fs, %fs)
|
||
|
ARG_AX_SET(x86_load_gs, %gs)
|
||
|
ARG_AX_SET(x86_load_ss, %ss)
|
||
|
|
||
|
/* FPU */
|
||
|
ARG_EAX_ACTION(fnsave, fnsave (%eax) ; fwait);
|
||
|
ARG_EAX_ACTION(fxsave, fxsave (%eax));
|
||
|
ARG_EAX_ACTION(fnstcw, fnstcw (%eax));
|
||
|
|
||
|
/* invlpg */
|
||
|
ARG_EAX_ACTION(i386_invlpg, invlpg (%eax));
|
||
|
|
||
|
ENTRY(x86_load_kerncs)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
mov 8(%ebp), %eax
|
||
|
jmp $KERN_CS_SELECTOR, $newcs
|
||
|
newcs:
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*
|
||
|
* Read the Model Specific Register (MSR) of IA32 architecture
|
||
|
*
|
||
|
* void ia32_msr_read(u32_t reg, u32_t * hi, u32_t * lo)
|
||
|
*/
|
||
|
ENTRY(ia32_msr_read)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
|
||
|
mov 8(%ebp), %ecx
|
||
|
rdmsr
|
||
|
mov 12(%ebp), %ecx
|
||
|
mov %edx, (%ecx)
|
||
|
mov 16(%ebp), %ecx
|
||
|
mov %eax, (%ecx)
|
||
|
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*
|
||
|
* Write the Model Specific Register (MSR) of IA32 architecture
|
||
|
*
|
||
|
* void ia32_msr_write(u32_t reg, u32_t hi, u32_t lo)
|
||
|
*/
|
||
|
ENTRY(ia32_msr_write)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
|
||
|
mov 12(%ebp), %edx
|
||
|
mov 16(%ebp), %eax
|
||
|
mov 8(%ebp), %ecx
|
||
|
wrmsr
|
||
|
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* __switch_address_space */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC void __switch_address_space(struct proc *p, struct ** ptproc)
|
||
|
*
|
||
|
* sets the %cr3 register to the supplied value if it is not already set to the
|
||
|
* same value in which case it would only result in an extra TLB flush which is
|
||
|
* not desirable
|
||
|
*/
|
||
|
ENTRY(__switch_address_space)
|
||
|
/* read the process pointer */
|
||
|
mov 4(%esp), %edx
|
||
|
/* get the new cr3 value */
|
||
|
movl P_CR3(%edx), %eax
|
||
|
/* test if the new cr3 != NULL */
|
||
|
cmpl $0, %eax
|
||
|
je 0f
|
||
|
|
||
|
/*
|
||
|
* test if the cr3 is loaded with the current value to avoid unnecessary
|
||
|
* TLB flushes
|
||
|
*/
|
||
|
mov %cr3, %ecx
|
||
|
cmp %ecx, %eax
|
||
|
je 0f
|
||
|
mov %eax, %cr3
|
||
|
/* get ptproc */
|
||
|
mov 8(%esp), %eax
|
||
|
mov %edx, (%eax)
|
||
|
0:
|
||
|
ret
|
||
|
|
||
|
/* acknowledge just the master PIC */
|
||
|
ENTRY(eoi_8259_master)
|
||
|
movb $END_OF_INT, %al
|
||
|
outb $INT_CTL
|
||
|
ret
|
||
|
|
||
|
/* we have to acknowledge both PICs */
|
||
|
ENTRY(eoi_8259_slave)
|
||
|
movb $END_OF_INT, %al
|
||
|
outb $INT_CTL
|
||
|
outb $INT2_CTL
|
||
|
ret
|
||
|
|
||
|
/* in some cases we need to force TLB update, reloading cr3 does the trick */
|
||
|
ENTRY(refresh_tlb)
|
||
|
mov %cr3, %eax
|
||
|
mov %eax, %cr3
|
||
|
ret
|
||
|
|
||
|
#ifdef CONFIG_SMP
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* smp_get_htt */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC int smp_get_htt(void); */
|
||
|
/* return true if the processor is hyper-threaded. */
|
||
|
ENTRY(smp_get_htt)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
pushf
|
||
|
pop %eax
|
||
|
mov %eax, %ebx
|
||
|
and $0x200000, %eax
|
||
|
je 0f
|
||
|
mov $0x1, %eax
|
||
|
/* FIXME don't use the byte code */
|
||
|
.byte 0x0f, 0xa2 /* opcode for cpuid */
|
||
|
mov %edx, %eax
|
||
|
pop %ebp
|
||
|
ret
|
||
|
0:
|
||
|
xor %eax, %eax
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* smp_get_num_htt */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC int smp_get_num_htt(void); */
|
||
|
/* Get the number of hyper-threaded processor cores */
|
||
|
ENTRY(smp_get_num_htt)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
pushf
|
||
|
pop %eax
|
||
|
mov %eax, %ebx
|
||
|
and $0x200000, %eax
|
||
|
je 0f
|
||
|
mov $0x1, %eax
|
||
|
/* FIXME don't use the byte code */
|
||
|
.byte 0x0f, 0xa2 /* opcode for cpuid */
|
||
|
mov %ebx, %eax
|
||
|
pop %ebp
|
||
|
ret
|
||
|
0:
|
||
|
xor %eax, %eax
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* smp_get_cores */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC int smp_get_cores(void); */
|
||
|
/* Get the number of cores. */
|
||
|
ENTRY(smp_get_cores)
|
||
|
push %ebp
|
||
|
mov %esp, %ebp
|
||
|
pushf
|
||
|
pop %eax
|
||
|
mov %eax, %ebx
|
||
|
and $0x200000, %eax
|
||
|
je 0f
|
||
|
push %ecx
|
||
|
xor %ecx, %ecx
|
||
|
mov $0x4, %eax
|
||
|
/* FIXME don't use the byte code */
|
||
|
.byte 0x0f, 0xa2 /* opcode for cpuid */
|
||
|
pop %ebp
|
||
|
ret
|
||
|
0:
|
||
|
xor %eax, %eax
|
||
|
pop %ebp
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* arch_spinlock_lock */
|
||
|
/*===========================================================================*/
|
||
|
/* void arch_spinlock_lock (u32_t *lock_data)
|
||
|
* {
|
||
|
* while (test_and_set(lock_data) == 1)
|
||
|
* while (*lock_data == 1)
|
||
|
* ;
|
||
|
* }
|
||
|
* eax register is clobbered.
|
||
|
*/
|
||
|
ENTRY(arch_spinlock_lock)
|
||
|
mov 4(%esp), %eax
|
||
|
mov $1, %edx
|
||
|
2:
|
||
|
mov $1, %ecx
|
||
|
xchg %ecx, (%eax)
|
||
|
test %ecx, %ecx
|
||
|
je 0f
|
||
|
|
||
|
cmp $(1<< 16), %edx
|
||
|
je 1f
|
||
|
shl %edx
|
||
|
1:
|
||
|
mov %edx, %ecx
|
||
|
3:
|
||
|
pause
|
||
|
sub $1, %ecx
|
||
|
test %ecx, %ecx
|
||
|
jz 2b
|
||
|
jmp 3b
|
||
|
0:
|
||
|
mfence
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* arch_spinlock_unlock */
|
||
|
/*===========================================================================*/
|
||
|
/* * void arch_spinlock_unlock (unsigned int *lockp) */
|
||
|
/* spin lock release routine. */
|
||
|
ENTRY(arch_spinlock_unlock)
|
||
|
mov 4(%esp), %eax
|
||
|
mov $0, %ecx
|
||
|
xchg %ecx, (%eax)
|
||
|
mfence
|
||
|
ret
|
||
|
|
||
|
#endif /* CONFIG_SMP */
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* mfence */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC void mfence (void); */
|
||
|
/* architecture specific memory barrier routine. */
|
||
|
ENTRY(mfence)
|
||
|
mfence
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* arch_pause */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC void arch_pause (void); */
|
||
|
/* architecture specific pause routine. */
|
||
|
ENTRY(arch_pause)
|
||
|
pause
|
||
|
ret
|
||
|
|
||
|
/*===========================================================================*/
|
||
|
/* read_ebp */
|
||
|
/*===========================================================================*/
|
||
|
/* PUBLIC u16_t cpuid(void) */
|
||
|
ENTRY(read_ebp)
|
||
|
mov %ebp, %eax
|
||
|
ret
|
||
|
|
||
|
ENTRY(interrupts_enable)
|
||
|
sti
|
||
|
ret
|
||
|
|
||
|
ENTRY(interrupts_disable)
|
||
|
cli
|
||
|
ret
|
||
|
|
||
|
|
||
|
/*
|
||
|
* void switch_k_stack(void * esp, void (* continuation)(void));
|
||
|
*
|
||
|
* sets the current stack pointer to the given value and continues execution at
|
||
|
* the given address
|
||
|
*/
|
||
|
ENTRY(switch_k_stack)
|
||
|
/* get the arguments from the stack */
|
||
|
mov 8(%esp), %eax
|
||
|
mov 4(%esp), %ecx
|
||
|
mov $0, %ebp /* reset %ebp for stack trace */
|
||
|
mov %ecx, %esp /* set the new stack */
|
||
|
jmp *%eax /* and jump to the continuation */
|
||
|
|
||
|
/* NOT_REACHABLE */
|
||
|
0: jmp 0b
|
||
|
|
||
|
.data
|
||
|
idt_ptr:
|
||
|
.short 0x3ff
|
||
|
.long 0x0
|
||
|
|
||
|
ldtsel:
|
||
|
.long LDT_SELECTOR
|