process.c 12.6 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12
/*
 *  linux/arch/arm/kernel/process.c
 *
 *  Copyright (C) 1996-2000 Russell King - Converted to ARM.
 *  Original Copyright (C) 1995  Linus Torvalds
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */
#include <stdarg.h>

13
#include <linux/export.h>
L
Linus Torvalds 已提交
14 15 16 17 18 19 20 21 22 23 24
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/stddef.h>
#include <linux/unistd.h>
#include <linux/user.h>
#include <linux/delay.h>
#include <linux/reboot.h>
#include <linux/interrupt.h>
#include <linux/kallsyms.h>
#include <linux/init.h>
25
#include <linux/cpu.h>
26
#include <linux/elfcore.h>
27
#include <linux/pm.h>
28
#include <linux/tick.h>
29
#include <linux/utsname.h>
30
#include <linux/uaccess.h>
31
#include <linux/random.h>
32
#include <linux/hw_breakpoint.h>
L
Len Brown 已提交
33
#include <linux/cpuidle.h>
L
Linus Torvalds 已提交
34

35
#include <asm/cacheflush.h>
L
Linus Torvalds 已提交
36 37
#include <asm/leds.h>
#include <asm/processor.h>
38
#include <asm/thread_notify.h>
39
#include <asm/stacktrace.h>
40
#include <asm/mach/time.h>
L
Linus Torvalds 已提交
41

42 43 44 45 46 47
#ifdef CONFIG_CC_STACKPROTECTOR
#include <linux/stackprotector.h>
unsigned long __stack_chk_guard __read_mostly;
EXPORT_SYMBOL(__stack_chk_guard);
#endif

48 49 50 51 52 53 54
static const char *processor_modes[] = {
  "USER_26", "FIQ_26" , "IRQ_26" , "SVC_26" , "UK4_26" , "UK5_26" , "UK6_26" , "UK7_26" ,
  "UK8_26" , "UK9_26" , "UK10_26", "UK11_26", "UK12_26", "UK13_26", "UK14_26", "UK15_26",
  "USER_32", "FIQ_32" , "IRQ_32" , "SVC_32" , "UK4_32" , "UK5_32" , "UK6_32" , "ABT_32" ,
  "UK8_32" , "UK9_32" , "UK10_32", "UND_32" , "UK12_32", "UK13_32", "UK14_32", "SYS_32"
};

55 56 57 58
static const char *isa_modes[] = {
  "ARM" , "Thumb" , "Jazelle", "ThumbEE"
};

59
extern void setup_mm_for_reboot(void);
L
Linus Torvalds 已提交
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91

static volatile int hlt_counter;

void disable_hlt(void)
{
	hlt_counter++;
}

EXPORT_SYMBOL(disable_hlt);

void enable_hlt(void)
{
	hlt_counter--;
}

EXPORT_SYMBOL(enable_hlt);

static int __init nohlt_setup(char *__unused)
{
	hlt_counter = 1;
	return 1;
}

static int __init hlt_setup(char *__unused)
{
	hlt_counter = 0;
	return 1;
}

__setup("nohlt", nohlt_setup);
__setup("hlt", hlt_setup);

92 93 94 95 96 97 98 99 100 101 102 103 104
extern void call_with_stack(void (*fn)(void *), void *arg, void *sp);
typedef void (*phys_reset_t)(unsigned long);

/*
 * A temporary stack to use for CPU reset. This is static so that we
 * don't clobber it with the identity mapping. When running with this
 * stack, any references to the current task *will not work* so you
 * should really do as little as possible before jumping to your reset
 * code.
 */
static u64 soft_restart_stack[16];

static void __soft_restart(void *addr)
105
{
106
	phys_reset_t phys_reset;
107

108
	/* Take out a flat memory mapping. */
109
	setup_mm_for_reboot();
110

111 112 113 114 115 116 117 118 119
	/* Clean and invalidate caches */
	flush_cache_all();

	/* Turn off caching */
	cpu_proc_fin();

	/* Push out any further dirty data, and ensure cache is empty */
	flush_cache_all();

120 121 122
	/* Switch to the identity mapping. */
	phys_reset = (phys_reset_t)(unsigned long)virt_to_phys(cpu_reset);
	phys_reset((unsigned long)addr);
123

124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144
	/* Should never get here. */
	BUG();
}

void soft_restart(unsigned long addr)
{
	u64 *stack = soft_restart_stack + ARRAY_SIZE(soft_restart_stack);

	/* Disable interrupts first */
	local_irq_disable();
	local_fiq_disable();

	/* Disable the L2 if we're the last man standing. */
	if (num_online_cpus() == 1)
		outer_disable();

	/* Change to the new stack and continue with the reset. */
	call_with_stack(__soft_restart, (void *)addr, (void *)stack);

	/* Should never get here. */
	BUG();
145 146
}

147
static void null_restart(char mode, const char *cmd)
148
{
149 150
}

L
Linus Torvalds 已提交
151
/*
152
 * Function pointers to optional machine specific functions
L
Linus Torvalds 已提交
153 154 155 156
 */
void (*pm_power_off)(void);
EXPORT_SYMBOL(pm_power_off);

157
void (*arm_pm_restart)(char str, const char *cmd) = null_restart;
158 159
EXPORT_SYMBOL_GPL(arm_pm_restart);

L
Linus Torvalds 已提交
160
/*
N
Nicolas Pitre 已提交
161
 * This is our default idle handler.
L
Linus Torvalds 已提交
162
 */
N
Nicolas Pitre 已提交
163 164 165

void (*arm_pm_idle)(void);

166
static void default_idle(void)
L
Linus Torvalds 已提交
167
{
N
Nicolas Pitre 已提交
168 169 170
	if (arm_pm_idle)
		arm_pm_idle();
	else
N
Nicolas Pitre 已提交
171
		cpu_do_idle();
172
	local_irq_enable();
L
Linus Torvalds 已提交
173 174
}

175 176 177
void (*pm_idle)(void) = default_idle;
EXPORT_SYMBOL(pm_idle);

L
Linus Torvalds 已提交
178
/*
179 180 181 182
 * The idle thread, has rather strange semantics for calling pm_idle,
 * but this is what x86 does and we need to do the same, so that
 * things like cpuidle get called in the same way.  The only difference
 * is that we always respect 'hlt_counter' to prevent low power idle.
L
Linus Torvalds 已提交
183 184 185 186 187 188 189
 */
void cpu_idle(void)
{
	local_fiq_enable();

	/* endless idle loop with no priority at all */
	while (1) {
190 191
		tick_nohz_idle_enter();
		rcu_idle_enter();
192 193
		leds_event(led_idle_start);
		while (!need_resched()) {
194
#ifdef CONFIG_HOTPLUG_CPU
195 196
			if (cpu_is_offline(smp_processor_id()))
				cpu_die();
197 198
#endif

N
Nicolas Pitre 已提交
199 200 201 202
			/*
			 * We need to disable interrupts here
			 * to ensure we don't miss a wakeup call.
			 */
203
			local_irq_disable();
204 205 206
#ifdef CONFIG_PL310_ERRATA_769419
			wmb();
#endif
207 208 209
			if (hlt_counter) {
				local_irq_enable();
				cpu_relax();
N
Nicolas Pitre 已提交
210
			} else if (!need_resched()) {
211
				stop_critical_timings();
212
				if (cpuidle_idle_call())
L
Len Brown 已提交
213
					pm_idle();
214 215
				start_critical_timings();
				/*
N
Nicolas Pitre 已提交
216 217
				 * pm_idle functions must always
				 * return with IRQs enabled.
218 219
				 */
				WARN_ON(irqs_disabled());
N
Nicolas Pitre 已提交
220
			} else
221 222
				local_irq_enable();
		}
L
Linus Torvalds 已提交
223
		leds_event(led_idle_end);
224 225
		rcu_idle_exit();
		tick_nohz_idle_exit();
226
		schedule_preempt_disabled();
L
Linus Torvalds 已提交
227 228 229 230 231 232 233 234 235 236 237 238 239
	}
}

static char reboot_mode = 'h';

int __init reboot_setup(char *str)
{
	reboot_mode = str[0];
	return 1;
}

__setup("reboot=", reboot_setup);

240
void machine_shutdown(void)
L
Linus Torvalds 已提交
241
{
242 243 244
#ifdef CONFIG_SMP
	smp_send_stop();
#endif
L
Linus Torvalds 已提交
245 246
}

247 248 249 250 251
void machine_halt(void)
{
	machine_shutdown();
	while (1);
}
L
Linus Torvalds 已提交
252 253 254

void machine_power_off(void)
{
255
	machine_shutdown();
L
Linus Torvalds 已提交
256 257 258 259
	if (pm_power_off)
		pm_power_off();
}

260
void machine_restart(char *cmd)
L
Linus Torvalds 已提交
261
{
262
	machine_shutdown();
263

264
	arm_pm_restart(reboot_mode, cmd);
265 266 267 268 269 270 271

	/* Give a grace period for failure to restart of 1s */
	mdelay(1000);

	/* Whoops - the platform was unable to reboot. Tell the user! */
	printk("Reboot failed -- System halted\n");
	while (1);
L
Linus Torvalds 已提交
272 273
}

R
Russell King 已提交
274
void __show_regs(struct pt_regs *regs)
L
Linus Torvalds 已提交
275
{
276 277
	unsigned long flags;
	char buf[64];
L
Linus Torvalds 已提交
278

279
	printk("CPU: %d    %s  (%s %.*s)\n",
280 281
		raw_smp_processor_id(), print_tainted(),
		init_utsname()->release,
282 283
		(int)strcspn(init_utsname()->version, " "),
		init_utsname()->version);
L
Linus Torvalds 已提交
284 285
	print_symbol("PC is at %s\n", instruction_pointer(regs));
	print_symbol("LR is at %s\n", regs->ARM_lr);
286
	printk("pc : [<%08lx>]    lr : [<%08lx>]    psr: %08lx\n"
L
Linus Torvalds 已提交
287
	       "sp : %08lx  ip : %08lx  fp : %08lx\n",
288 289
		regs->ARM_pc, regs->ARM_lr, regs->ARM_cpsr,
		regs->ARM_sp, regs->ARM_ip, regs->ARM_fp);
L
Linus Torvalds 已提交
290 291 292 293 294 295 296 297 298
	printk("r10: %08lx  r9 : %08lx  r8 : %08lx\n",
		regs->ARM_r10, regs->ARM_r9,
		regs->ARM_r8);
	printk("r7 : %08lx  r6 : %08lx  r5 : %08lx  r4 : %08lx\n",
		regs->ARM_r7, regs->ARM_r6,
		regs->ARM_r5, regs->ARM_r4);
	printk("r3 : %08lx  r2 : %08lx  r1 : %08lx  r0 : %08lx\n",
		regs->ARM_r3, regs->ARM_r2,
		regs->ARM_r1, regs->ARM_r0);
299 300 301 302 303 304 305 306

	flags = regs->ARM_cpsr;
	buf[0] = flags & PSR_N_BIT ? 'N' : 'n';
	buf[1] = flags & PSR_Z_BIT ? 'Z' : 'z';
	buf[2] = flags & PSR_C_BIT ? 'C' : 'c';
	buf[3] = flags & PSR_V_BIT ? 'V' : 'v';
	buf[4] = '\0';

307
	printk("Flags: %s  IRQs o%s  FIQs o%s  Mode %s  ISA %s  Segment %s\n",
308
		buf, interrupts_enabled(regs) ? "n" : "ff",
L
Linus Torvalds 已提交
309 310
		fast_interrupts_enabled(regs) ? "n" : "ff",
		processor_modes[processor_mode(regs)],
311
		isa_modes[isa_mode(regs)],
L
Linus Torvalds 已提交
312
		get_fs() == get_ds() ? "kernel" : "user");
313
#ifdef CONFIG_CPU_CP15
L
Linus Torvalds 已提交
314
	{
315
		unsigned int ctrl;
316 317

		buf[0] = '\0';
318
#ifdef CONFIG_CPU_CP15_MMU
319 320 321 322 323 324 325 326
		{
			unsigned int transbase, dac;
			asm("mrc p15, 0, %0, c2, c0\n\t"
			    "mrc p15, 0, %1, c3, c0\n"
			    : "=r" (transbase), "=r" (dac));
			snprintf(buf, sizeof(buf), "  Table: %08x  DAC: %08x",
			  	transbase, dac);
		}
327
#endif
328 329 330 331
		asm("mrc p15, 0, %0, c1, c0\n" : "=r" (ctrl));

		printk("Control: %08x%s\n", ctrl, buf);
	}
332
#endif
L
Linus Torvalds 已提交
333 334
}

R
Russell King 已提交
335 336 337
void show_regs(struct pt_regs * regs)
{
	printk("\n");
338
	printk("Pid: %d, comm: %20s\n", task_pid_nr(current), current->comm);
R
Russell King 已提交
339
	__show_regs(regs);
340
	dump_stack();
R
Russell King 已提交
341 342
}

343 344 345 346
ATOMIC_NOTIFIER_HEAD(thread_notify_head);

EXPORT_SYMBOL_GPL(thread_notify_head);

L
Linus Torvalds 已提交
347 348 349 350 351
/*
 * Free current thread data structures etc..
 */
void exit_thread(void)
{
352
	thread_notify(THREAD_NOTIFY_EXIT, current_thread_info());
L
Linus Torvalds 已提交
353 354 355 356 357 358 359
}

void flush_thread(void)
{
	struct thread_info *thread = current_thread_info();
	struct task_struct *tsk = current;

360 361
	flush_ptrace_hw_breakpoint(tsk);

L
Linus Torvalds 已提交
362 363
	memset(thread->used_cp, 0, sizeof(thread->used_cp));
	memset(&tsk->thread.debug, 0, sizeof(struct debug_info));
364 365 366
	memset(&thread->fpstate, 0, sizeof(union fp_state));

	thread_notify(THREAD_NOTIFY_FLUSH, thread);
L
Linus Torvalds 已提交
367 368 369 370 371 372 373 374 375
}

void release_thread(struct task_struct *dead_task)
{
}

asmlinkage void ret_from_fork(void) __asm__("ret_from_fork");

int
A
Alexey Dobriyan 已提交
376
copy_thread(unsigned long clone_flags, unsigned long stack_start,
L
Linus Torvalds 已提交
377 378
	    unsigned long stk_sz, struct task_struct *p, struct pt_regs *regs)
{
A
Al Viro 已提交
379 380
	struct thread_info *thread = task_thread_info(p);
	struct pt_regs *childregs = task_pt_regs(p);
L
Linus Torvalds 已提交
381 382 383 384 385 386 387 388 389

	*childregs = *regs;
	childregs->ARM_r0 = 0;
	childregs->ARM_sp = stack_start;

	memset(&thread->cpu_context, 0, sizeof(struct cpu_context_save));
	thread->cpu_context.sp = (unsigned long)childregs;
	thread->cpu_context.pc = (unsigned long)ret_from_fork;

390 391
	clear_ptrace_hw_breakpoint(p);

L
Linus Torvalds 已提交
392 393 394
	if (clone_flags & CLONE_SETTLS)
		thread->tp_value = regs->ARM_r3;

395 396
	thread_notify(THREAD_NOTIFY_COPY, thread);

L
Linus Torvalds 已提交
397 398 399
	return 0;
}

400 401 402 403 404 405 406 407 408
/*
 * Fill in the task's elfregs structure for a core dump.
 */
int dump_task_regs(struct task_struct *t, elf_gregset_t *elfregs)
{
	elf_core_copy_regs(elfregs, task_pt_regs(t));
	return 1;
}

L
Linus Torvalds 已提交
409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425
/*
 * fill in the fpe structure for a core dump...
 */
int dump_fpu (struct pt_regs *regs, struct user_fp *fp)
{
	struct thread_info *thread = current_thread_info();
	int used_math = thread->used_cp[1] | thread->used_cp[2];

	if (used_math)
		memcpy(fp, &thread->fpstate.soft, sizeof (*fp));

	return used_math != 0;
}
EXPORT_SYMBOL(dump_fpu);

/*
 * Shuffle the argument into the correct register before calling the
426 427
 * thread function.  r4 is the thread argument, r5 is the pointer to
 * the thread function, and r6 points to the exit function.
L
Linus Torvalds 已提交
428 429
 */
extern void kernel_thread_helper(void);
430
asm(	".pushsection .text\n"
L
Linus Torvalds 已提交
431 432 433
"	.align\n"
"	.type	kernel_thread_helper, #function\n"
"kernel_thread_helper:\n"
434 435 436 437 438 439 440
#ifdef CONFIG_TRACE_IRQFLAGS
"	bl	trace_hardirqs_on\n"
#endif
"	msr	cpsr_c, r7\n"
"	mov	r0, r4\n"
"	mov	lr, r6\n"
"	mov	pc, r5\n"
L
Linus Torvalds 已提交
441
"	.size	kernel_thread_helper, . - kernel_thread_helper\n"
442
"	.popsection");
L
Linus Torvalds 已提交
443

444 445
#ifdef CONFIG_ARM_UNWIND
extern void kernel_thread_exit(long code);
446
asm(	".pushsection .text\n"
447 448 449 450 451 452 453 454 455
"	.align\n"
"	.type	kernel_thread_exit, #function\n"
"kernel_thread_exit:\n"
"	.fnstart\n"
"	.cantunwind\n"
"	bl	do_exit\n"
"	nop\n"
"	.fnend\n"
"	.size	kernel_thread_exit, . - kernel_thread_exit\n"
456
"	.popsection");
457 458 459 460
#else
#define kernel_thread_exit	do_exit
#endif

L
Linus Torvalds 已提交
461 462 463 464 465 466 467 468 469
/*
 * Create a kernel thread.
 */
pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)
{
	struct pt_regs regs;

	memset(&regs, 0, sizeof(regs));

470 471 472 473
	regs.ARM_r4 = (unsigned long)arg;
	regs.ARM_r5 = (unsigned long)fn;
	regs.ARM_r6 = (unsigned long)kernel_thread_exit;
	regs.ARM_r7 = SVC_MODE | PSR_ENDSTATE | PSR_ISETSTATE;
L
Linus Torvalds 已提交
474
	regs.ARM_pc = (unsigned long)kernel_thread_helper;
475
	regs.ARM_cpsr = regs.ARM_r7 | PSR_I_BIT;
L
Linus Torvalds 已提交
476 477 478 479 480 481 482

	return do_fork(flags|CLONE_VM|CLONE_UNTRACED, 0, &regs, 0, NULL, NULL);
}
EXPORT_SYMBOL(kernel_thread);

unsigned long get_wchan(struct task_struct *p)
{
483
	struct stackframe frame;
L
Linus Torvalds 已提交
484 485 486 487
	int count = 0;
	if (!p || p == current || p->state == TASK_RUNNING)
		return 0;

488 489 490 491
	frame.fp = thread_saved_fp(p);
	frame.sp = thread_saved_sp(p);
	frame.lr = 0;			/* recovered from the stack */
	frame.pc = thread_saved_pc(p);
L
Linus Torvalds 已提交
492
	do {
493 494
		int ret = unwind_frame(&frame);
		if (ret < 0)
L
Linus Torvalds 已提交
495
			return 0;
496 497
		if (!in_sched_functions(frame.pc))
			return frame.pc;
L
Linus Torvalds 已提交
498 499 500
	} while (count ++ < 16);
	return 0;
}
501 502 503 504 505 506

unsigned long arch_randomize_brk(struct mm_struct *mm)
{
	unsigned long range_end = mm->brk + 0x02000000;
	return randomize_range(mm->brk, range_end, 0) ? : mm->brk;
}
507

508
#ifdef CONFIG_MMU
509 510
/*
 * The vectors page is always readable from user space for the
511 512
 * atomic helpers and the signal restart code. Insert it into the
 * gate_vma so that it is visible through ptrace and /proc/<pid>/mem.
513
 */
514
static struct vm_area_struct gate_vma;
515

516
static int __init gate_vma_init(void)
517
{
518 519 520 521
	gate_vma.vm_start	= 0xffff0000;
	gate_vma.vm_end		= 0xffff0000 + PAGE_SIZE;
	gate_vma.vm_page_prot	= PAGE_READONLY_EXEC;
	gate_vma.vm_flags	= VM_READ | VM_EXEC |
522
				  VM_MAYREAD | VM_MAYEXEC;
523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539
	return 0;
}
arch_initcall(gate_vma_init);

struct vm_area_struct *get_gate_vma(struct mm_struct *mm)
{
	return &gate_vma;
}

int in_gate_area(struct mm_struct *mm, unsigned long addr)
{
	return (addr >= gate_vma.vm_start) && (addr < gate_vma.vm_end);
}

int in_gate_area_no_mm(unsigned long addr)
{
	return in_gate_area(NULL, addr);
540 541 542 543
}

const char *arch_vma_name(struct vm_area_struct *vma)
{
544
	return (vma == &gate_vma) ? "[vectors]" : NULL;
545
}
546
#endif