diff options
Diffstat (limited to 'arch/powerpc')
-rw-r--r-- | arch/powerpc/Kconfig | 3 | ||||
-rw-r--r-- | arch/powerpc/kernel/setup_64.c | 61 | ||||
-rw-r--r-- | arch/powerpc/kernel/vmlinux.lds.S | 9 | ||||
-rw-r--r-- | arch/powerpc/mm/stab.c | 2 | ||||
-rw-r--r-- | arch/powerpc/platforms/ps3/smp.c | 2 |
5 files changed, 53 insertions, 24 deletions
diff --git a/arch/powerpc/Kconfig b/arch/powerpc/Kconfig index d00131ca0835..2c42e1526d03 100644 --- a/arch/powerpc/Kconfig +++ b/arch/powerpc/Kconfig | |||
@@ -49,6 +49,9 @@ config GENERIC_HARDIRQS_NO__DO_IRQ | |||
49 | config HAVE_SETUP_PER_CPU_AREA | 49 | config HAVE_SETUP_PER_CPU_AREA |
50 | def_bool PPC64 | 50 | def_bool PPC64 |
51 | 51 | ||
52 | config NEED_PER_CPU_EMBED_FIRST_CHUNK | ||
53 | def_bool PPC64 | ||
54 | |||
52 | config IRQ_PER_CPU | 55 | config IRQ_PER_CPU |
53 | bool | 56 | bool |
54 | default y | 57 | default y |
diff --git a/arch/powerpc/kernel/setup_64.c b/arch/powerpc/kernel/setup_64.c index 1f6816003ebe..aa6e4500635f 100644 --- a/arch/powerpc/kernel/setup_64.c +++ b/arch/powerpc/kernel/setup_64.c | |||
@@ -57,6 +57,7 @@ | |||
57 | #include <asm/cache.h> | 57 | #include <asm/cache.h> |
58 | #include <asm/page.h> | 58 | #include <asm/page.h> |
59 | #include <asm/mmu.h> | 59 | #include <asm/mmu.h> |
60 | #include <asm/mmu-hash64.h> | ||
60 | #include <asm/firmware.h> | 61 | #include <asm/firmware.h> |
61 | #include <asm/xmon.h> | 62 | #include <asm/xmon.h> |
62 | #include <asm/udbg.h> | 63 | #include <asm/udbg.h> |
@@ -569,25 +570,53 @@ void cpu_die(void) | |||
569 | } | 570 | } |
570 | 571 | ||
571 | #ifdef CONFIG_SMP | 572 | #ifdef CONFIG_SMP |
572 | void __init setup_per_cpu_areas(void) | 573 | #define PCPU_DYN_SIZE () |
574 | |||
575 | static void * __init pcpu_fc_alloc(unsigned int cpu, size_t size, size_t align) | ||
573 | { | 576 | { |
574 | int i; | 577 | return __alloc_bootmem_node(NODE_DATA(cpu_to_node(cpu)), size, align, |
575 | unsigned long size; | 578 | __pa(MAX_DMA_ADDRESS)); |
576 | char *ptr; | 579 | } |
577 | |||
578 | /* Copy section for each CPU (we discard the original) */ | ||
579 | size = ALIGN(__per_cpu_end - __per_cpu_start, PAGE_SIZE); | ||
580 | #ifdef CONFIG_MODULES | ||
581 | if (size < PERCPU_ENOUGH_ROOM) | ||
582 | size = PERCPU_ENOUGH_ROOM; | ||
583 | #endif | ||
584 | 580 | ||
585 | for_each_possible_cpu(i) { | 581 | static void __init pcpu_fc_free(void *ptr, size_t size) |
586 | ptr = alloc_bootmem_pages_node(NODE_DATA(cpu_to_node(i)), size); | 582 | { |
583 | free_bootmem(__pa(ptr), size); | ||
584 | } | ||
587 | 585 | ||
588 | paca[i].data_offset = ptr - __per_cpu_start; | 586 | static int pcpu_cpu_distance(unsigned int from, unsigned int to) |
589 | memcpy(ptr, __per_cpu_start, __per_cpu_end - __per_cpu_start); | 587 | { |
590 | } | 588 | if (cpu_to_node(from) == cpu_to_node(to)) |
589 | return LOCAL_DISTANCE; | ||
590 | else | ||
591 | return REMOTE_DISTANCE; | ||
592 | } | ||
593 | |||
594 | void __init setup_per_cpu_areas(void) | ||
595 | { | ||
596 | const size_t dyn_size = PERCPU_MODULE_RESERVE + PERCPU_DYNAMIC_RESERVE; | ||
597 | size_t atom_size; | ||
598 | unsigned long delta; | ||
599 | unsigned int cpu; | ||
600 | int rc; | ||
601 | |||
602 | /* | ||
603 | * Linear mapping is one of 4K, 1M and 16M. For 4K, no need | ||
604 | * to group units. For larger mappings, use 1M atom which | ||
605 | * should be large enough to contain a number of units. | ||
606 | */ | ||
607 | if (mmu_linear_psize == MMU_PAGE_4K) | ||
608 | atom_size = PAGE_SIZE; | ||
609 | else | ||
610 | atom_size = 1 << 20; | ||
611 | |||
612 | rc = pcpu_embed_first_chunk(0, dyn_size, atom_size, pcpu_cpu_distance, | ||
613 | pcpu_fc_alloc, pcpu_fc_free); | ||
614 | if (rc < 0) | ||
615 | panic("cannot initialize percpu area (err=%d)", rc); | ||
616 | |||
617 | delta = (unsigned long)pcpu_base_addr - (unsigned long)__per_cpu_start; | ||
618 | for_each_possible_cpu(cpu) | ||
619 | paca[cpu].data_offset = delta + pcpu_unit_offsets[cpu]; | ||
591 | } | 620 | } |
592 | #endif | 621 | #endif |
593 | 622 | ||
diff --git a/arch/powerpc/kernel/vmlinux.lds.S b/arch/powerpc/kernel/vmlinux.lds.S index 8ef8a14abc95..244e3658983c 100644 --- a/arch/powerpc/kernel/vmlinux.lds.S +++ b/arch/powerpc/kernel/vmlinux.lds.S | |||
@@ -37,12 +37,6 @@ jiffies = jiffies_64 + 4; | |||
37 | #endif | 37 | #endif |
38 | SECTIONS | 38 | SECTIONS |
39 | { | 39 | { |
40 | /* Sections to be discarded. */ | ||
41 | /DISCARD/ : { | ||
42 | *(.exitcall.exit) | ||
43 | EXIT_DATA | ||
44 | } | ||
45 | |||
46 | . = KERNELBASE; | 40 | . = KERNELBASE; |
47 | 41 | ||
48 | /* | 42 | /* |
@@ -298,4 +292,7 @@ SECTIONS | |||
298 | . = ALIGN(PAGE_SIZE); | 292 | . = ALIGN(PAGE_SIZE); |
299 | _end = . ; | 293 | _end = . ; |
300 | PROVIDE32 (end = .); | 294 | PROVIDE32 (end = .); |
295 | |||
296 | /* Sections to be discarded. */ | ||
297 | DISCARDS | ||
301 | } | 298 | } |
diff --git a/arch/powerpc/mm/stab.c b/arch/powerpc/mm/stab.c index ab5fb48b3e90..687fddaa24c5 100644 --- a/arch/powerpc/mm/stab.c +++ b/arch/powerpc/mm/stab.c | |||
@@ -31,7 +31,7 @@ struct stab_entry { | |||
31 | 31 | ||
32 | #define NR_STAB_CACHE_ENTRIES 8 | 32 | #define NR_STAB_CACHE_ENTRIES 8 |
33 | static DEFINE_PER_CPU(long, stab_cache_ptr); | 33 | static DEFINE_PER_CPU(long, stab_cache_ptr); |
34 | static DEFINE_PER_CPU(long, stab_cache[NR_STAB_CACHE_ENTRIES]); | 34 | static DEFINE_PER_CPU(long [NR_STAB_CACHE_ENTRIES], stab_cache); |
35 | 35 | ||
36 | /* | 36 | /* |
37 | * Create a segment table entry for the given esid/vsid pair. | 37 | * Create a segment table entry for the given esid/vsid pair. |
diff --git a/arch/powerpc/platforms/ps3/smp.c b/arch/powerpc/platforms/ps3/smp.c index f6e04bcc70ef..51ffde40af2b 100644 --- a/arch/powerpc/platforms/ps3/smp.c +++ b/arch/powerpc/platforms/ps3/smp.c | |||
@@ -37,7 +37,7 @@ | |||
37 | */ | 37 | */ |
38 | 38 | ||
39 | #define MSG_COUNT 4 | 39 | #define MSG_COUNT 4 |
40 | static DEFINE_PER_CPU(unsigned int, ps3_ipi_virqs[MSG_COUNT]); | 40 | static DEFINE_PER_CPU(unsigned int [MSG_COUNT], ps3_ipi_virqs); |
41 | 41 | ||
42 | static void do_message_pass(int target, int msg) | 42 | static void do_message_pass(int target, int msg) |
43 | { | 43 | { |