mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 6727ad9e20
			
		
	
	
		6727ad9e20
		
	
	
	
	
		
			
			When doing an nmi backtrace of many cores, most of which are idle, the output is a little overwhelming and very uninformative. Suppress messages for cpus that are idling when they are interrupted and just emit one line, "NMI backtrace for N skipped: idling at pc 0xNNN". We do this by grouping all the cpuidle code together into a new .cpuidle.text section, and then checking the address of the interrupted PC to see if it lies within that section. This commit suitably tags x86 and tile idle routines, and only adds in the minimal framework for other architectures. Link: http://lkml.kernel.org/r/1472487169-14923-5-git-send-email-cmetcalf@mellanox.com Signed-off-by: Chris Metcalf <cmetcalf@mellanox.com> Acked-by: Peter Zijlstra (Intel) <peterz@infradead.org> Tested-by: Peter Zijlstra (Intel) <peterz@infradead.org> Tested-by: Daniel Thompson <daniel.thompson@linaro.org> [arm] Tested-by: Petr Mladek <pmladek@suse.com> Cc: Aaron Tomlin <atomlin@redhat.com> Cc: Peter Zijlstra (Intel) <peterz@infradead.org> Cc: "Rafael J. Wysocki" <rjw@rjwysocki.net> Cc: Russell King <linux@arm.linux.org.uk> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: Ingo Molnar <mingo@elte.hu> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			170 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| #include <asm-generic/vmlinux.lds.h>
 | |
| #include <asm/page.h>
 | |
| 
 | |
| OUTPUT_FORMAT(ELF_FORMAT)
 | |
| OUTPUT_ARCH(ELF_ARCH)
 | |
| ENTRY(_start)
 | |
| jiffies = jiffies_64;
 | |
| 
 | |
| SECTIONS
 | |
| {
 | |
|   PROVIDE (__executable_start = START);
 | |
|   . = START + SIZEOF_HEADERS;
 | |
|   .interp         : { *(.interp) }
 | |
|   __binary_start = .;
 | |
|   . = ALIGN(4096);		/* Init code and data */
 | |
|   _text = .;
 | |
|   INIT_TEXT_SECTION(PAGE_SIZE)
 | |
| 
 | |
|   . = ALIGN(PAGE_SIZE);
 | |
| 
 | |
|   /* Read-only sections, merged into text segment: */
 | |
|   .hash           : { *(.hash) }
 | |
|   .gnu.hash       : { *(.gnu.hash) }
 | |
|   .dynsym         : { *(.dynsym) }
 | |
|   .dynstr         : { *(.dynstr) }
 | |
|   .gnu.version    : { *(.gnu.version) }
 | |
|   .gnu.version_d  : { *(.gnu.version_d) }
 | |
|   .gnu.version_r  : { *(.gnu.version_r) }
 | |
|   .rel.init       : { *(.rel.init) }
 | |
|   .rela.init      : { *(.rela.init) }
 | |
|   .rel.text       : { *(.rel.text .rel.text.* .rel.gnu.linkonce.t.*) }
 | |
|   .rela.text      : { *(.rela.text .rela.text.* .rela.gnu.linkonce.t.*) }
 | |
|   .rel.fini       : { *(.rel.fini) }
 | |
|   .rela.fini      : { *(.rela.fini) }
 | |
|   .rel.rodata     : { *(.rel.rodata .rel.rodata.* .rel.gnu.linkonce.r.*) }
 | |
|   .rela.rodata    : { *(.rela.rodata .rela.rodata.* .rela.gnu.linkonce.r.*) }
 | |
|   .rel.data       : { *(.rel.data .rel.data.* .rel.gnu.linkonce.d.*) }
 | |
|   .rela.data      : { *(.rela.data .rela.data.* .rela.gnu.linkonce.d.*) }
 | |
|   .rel.tdata	  : { *(.rel.tdata .rel.tdata.* .rel.gnu.linkonce.td.*) }
 | |
|   .rela.tdata	  : { *(.rela.tdata .rela.tdata.* .rela.gnu.linkonce.td.*) }
 | |
|   .rel.tbss	  : { *(.rel.tbss .rel.tbss.* .rel.gnu.linkonce.tb.*) }
 | |
|   .rela.tbss	  : { *(.rela.tbss .rela.tbss.* .rela.gnu.linkonce.tb.*) }
 | |
|   .rel.ctors      : { *(.rel.ctors) }
 | |
|   .rela.ctors     : { *(.rela.ctors) }
 | |
|   .rel.dtors      : { *(.rel.dtors) }
 | |
|   .rela.dtors     : { *(.rela.dtors) }
 | |
|   .rel.got        : { *(.rel.got) }
 | |
|   .rela.got       : { *(.rela.got) }
 | |
|   .rel.bss        : { *(.rel.bss .rel.bss.* .rel.gnu.linkonce.b.*) }
 | |
|   .rela.bss       : { *(.rela.bss .rela.bss.* .rela.gnu.linkonce.b.*) }
 | |
|   .rel.plt : {
 | |
| 	*(.rel.plt)
 | |
| 	PROVIDE_HIDDEN(__rel_iplt_start = .);
 | |
| 	*(.rel.iplt)
 | |
| 	PROVIDE_HIDDEN(__rel_iplt_end = .);
 | |
|   }
 | |
|   .rela.plt : {
 | |
| 	*(.rela.plt)
 | |
| 	PROVIDE_HIDDEN(__rela_iplt_start = .);
 | |
| 	*(.rela.iplt)
 | |
| 	PROVIDE_HIDDEN(__rela_iplt_end = .);
 | |
|   }
 | |
|   .init           : {
 | |
|     KEEP (*(.init))
 | |
|   } =0x90909090
 | |
|   .plt            : { *(.plt) }
 | |
|   .text           : {
 | |
|     _stext = .;
 | |
|     TEXT_TEXT
 | |
|     SCHED_TEXT
 | |
|     CPUIDLE_TEXT
 | |
|     LOCK_TEXT
 | |
|     *(.fixup)
 | |
|     *(.stub .text.* .gnu.linkonce.t.*)
 | |
|     /* .gnu.warning sections are handled specially by elf32.em.  */
 | |
|     *(.gnu.warning)
 | |
| 
 | |
|     . = ALIGN(PAGE_SIZE);
 | |
|   } =0x90909090
 | |
|   . = ALIGN(PAGE_SIZE);
 | |
|   .syscall_stub : {
 | |
| 	__syscall_stub_start = .;
 | |
| 	*(.__syscall_stub*)
 | |
| 	__syscall_stub_end = .;
 | |
|   }
 | |
|   .fini           : {
 | |
|     KEEP (*(.fini))
 | |
|   } =0x90909090
 | |
| 
 | |
|   .kstrtab : { *(.kstrtab) }
 | |
| 
 | |
|   #include <asm/common.lds.S>
 | |
| 
 | |
|   __init_begin = .;
 | |
|   init.data : { INIT_DATA }
 | |
|   __init_end = .;
 | |
| 
 | |
|   /* Ensure the __preinit_array_start label is properly aligned.  We
 | |
|      could instead move the label definition inside the section, but
 | |
|      the linker would then create the section even if it turns out to
 | |
|      be empty, which isn't pretty.  */
 | |
|   . = ALIGN(32 / 8);
 | |
|   .preinit_array     : { *(.preinit_array) }
 | |
|   .init_array     : { *(.init_array) }
 | |
|   .fini_array     : { *(.fini_array) }
 | |
|   .data           : {
 | |
|     INIT_TASK_DATA(KERNEL_STACK_SIZE)
 | |
|     . = ALIGN(KERNEL_STACK_SIZE);
 | |
|     *(.data..init_irqstack)
 | |
|     DATA_DATA
 | |
|     *(.data.* .gnu.linkonce.d.*)
 | |
|     SORT(CONSTRUCTORS)
 | |
|   }
 | |
|   .data1          : { *(.data1) }
 | |
|   .tdata	  : { *(.tdata .tdata.* .gnu.linkonce.td.*) }
 | |
|   .tbss		  : { *(.tbss .tbss.* .gnu.linkonce.tb.*) *(.tcommon) }
 | |
|   .eh_frame       : { KEEP (*(.eh_frame)) }
 | |
|   .gcc_except_table   : { *(.gcc_except_table) }
 | |
|   .dynamic        : { *(.dynamic) }
 | |
|   .ctors          : {
 | |
|     /* gcc uses crtbegin.o to find the start of
 | |
|        the constructors, so we make sure it is
 | |
|        first.  Because this is a wildcard, it
 | |
|        doesn't matter if the user does not
 | |
|        actually link against crtbegin.o; the
 | |
|        linker won't look for a file to match a
 | |
|        wildcard.  The wildcard also means that it
 | |
|        doesn't matter which directory crtbegin.o
 | |
|        is in.  */
 | |
|     KEEP (*crtbegin.o(.ctors))
 | |
|     /* We don't want to include the .ctor section from
 | |
|        from the crtend.o file until after the sorted ctors.
 | |
|        The .ctor section from the crtend file contains the
 | |
|        end of ctors marker and it must be last */
 | |
|     KEEP (*(EXCLUDE_FILE (*crtend.o ) .ctors))
 | |
|     KEEP (*(SORT(.ctors.*)))
 | |
|     KEEP (*(.ctors))
 | |
|   }
 | |
|   .dtors          : {
 | |
|     KEEP (*crtbegin.o(.dtors))
 | |
|     KEEP (*(EXCLUDE_FILE (*crtend.o ) .dtors))
 | |
|     KEEP (*(SORT(.dtors.*)))
 | |
|     KEEP (*(.dtors))
 | |
|   }
 | |
|   .jcr            : { KEEP (*(.jcr)) }
 | |
|   .got            : { *(.got.plt) *(.got) }
 | |
|   _edata = .;
 | |
|   PROVIDE (edata = .);
 | |
|   .bss            : {
 | |
|    __bss_start = .;
 | |
|    *(.dynbss)
 | |
|    *(.bss .bss.* .gnu.linkonce.b.*)
 | |
|    *(COMMON)
 | |
|    /* Align here to ensure that the .bss section occupies space up to
 | |
|       _end.  Align after .bss to ensure correct alignment even if the
 | |
|       .bss section disappears because there are no input sections.  */
 | |
|    . = ALIGN(32 / 8);
 | |
|   . = ALIGN(32 / 8);
 | |
|   }
 | |
|    __bss_stop = .;
 | |
|   _end = .;
 | |
|   PROVIDE (end = .);
 | |
| 
 | |
|   STABS_DEBUG
 | |
| 
 | |
|   DWARF_DEBUG
 | |
| 
 | |
|   DISCARDS
 | |
| }
 |