desc.h 9.9 KB
Newer Older
H
H. Peter Anvin 已提交
1 2
#ifndef _ASM_X86_DESC_H
#define _ASM_X86_DESC_H
3 4 5

#include <asm/desc_defs.h>
#include <asm/ldt.h>
6
#include <asm/mmu.h>
I
Ingo Molnar 已提交
7

8
#include <linux/smp.h>
9

I
Ingo Molnar 已提交
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
static inline void fill_ldt(struct desc_struct *desc, const struct user_desc *info)
{
	desc->limit0		= info->limit & 0x0ffff;

	desc->base0		= (info->base_addr & 0x0000ffff);
	desc->base1		= (info->base_addr & 0x00ff0000) >> 16;

	desc->type		= (info->read_exec_only ^ 1) << 1;
	desc->type	       |= info->contents << 2;

	desc->s			= 1;
	desc->dpl		= 0x3;
	desc->p			= info->seg_not_present ^ 1;
	desc->limit		= (info->limit & 0xf0000) >> 16;
	desc->avl		= info->useable;
	desc->d			= info->seg_32bit;
	desc->g			= info->limit_in_pages;

	desc->base2		= (info->base_addr & 0xff000000) >> 24;
29
	/*
30 31
	 * Don't allow setting of the lm bit. It would confuse
	 * user_64bit_mode and would get overridden by sysret anyway.
32
	 */
I
Ingo Molnar 已提交
33
	desc->l			= 0;
34 35
}

36 37
extern struct desc_ptr idt_descr;
extern gate_desc idt_table[];
38

39 40 41
struct gdt_page {
	struct desc_struct gdt[GDT_ENTRIES];
} __attribute__((aligned(PAGE_SIZE)));
I
Ingo Molnar 已提交
42

43
DECLARE_PER_CPU_PAGE_ALIGNED(struct gdt_page, gdt_page);
44 45 46 47 48 49

static inline struct desc_struct *get_cpu_gdt_table(unsigned int cpu)
{
	return per_cpu(gdt_page, cpu).gdt;
}

50
#ifdef CONFIG_X86_64
51 52 53 54

static inline void pack_gate(gate_desc *gate, unsigned type, unsigned long func,
			     unsigned dpl, unsigned ist, unsigned seg)
{
I
Ingo Molnar 已提交
55 56 57 58 59 60 61 62 63 64
	gate->offset_low	= PTR_LOW(func);
	gate->segment		= __KERNEL_CS;
	gate->ist		= ist;
	gate->p			= 1;
	gate->dpl		= dpl;
	gate->zero0		= 0;
	gate->zero1		= 0;
	gate->type		= type;
	gate->offset_middle	= PTR_MIDDLE(func);
	gate->offset_high	= PTR_HIGH(func);
65 66
}

67
#else
68
static inline void pack_gate(gate_desc *gate, unsigned char type,
69 70
			     unsigned long base, unsigned dpl, unsigned flags,
			     unsigned short seg)
71 72
{
	gate->a = (seg << 16) | (base & 0xffff);
I
Ingo Molnar 已提交
73
	gate->b = (base & 0xffff0000) | (((0x80 | type | (dpl << 5)) & 0xff) << 8);
74 75
}

76 77
#endif

78 79 80
static inline int desc_empty(const void *ptr)
{
	const u32 *desc = ptr;
I
Ingo Molnar 已提交
81

82 83 84
	return !(desc[0] | desc[1]);
}

85 86 87
#ifdef CONFIG_PARAVIRT
#include <asm/paravirt.h>
#else
I
Ingo Molnar 已提交
88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
#define load_TR_desc()				native_load_tr_desc()
#define load_gdt(dtr)				native_load_gdt(dtr)
#define load_idt(dtr)				native_load_idt(dtr)
#define load_tr(tr)				asm volatile("ltr %0"::"m" (tr))
#define load_ldt(ldt)				asm volatile("lldt %0"::"m" (ldt))

#define store_gdt(dtr)				native_store_gdt(dtr)
#define store_idt(dtr)				native_store_idt(dtr)
#define store_tr(tr)				(tr = native_store_tr())

#define load_TLS(t, cpu)			native_load_tls(t, cpu)
#define set_ldt					native_set_ldt

#define write_ldt_entry(dt, entry, desc)	native_write_ldt_entry(dt, entry, desc)
#define write_gdt_entry(dt, entry, desc, type)	native_write_gdt_entry(dt, entry, desc, type)
#define write_idt_entry(dt, entry, g)		native_write_idt_entry(dt, entry, g)
104 105 106 107 108 109 110 111 112

static inline void paravirt_alloc_ldt(struct desc_struct *ldt, unsigned entries)
{
}

static inline void paravirt_free_ldt(struct desc_struct *ldt, unsigned entries)
{
}
#endif	/* CONFIG_PARAVIRT */
113

114 115
#define store_ldt(ldt) asm("sldt %0" : "=m"(ldt))

I
Ingo Molnar 已提交
116
static inline void native_write_idt_entry(gate_desc *idt, int entry, const gate_desc *gate)
117 118 119 120
{
	memcpy(&idt[entry], gate, sizeof(*gate));
}

I
Ingo Molnar 已提交
121
static inline void native_write_ldt_entry(struct desc_struct *ldt, int entry, const void *desc)
122 123 124 125
{
	memcpy(&ldt[entry], desc, 8);
}

I
Ingo Molnar 已提交
126 127
static inline void
native_write_gdt_entry(struct desc_struct *gdt, int entry, const void *desc, int type)
128 129
{
	unsigned int size;
I
Ingo Molnar 已提交
130

131
	switch (type) {
I
Ingo Molnar 已提交
132 133 134
	case DESC_TSS:	size = sizeof(tss_desc);	break;
	case DESC_LDT:	size = sizeof(ldt_desc);	break;
	default:	size = sizeof(*gdt);		break;
135
	}
I
Ingo Molnar 已提交
136

137 138 139 140 141 142 143 144 145
	memcpy(&gdt[entry], desc, size);
}

static inline void pack_descriptor(struct desc_struct *desc, unsigned long base,
				   unsigned long limit, unsigned char type,
				   unsigned char flags)
{
	desc->a = ((base & 0xffff) << 16) | (limit & 0xffff);
	desc->b = (base & 0xff000000) | ((base & 0xff0000) >> 16) |
146 147
		(limit & 0x000f0000) | ((type & 0xff) << 8) |
		((flags & 0xf) << 20);
148 149 150 151
	desc->p = 1;
}


I
Ingo Molnar 已提交
152
static inline void set_tssldt_descriptor(void *d, unsigned long addr, unsigned type, unsigned size)
153 154
{
#ifdef CONFIG_X86_64
155
	struct ldttss_desc64 *desc = d;
I
Ingo Molnar 已提交
156

157
	memset(desc, 0, sizeof(*desc));
I
Ingo Molnar 已提交
158 159 160 161 162 163 164 165 166

	desc->limit0		= size & 0xFFFF;
	desc->base0		= PTR_LOW(addr);
	desc->base1		= PTR_MIDDLE(addr) & 0xFF;
	desc->type		= type;
	desc->p			= 1;
	desc->limit1		= (size >> 16) & 0xF;
	desc->base2		= (PTR_MIDDLE(addr) >> 8) & 0xFF;
	desc->base3		= PTR_HIGH(addr);
167
#else
168
	pack_descriptor((struct desc_struct *)d, addr, size, 0x80 | type, 0);
169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
#endif
}

static inline void __set_tss_desc(unsigned cpu, unsigned int entry, void *addr)
{
	struct desc_struct *d = get_cpu_gdt_table(cpu);
	tss_desc tss;

	/*
	 * sizeof(unsigned long) coming from an extra "long" at the end
	 * of the iobitmap. See tss_struct definition in processor.h
	 *
	 * -1? seg base+limit should be pointing to the address of the
	 * last valid byte
	 */
184
	set_tssldt_descriptor(&tss, (unsigned long)addr, DESC_TSS,
185 186
			      IO_BITMAP_OFFSET + IO_BITMAP_BYTES +
			      sizeof(unsigned long) - 1);
187 188 189 190 191
	write_gdt_entry(d, entry, &tss, DESC_TSS);
}

#define set_tss_desc(cpu, addr) __set_tss_desc(cpu, GDT_ENTRY_TSS, addr)

192 193 194
static inline void native_set_ldt(const void *addr, unsigned int entries)
{
	if (likely(entries == 0))
195
		asm volatile("lldt %w0"::"q" (0));
196 197 198 199
	else {
		unsigned cpu = smp_processor_id();
		ldt_desc ldt;

M
Michael Karcher 已提交
200 201
		set_tssldt_descriptor(&ldt, (unsigned long)addr, DESC_LDT,
				      entries * LDT_ENTRY_SIZE - 1);
202 203
		write_gdt_entry(get_cpu_gdt_table(cpu), GDT_ENTRY_LDT,
				&ldt, DESC_LDT);
204
		asm volatile("lldt %w0"::"q" (GDT_ENTRY_LDT*8));
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235
	}
}

static inline void native_load_tr_desc(void)
{
	asm volatile("ltr %w0"::"q" (GDT_ENTRY_TSS*8));
}

static inline void native_load_gdt(const struct desc_ptr *dtr)
{
	asm volatile("lgdt %0"::"m" (*dtr));
}

static inline void native_load_idt(const struct desc_ptr *dtr)
{
	asm volatile("lidt %0"::"m" (*dtr));
}

static inline void native_store_gdt(struct desc_ptr *dtr)
{
	asm volatile("sgdt %0":"=m" (*dtr));
}

static inline void native_store_idt(struct desc_ptr *dtr)
{
	asm volatile("sidt %0":"=m" (*dtr));
}

static inline unsigned long native_store_tr(void)
{
	unsigned long tr;
I
Ingo Molnar 已提交
236

237
	asm volatile("str %0":"=r" (tr));
I
Ingo Molnar 已提交
238

239 240 241 242 243 244
	return tr;
}

static inline void native_load_tls(struct thread_struct *t, unsigned int cpu)
{
	struct desc_struct *gdt = get_cpu_gdt_table(cpu);
I
Ingo Molnar 已提交
245
	unsigned int i;
246 247 248 249 250

	for (i = 0; i < GDT_ENTRY_TLS_ENTRIES; i++)
		gdt[GDT_ENTRY_TLS_MIN + i] = t->tls_array[i];
}

251 252 253 254 255 256 257 258 259
#define _LDT_empty(info)				\
	((info)->base_addr		== 0	&&	\
	 (info)->limit			== 0	&&	\
	 (info)->contents		== 0	&&	\
	 (info)->read_exec_only		== 1	&&	\
	 (info)->seg_32bit		== 0	&&	\
	 (info)->limit_in_pages		== 0	&&	\
	 (info)->seg_not_present	== 1	&&	\
	 (info)->useable		== 0)
260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286

#ifdef CONFIG_X86_64
#define LDT_empty(info) (_LDT_empty(info) && ((info)->lm == 0))
#else
#define LDT_empty(info) (_LDT_empty(info))
#endif

static inline void clear_LDT(void)
{
	set_ldt(NULL, 0);
}

/*
 * load one particular LDT into the current CPU
 */
static inline void load_LDT_nolock(mm_context_t *pc)
{
	set_ldt(pc->ldt, pc->size);
}

static inline void load_LDT(mm_context_t *pc)
{
	preempt_disable();
	load_LDT_nolock(pc);
	preempt_enable();
}

287
static inline unsigned long get_desc_base(const struct desc_struct *desc)
288
{
289
	return (unsigned)(desc->base0 | ((desc->base1) << 16) | ((desc->base2) << 24));
290
}
291

292 293 294 295 296 297 298
static inline void set_desc_base(struct desc_struct *desc, unsigned long base)
{
	desc->base0 = base & 0xffff;
	desc->base1 = (base >> 16) & 0xff;
	desc->base2 = (base >> 24) & 0xff;
}

299 300 301 302 303
static inline unsigned long get_desc_limit(const struct desc_struct *desc)
{
	return desc->limit0 | (desc->limit << 16);
}

304 305 306 307 308 309
static inline void set_desc_limit(struct desc_struct *desc, unsigned long limit)
{
	desc->limit0 = limit & 0xffff;
	desc->limit = (limit >> 16) & 0xf;
}

310
static inline void _set_gate(int gate, unsigned type, void *addr,
311
			     unsigned dpl, unsigned ist, unsigned seg)
312 313
{
	gate_desc s;
I
Ingo Molnar 已提交
314

315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
	pack_gate(&s, type, (unsigned long)addr, dpl, ist, seg);
	/*
	 * does not need to be atomic because it is only done once at
	 * setup time
	 */
	write_idt_entry(idt_table, gate, &s);
}

/*
 * This needs to use 'idt_table' rather than 'idt', and
 * thus use the _nonmapped_ version of the IDT, as the
 * Pentium F0 0F bugfix can have resulted in the mapped
 * IDT being write-protected.
 */
static inline void set_intr_gate(unsigned int n, void *addr)
{
	BUG_ON((unsigned)n > 0xFF);
	_set_gate(n, GATE_INTERRUPT, addr, 0, 0, __KERNEL_CS);
}

335
extern int first_system_vector;
336 337
/* used_vectors is BITMAP for irq is not managed by percpu vector_irq */
extern unsigned long used_vectors[];
338 339 340

static inline void alloc_system_vector(int vector)
{
341 342
	if (!test_bit(vector, used_vectors)) {
		set_bit(vector, used_vectors);
343 344
		if (first_system_vector > vector)
			first_system_vector = vector;
I
Ingo Molnar 已提交
345
	} else {
346
		BUG();
I
Ingo Molnar 已提交
347
	}
348 349 350 351 352 353 354 355
}

static inline void alloc_intr_gate(unsigned int n, void *addr)
{
	alloc_system_vector(n);
	set_intr_gate(n, addr);
}

356 357 358 359 360 361 362 363 364
/*
 * This routine sets up an interrupt gate at directory privilege level 3.
 */
static inline void set_system_intr_gate(unsigned int n, void *addr)
{
	BUG_ON((unsigned)n > 0xFF);
	_set_gate(n, GATE_INTERRUPT, addr, 0x3, 0, __KERNEL_CS);
}

365
static inline void set_system_trap_gate(unsigned int n, void *addr)
366 367
{
	BUG_ON((unsigned)n > 0xFF);
368
	_set_gate(n, GATE_TRAP, addr, 0x3, 0, __KERNEL_CS);
369 370
}

371
static inline void set_trap_gate(unsigned int n, void *addr)
372 373
{
	BUG_ON((unsigned)n > 0xFF);
374
	_set_gate(n, GATE_TRAP, addr, 0, 0, __KERNEL_CS);
375 376 377 378 379 380 381 382 383 384 385 386 387 388
}

static inline void set_task_gate(unsigned int n, unsigned int gdt_entry)
{
	BUG_ON((unsigned)n > 0xFF);
	_set_gate(n, GATE_TASK, (void *)0, 0, 0, (gdt_entry<<3));
}

static inline void set_intr_gate_ist(int n, void *addr, unsigned ist)
{
	BUG_ON((unsigned)n > 0xFF);
	_set_gate(n, GATE_INTERRUPT, addr, 0, ist, __KERNEL_CS);
}

389
static inline void set_system_intr_gate_ist(int n, void *addr, unsigned ist)
390 391 392 393
{
	BUG_ON((unsigned)n > 0xFF);
	_set_gate(n, GATE_INTERRUPT, addr, 0x3, ist, __KERNEL_CS);
}
394

H
H. Peter Anvin 已提交
395
#endif /* _ASM_X86_DESC_H */