process.c 12.9 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>
R
Russell King 已提交
38
#include <asm/system.h>
39
#include <asm/thread_notify.h>
40
#include <asm/stacktrace.h>
41
#include <asm/mach/time.h>
L
Linus Torvalds 已提交
42

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

49 50 51 52 53 54 55
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"
};

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

60
extern void setup_mm_for_reboot(void);
L
Linus Torvalds 已提交
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 92

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);

93 94 95 96 97 98 99 100 101 102 103 104 105
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)
106
{
107
	phys_reset_t phys_reset;
108

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

112 113 114 115 116 117 118 119 120
	/* 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();

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

125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145
	/* 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();
146 147
}

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

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

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

161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179
static void do_nothing(void *unused)
{
}

/*
 * cpu_idle_wait - Used to ensure that all the CPUs discard old value of
 * pm_idle and update to new pm_idle value. Required while changing pm_idle
 * handler on SMP systems.
 *
 * Caller must have changed pm_idle to the new value before the call. Old
 * pm_idle value will not be used by any CPU after the return of this function.
 */
void cpu_idle_wait(void)
{
	smp_mb();
	/* kick all the CPUs so that they exit out of pm_idle */
	smp_call_function(do_nothing, NULL, 1);
}
EXPORT_SYMBOL_GPL(cpu_idle_wait);
180

L
Linus Torvalds 已提交
181
/*
N
Nicolas Pitre 已提交
182
 * This is our default idle handler.
L
Linus Torvalds 已提交
183
 */
N
Nicolas Pitre 已提交
184 185 186

void (*arm_pm_idle)(void);

187
static void default_idle(void)
L
Linus Torvalds 已提交
188
{
N
Nicolas Pitre 已提交
189 190 191
	if (arm_pm_idle)
		arm_pm_idle();
	else
N
Nicolas Pitre 已提交
192
		cpu_do_idle();
193
	local_irq_enable();
L
Linus Torvalds 已提交
194 195
}

196 197 198
void (*pm_idle)(void) = default_idle;
EXPORT_SYMBOL(pm_idle);

L
Linus Torvalds 已提交
199
/*
200 201 202 203
 * 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 已提交
204 205 206 207 208 209 210
 */
void cpu_idle(void)
{
	local_fiq_enable();

	/* endless idle loop with no priority at all */
	while (1) {
211 212
		tick_nohz_idle_enter();
		rcu_idle_enter();
213 214
		leds_event(led_idle_start);
		while (!need_resched()) {
215
#ifdef CONFIG_HOTPLUG_CPU
216 217
			if (cpu_is_offline(smp_processor_id()))
				cpu_die();
218 219
#endif

N
Nicolas Pitre 已提交
220 221 222 223
			/*
			 * We need to disable interrupts here
			 * to ensure we don't miss a wakeup call.
			 */
224
			local_irq_disable();
225 226 227
#ifdef CONFIG_PL310_ERRATA_769419
			wmb();
#endif
228 229 230
			if (hlt_counter) {
				local_irq_enable();
				cpu_relax();
N
Nicolas Pitre 已提交
231
			} else if (!need_resched()) {
232
				stop_critical_timings();
233
				if (cpuidle_idle_call())
L
Len Brown 已提交
234
					pm_idle();
235 236
				start_critical_timings();
				/*
N
Nicolas Pitre 已提交
237 238
				 * pm_idle functions must always
				 * return with IRQs enabled.
239 240
				 */
				WARN_ON(irqs_disabled());
N
Nicolas Pitre 已提交
241
			} else
242 243
				local_irq_enable();
		}
L
Linus Torvalds 已提交
244
		leds_event(led_idle_end);
245 246
		rcu_idle_exit();
		tick_nohz_idle_exit();
247
		preempt_enable_no_resched();
L
Linus Torvalds 已提交
248
		schedule();
249
		preempt_disable();
L
Linus Torvalds 已提交
250 251 252 253 254 255 256 257 258 259 260 261 262
	}
}

static char reboot_mode = 'h';

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

__setup("reboot=", reboot_setup);

263
void machine_shutdown(void)
L
Linus Torvalds 已提交
264
{
265 266 267
#ifdef CONFIG_SMP
	smp_send_stop();
#endif
L
Linus Torvalds 已提交
268 269
}

270 271 272 273 274
void machine_halt(void)
{
	machine_shutdown();
	while (1);
}
L
Linus Torvalds 已提交
275 276 277

void machine_power_off(void)
{
278
	machine_shutdown();
L
Linus Torvalds 已提交
279 280 281 282
	if (pm_power_off)
		pm_power_off();
}

283
void machine_restart(char *cmd)
L
Linus Torvalds 已提交
284
{
285
	machine_shutdown();
286

287
	arm_pm_restart(reboot_mode, cmd);
288 289 290 291 292 293 294

	/* 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 已提交
295 296
}

R
Russell King 已提交
297
void __show_regs(struct pt_regs *regs)
L
Linus Torvalds 已提交
298
{
299 300
	unsigned long flags;
	char buf[64];
L
Linus Torvalds 已提交
301

302
	printk("CPU: %d    %s  (%s %.*s)\n",
303 304
		raw_smp_processor_id(), print_tainted(),
		init_utsname()->release,
305 306
		(int)strcspn(init_utsname()->version, " "),
		init_utsname()->version);
L
Linus Torvalds 已提交
307 308
	print_symbol("PC is at %s\n", instruction_pointer(regs));
	print_symbol("LR is at %s\n", regs->ARM_lr);
309
	printk("pc : [<%08lx>]    lr : [<%08lx>]    psr: %08lx\n"
L
Linus Torvalds 已提交
310
	       "sp : %08lx  ip : %08lx  fp : %08lx\n",
311 312
		regs->ARM_pc, regs->ARM_lr, regs->ARM_cpsr,
		regs->ARM_sp, regs->ARM_ip, regs->ARM_fp);
L
Linus Torvalds 已提交
313 314 315 316 317 318 319 320 321
	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);
322 323 324 325 326 327 328 329

	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';

330
	printk("Flags: %s  IRQs o%s  FIQs o%s  Mode %s  ISA %s  Segment %s\n",
331
		buf, interrupts_enabled(regs) ? "n" : "ff",
L
Linus Torvalds 已提交
332 333
		fast_interrupts_enabled(regs) ? "n" : "ff",
		processor_modes[processor_mode(regs)],
334
		isa_modes[isa_mode(regs)],
L
Linus Torvalds 已提交
335
		get_fs() == get_ds() ? "kernel" : "user");
336
#ifdef CONFIG_CPU_CP15
L
Linus Torvalds 已提交
337
	{
338
		unsigned int ctrl;
339 340

		buf[0] = '\0';
341
#ifdef CONFIG_CPU_CP15_MMU
342 343 344 345 346 347 348 349
		{
			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);
		}
350
#endif
351 352 353 354
		asm("mrc p15, 0, %0, c1, c0\n" : "=r" (ctrl));

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

R
Russell King 已提交
358 359 360
void show_regs(struct pt_regs * regs)
{
	printk("\n");
361
	printk("Pid: %d, comm: %20s\n", task_pid_nr(current), current->comm);
R
Russell King 已提交
362
	__show_regs(regs);
363
	dump_stack();
R
Russell King 已提交
364 365
}

366 367 368 369
ATOMIC_NOTIFIER_HEAD(thread_notify_head);

EXPORT_SYMBOL_GPL(thread_notify_head);

L
Linus Torvalds 已提交
370 371 372 373 374
/*
 * Free current thread data structures etc..
 */
void exit_thread(void)
{
375
	thread_notify(THREAD_NOTIFY_EXIT, current_thread_info());
L
Linus Torvalds 已提交
376 377 378 379 380 381 382
}

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

383 384
	flush_ptrace_hw_breakpoint(tsk);

L
Linus Torvalds 已提交
385 386
	memset(thread->used_cp, 0, sizeof(thread->used_cp));
	memset(&tsk->thread.debug, 0, sizeof(struct debug_info));
387 388 389
	memset(&thread->fpstate, 0, sizeof(union fp_state));

	thread_notify(THREAD_NOTIFY_FLUSH, thread);
L
Linus Torvalds 已提交
390 391 392 393 394 395 396 397 398
}

void release_thread(struct task_struct *dead_task)
{
}

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

int
A
Alexey Dobriyan 已提交
399
copy_thread(unsigned long clone_flags, unsigned long stack_start,
L
Linus Torvalds 已提交
400 401
	    unsigned long stk_sz, struct task_struct *p, struct pt_regs *regs)
{
A
Al Viro 已提交
402 403
	struct thread_info *thread = task_thread_info(p);
	struct pt_regs *childregs = task_pt_regs(p);
L
Linus Torvalds 已提交
404 405 406 407 408 409 410 411 412

	*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;

413 414
	clear_ptrace_hw_breakpoint(p);

L
Linus Torvalds 已提交
415 416 417
	if (clone_flags & CLONE_SETTLS)
		thread->tp_value = regs->ARM_r3;

418 419
	thread_notify(THREAD_NOTIFY_COPY, thread);

L
Linus Torvalds 已提交
420 421 422
	return 0;
}

423 424 425 426 427 428 429 430 431
/*
 * 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 已提交
432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448
/*
 * 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
449 450
 * 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 已提交
451 452
 */
extern void kernel_thread_helper(void);
453
asm(	".pushsection .text\n"
L
Linus Torvalds 已提交
454 455 456
"	.align\n"
"	.type	kernel_thread_helper, #function\n"
"kernel_thread_helper:\n"
457 458 459 460 461 462 463
#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 已提交
464
"	.size	kernel_thread_helper, . - kernel_thread_helper\n"
465
"	.popsection");
L
Linus Torvalds 已提交
466

467 468
#ifdef CONFIG_ARM_UNWIND
extern void kernel_thread_exit(long code);
469
asm(	".pushsection .text\n"
470 471 472 473 474 475 476 477 478
"	.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"
479
"	.popsection");
480 481 482 483
#else
#define kernel_thread_exit	do_exit
#endif

L
Linus Torvalds 已提交
484 485 486 487 488 489 490 491 492
/*
 * 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));

493 494 495 496
	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 已提交
497
	regs.ARM_pc = (unsigned long)kernel_thread_helper;
498
	regs.ARM_cpsr = regs.ARM_r7 | PSR_I_BIT;
L
Linus Torvalds 已提交
499 500 501 502 503 504 505

	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)
{
506
	struct stackframe frame;
L
Linus Torvalds 已提交
507 508 509 510
	int count = 0;
	if (!p || p == current || p->state == TASK_RUNNING)
		return 0;

511 512 513 514
	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 已提交
515
	do {
516 517
		int ret = unwind_frame(&frame);
		if (ret < 0)
L
Linus Torvalds 已提交
518
			return 0;
519 520
		if (!in_sched_functions(frame.pc))
			return frame.pc;
L
Linus Torvalds 已提交
521 522 523
	} while (count ++ < 16);
	return 0;
}
524 525 526 527 528 529

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;
}
530

531
#ifdef CONFIG_MMU
532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551
/*
 * The vectors page is always readable from user space for the
 * atomic helpers and the signal restart code.  Let's declare a mapping
 * for it so it is visible through ptrace and /proc/<pid>/mem.
 */

int vectors_user_mapping(void)
{
	struct mm_struct *mm = current->mm;
	return install_special_mapping(mm, 0xffff0000, PAGE_SIZE,
				       VM_READ | VM_EXEC |
				       VM_MAYREAD | VM_MAYEXEC |
				       VM_ALWAYSDUMP | VM_RESERVED,
				       NULL);
}

const char *arch_vma_name(struct vm_area_struct *vma)
{
	return (vma->vm_start == 0xffff0000) ? "[vectors]" : NULL;
}
552
#endif