mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-26 12:47:26 +08:00
bafa3a8eb3
currently only read okay, write is not supported, this is enough for me to study ELF, but the floppy is really small, now only about 400kb space free showed by df("/"). Signed-off-by: parai <parai@foxmail.com>
148 lines
4.6 KiB
C
148 lines
4.6 KiB
C
#ifndef __I386_H_
|
|
#define __I386_H_
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
static __inline unsigned char inb(int port)
|
|
{
|
|
unsigned char data;
|
|
__asm __volatile("inb %w1,%0" : "=a" (data) : "d" (port));
|
|
return data;
|
|
}
|
|
static __inline unsigned char inb_p(unsigned short port)
|
|
{
|
|
unsigned char _v;
|
|
__asm__ __volatile__ ("inb %1, %0\n\t"
|
|
// "outb %0,$0x80\n\t"
|
|
// "outb %0,$0x80\n\t"
|
|
// "outb %0,$0x80\n\t"
|
|
"outb %0,$0x80"
|
|
:"=a" (_v)
|
|
:"d" ((unsigned short) port));
|
|
return _v;
|
|
}
|
|
|
|
static __inline unsigned short inw(int port)
|
|
{
|
|
unsigned short data;
|
|
__asm __volatile("inw %w1,%0" : "=a" (data) : "d" (port));
|
|
return data;
|
|
}
|
|
|
|
static __inline unsigned int inl(int port)
|
|
{
|
|
unsigned int data;
|
|
__asm __volatile("inl %w1,%0" : "=a" (data) : "d" (port));
|
|
return data;
|
|
}
|
|
|
|
static __inline void insl(int port, void *addr, int cnt)
|
|
{
|
|
__asm __volatile("cld\n\trepne\n\tinsl" :
|
|
"=D" (addr), "=c" (cnt) :
|
|
"d" (port), "0" (addr), "1" (cnt) :
|
|
"memory", "cc");
|
|
}
|
|
|
|
static __inline void outb(int port, unsigned char data)
|
|
{
|
|
__asm __volatile("outb %0,%w1" : : "a" (data), "d" (port));
|
|
}
|
|
|
|
|
|
static __inline void outb_p(char value, unsigned short port)
|
|
{
|
|
__asm__ __volatile__ ("outb %0,%1\n\t"
|
|
"outb %0,$0x80"
|
|
::"a" ((char) value),"d" ((unsigned short) port));
|
|
}
|
|
|
|
static __inline void outw(int port, unsigned short data)
|
|
{
|
|
__asm __volatile("outw %0,%w1" : : "a" (data), "d" (port));
|
|
}
|
|
|
|
static __inline unsigned char readcmos(int reg)
|
|
{
|
|
outb(0x70,reg);
|
|
return (unsigned char) inb(0x71);
|
|
}
|
|
|
|
#define io_delay() \
|
|
__asm__ __volatile__ ("pushal \n\t"\
|
|
"mov $0x3F6, %dx \n\t" \
|
|
"inb %dx, %al \n\t" \
|
|
"inb %dx, %al \n\t" \
|
|
"inb %dx, %al \n\t" \
|
|
"inb %dx, %al \n\t" \
|
|
"popal")
|
|
|
|
/* Gate descriptors are slightly different*/
|
|
struct Gatedesc {
|
|
unsigned gd_off_15_0 : 16; // low 16 bits of offset in segment
|
|
unsigned gd_ss : 16; // segment selector
|
|
unsigned gd_args : 5; // # args, 0 for interrupt/trap gates
|
|
unsigned gd_rsv1 : 3; // reserved(should be zero I guess)
|
|
unsigned gd_type :4; // type(STS_{TG,IG32,TG32})
|
|
unsigned gd_s : 1; // must be 0 (system)
|
|
unsigned gd_dpl : 2; // descriptor(meaning new) privilege level
|
|
unsigned gd_p : 1; // Present
|
|
unsigned gd_off_31_16 : 16; // high bits of offset in segment
|
|
};
|
|
|
|
/* Pseudo-descriptors used for LGDT, LLDT and LIDT instructions*/
|
|
struct Pseudodesc {
|
|
rt_uint16_t pd__garbage; // LGDT supposed to be from address 4N+2
|
|
rt_uint16_t pd_lim; // Limit
|
|
rt_uint32_t pd_base __attribute__ ((packed)); // Base address
|
|
};
|
|
|
|
#define SETGATE(gate, istrap, sel, off, dpl) \
|
|
{ \
|
|
(gate).gd_off_15_0 = (rt_uint32_t) (off) & 0xffff; \
|
|
(gate).gd_ss = (sel); \
|
|
(gate).gd_args = 0; \
|
|
(gate).gd_rsv1 = 0; \
|
|
(gate).gd_type = (istrap) ? STS_TG32 : STS_IG32; \
|
|
(gate).gd_s = 0; \
|
|
(gate).gd_dpl = dpl; \
|
|
(gate).gd_p = 1; \
|
|
(gate).gd_off_31_16 = (rt_uint32_t) (off) >> 16; \
|
|
}
|
|
|
|
/* Global descriptor numbers*/
|
|
#define GD_KT 0x08 // kernel text
|
|
#define GD_KD 0x10 // kernel data
|
|
#define GD_UT 0x18 // user text
|
|
#define GD_UD 0x20 // user data
|
|
|
|
/* Application segment type bits*/
|
|
#define STA_X 0x8 // Executable segment
|
|
#define STA_E 0x4 // Expand down(non-executable segments)
|
|
#define STA_C 0x4 // Conforming code segment(executable only)
|
|
#define STA_W 0x2 // Writeable(non-executable segments)
|
|
#define STA_R 0x2 // Readable(executable segments)
|
|
#define STA_A 0x1 // Accessed
|
|
|
|
/* System segment type bits*/
|
|
#define STS_T16A 0x1 // Available 16-bit TSS
|
|
#define STS_LDT 0x2 // Local Descriptor Table
|
|
#define STS_T16B 0x3 // Busy 16-bit TSS
|
|
#define STS_CG16 0x4 // 16-bit Call Gate
|
|
#define STS_TG 0x5 // Task Gate / Coum Transmitions
|
|
#define STS_IG16 0x6 // 16-bit Interrupt Gate
|
|
#define STS_TG16 0x7 // 16-bit Trap Gate
|
|
#define STS_T32A 0x9 // Available 32-bit TSS
|
|
#define STS_T32B 0xb // Busy 32-bit TSS
|
|
#define STS_CG32 0xc // 32-bit Call Gate
|
|
#define STS_IG32 0xe // 32-bit Interrupt Gate
|
|
#define STS_TG32 0xf // 32-bit Trap Gate
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif
|