diff options
author | Glauber de Oliveira Costa <gcosta@redhat.com> | 2008-03-19 13:25:45 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-04-17 11:41:02 -0400 |
commit | 365c894c65b98da944992199ea24206f531674de (patch) | |
tree | 5d3048f9e101c36d47aa823f282c865dcffa6dbe /arch | |
parent | ddd10ecfa231c88382fc2f10a3120d2ad8e92381 (diff) |
x86: use create_idle struct in do_boot_cpu
Use a new worker, with help of the create_idle struct
to fork the idle thread. We now have two workers, the first
of them triggered by __smp_prepare_cpu. But the later is
going away soon.
Signed-off-by: Glauber Costa <gcosta@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch')
-rw-r--r-- | arch/x86/kernel/smpboot_32.c | 86 |
1 files changed, 59 insertions, 27 deletions
diff --git a/arch/x86/kernel/smpboot_32.c b/arch/x86/kernel/smpboot_32.c index c30abed0892..fc1eb5255f6 100644 --- a/arch/x86/kernel/smpboot_32.c +++ b/arch/x86/kernel/smpboot_32.c | |||
@@ -79,6 +79,24 @@ static void map_cpu_to_logical_apicid(void); | |||
79 | /* State of each CPU. */ | 79 | /* State of each CPU. */ |
80 | DEFINE_PER_CPU(int, cpu_state) = { 0 }; | 80 | DEFINE_PER_CPU(int, cpu_state) = { 0 }; |
81 | 81 | ||
82 | /* Store all idle threads, this can be reused instead of creating | ||
83 | * a new thread. Also avoids complicated thread destroy functionality | ||
84 | * for idle threads. | ||
85 | */ | ||
86 | #ifdef CONFIG_HOTPLUG_CPU | ||
87 | /* | ||
88 | * Needed only for CONFIG_HOTPLUG_CPU because __cpuinitdata is | ||
89 | * removed after init for !CONFIG_HOTPLUG_CPU. | ||
90 | */ | ||
91 | static DEFINE_PER_CPU(struct task_struct *, idle_thread_array); | ||
92 | #define get_idle_for_cpu(x) (per_cpu(idle_thread_array, x)) | ||
93 | #define set_idle_for_cpu(x, p) (per_cpu(idle_thread_array, x) = (p)) | ||
94 | #else | ||
95 | struct task_struct *idle_thread_array[NR_CPUS] __cpuinitdata ; | ||
96 | #define get_idle_for_cpu(x) (idle_thread_array[(x)]) | ||
97 | #define set_idle_for_cpu(x, p) (idle_thread_array[(x)] = (p)) | ||
98 | #endif | ||
99 | |||
82 | static atomic_t init_deasserted; | 100 | static atomic_t init_deasserted; |
83 | 101 | ||
84 | static void __cpuinit smp_callin(void) | 102 | static void __cpuinit smp_callin(void) |
@@ -513,30 +531,21 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
513 | 531 | ||
514 | extern cpumask_t cpu_initialized; | 532 | extern cpumask_t cpu_initialized; |
515 | 533 | ||
516 | #ifdef CONFIG_HOTPLUG_CPU | 534 | struct create_idle { |
517 | static struct task_struct * __cpuinitdata cpu_idle_tasks[NR_CPUS]; | 535 | struct work_struct work; |
518 | static inline struct task_struct * __cpuinit alloc_idle_task(int cpu) | ||
519 | { | ||
520 | struct task_struct *idle; | 536 | struct task_struct *idle; |
537 | struct completion done; | ||
538 | int cpu; | ||
539 | }; | ||
521 | 540 | ||
522 | if ((idle = cpu_idle_tasks[cpu]) != NULL) { | 541 | static void __cpuinit do_fork_idle(struct work_struct *work) |
523 | /* initialize thread_struct. we really want to avoid destroy | 542 | { |
524 | * idle tread | 543 | struct create_idle *c_idle = |
525 | */ | 544 | container_of(work, struct create_idle, work); |
526 | idle->thread.sp = (unsigned long)task_pt_regs(idle); | ||
527 | init_idle(idle, cpu); | ||
528 | return idle; | ||
529 | } | ||
530 | idle = fork_idle(cpu); | ||
531 | 545 | ||
532 | if (!IS_ERR(idle)) | 546 | c_idle->idle = fork_idle(c_idle->cpu); |
533 | cpu_idle_tasks[cpu] = idle; | 547 | complete(&c_idle->done); |
534 | return idle; | ||
535 | } | 548 | } |
536 | #else | ||
537 | #define alloc_idle_task(cpu) fork_idle(cpu) | ||
538 | #endif | ||
539 | |||
540 | static int __cpuinit do_boot_cpu(int apicid, int cpu) | 549 | static int __cpuinit do_boot_cpu(int apicid, int cpu) |
541 | /* | 550 | /* |
542 | * NOTE - on most systems this is a PHYSICAL apic ID, but on multiquad | 551 | * NOTE - on most systems this is a PHYSICAL apic ID, but on multiquad |
@@ -544,11 +553,15 @@ static int __cpuinit do_boot_cpu(int apicid, int cpu) | |||
544 | * Returns zero if CPU booted OK, else error code from wakeup_secondary_cpu. | 553 | * Returns zero if CPU booted OK, else error code from wakeup_secondary_cpu. |
545 | */ | 554 | */ |
546 | { | 555 | { |
547 | struct task_struct *idle; | ||
548 | unsigned long boot_error; | 556 | unsigned long boot_error; |
549 | int timeout; | 557 | int timeout; |
550 | unsigned long start_eip; | 558 | unsigned long start_eip; |
551 | unsigned short nmi_high = 0, nmi_low = 0; | 559 | unsigned short nmi_high = 0, nmi_low = 0; |
560 | struct create_idle c_idle = { | ||
561 | .cpu = cpu, | ||
562 | .done = COMPLETION_INITIALIZER_ONSTACK(c_idle.done), | ||
563 | }; | ||
564 | INIT_WORK(&c_idle.work, do_fork_idle); | ||
552 | 565 | ||
553 | /* | 566 | /* |
554 | * Save current MTRR state in case it was changed since early boot | 567 | * Save current MTRR state in case it was changed since early boot |
@@ -556,19 +569,38 @@ static int __cpuinit do_boot_cpu(int apicid, int cpu) | |||
556 | */ | 569 | */ |
557 | mtrr_save_state(); | 570 | mtrr_save_state(); |
558 | 571 | ||
572 | c_idle.idle = get_idle_for_cpu(cpu); | ||
573 | |||
559 | /* | 574 | /* |
560 | * We can't use kernel_thread since we must avoid to | 575 | * We can't use kernel_thread since we must avoid to |
561 | * reschedule the child. | 576 | * reschedule the child. |
562 | */ | 577 | */ |
563 | idle = alloc_idle_task(cpu); | 578 | if (c_idle.idle) { |
564 | if (IS_ERR(idle)) | 579 | c_idle.idle->thread.sp = (unsigned long) (((struct pt_regs *) |
565 | panic("failed fork for CPU %d", cpu); | 580 | (THREAD_SIZE + task_stack_page(c_idle.idle))) - 1); |
581 | init_idle(c_idle.idle, cpu); | ||
582 | goto do_rest; | ||
583 | } | ||
584 | |||
585 | if (!keventd_up() || current_is_keventd()) | ||
586 | c_idle.work.func(&c_idle.work); | ||
587 | else { | ||
588 | schedule_work(&c_idle.work); | ||
589 | wait_for_completion(&c_idle.done); | ||
590 | } | ||
591 | |||
592 | if (IS_ERR(c_idle.idle)) { | ||
593 | printk(KERN_ERR "failed fork for CPU %d\n", cpu); | ||
594 | return PTR_ERR(c_idle.idle); | ||
595 | } | ||
566 | 596 | ||
597 | set_idle_for_cpu(cpu, c_idle.idle); | ||
598 | do_rest: | ||
599 | per_cpu(current_task, cpu) = c_idle.idle; | ||
567 | init_gdt(cpu); | 600 | init_gdt(cpu); |
568 | per_cpu(current_task, cpu) = idle; | ||
569 | early_gdt_descr.address = (unsigned long)get_cpu_gdt_table(cpu); | 601 | early_gdt_descr.address = (unsigned long)get_cpu_gdt_table(cpu); |
570 | 602 | ||
571 | idle->thread.ip = (unsigned long) start_secondary; | 603 | c_idle.idle->thread.ip = (unsigned long) start_secondary; |
572 | /* start_eip had better be page-aligned! */ | 604 | /* start_eip had better be page-aligned! */ |
573 | start_eip = setup_trampoline(); | 605 | start_eip = setup_trampoline(); |
574 | 606 | ||
@@ -577,7 +609,7 @@ static int __cpuinit do_boot_cpu(int apicid, int cpu) | |||
577 | /* So we see what's up */ | 609 | /* So we see what's up */ |
578 | printk("Booting processor %d/%d ip %lx\n", cpu, apicid, start_eip); | 610 | printk("Booting processor %d/%d ip %lx\n", cpu, apicid, start_eip); |
579 | /* Stack for startup_32 can be just as for start_secondary onwards */ | 611 | /* Stack for startup_32 can be just as for start_secondary onwards */ |
580 | stack_start.sp = (void *) idle->thread.sp; | 612 | stack_start.sp = (void *) c_idle.idle->thread.sp; |
581 | 613 | ||
582 | irq_ctx_init(cpu); | 614 | irq_ctx_init(cpu); |
583 | 615 | ||