2005-06-26 01:57:52 +04:00
/*
* kexec . c - kexec system call
* Copyright ( C ) 2002 - 2004 Eric Biederman < ebiederm @ xmission . com >
*
* This source code is licensed under the GNU General Public License ,
* Version 2. See the file COPYING for more details .
*/
2006-01-11 23:17:46 +03:00
# include <linux/capability.h>
2005-06-26 01:57:52 +04:00
# include <linux/mm.h>
# include <linux/file.h>
# include <linux/slab.h>
# include <linux/fs.h>
# include <linux/kexec.h>
# include <linux/spinlock.h>
# include <linux/list.h>
# include <linux/highmem.h>
# include <linux/syscalls.h>
# include <linux/reboot.h>
# include <linux/ioport.h>
2005-06-26 01:58:26 +04:00
# include <linux/hardirq.h>
2006-12-07 07:40:41 +03:00
# include <linux/elf.h>
# include <linux/elfcore.h>
2007-10-17 10:27:27 +04:00
# include <linux/utsrelease.h>
# include <linux/utsname.h>
# include <linux/numa.h>
2005-06-26 01:58:26 +04:00
2005-06-26 01:57:52 +04:00
# include <asm/page.h>
# include <asm/uaccess.h>
# include <asm/io.h>
# include <asm/system.h>
# include <asm/semaphore.h>
2007-10-17 10:27:27 +04:00
# include <asm/sections.h>
2005-06-26 01:57:52 +04:00
2006-01-10 07:51:41 +03:00
/* Per cpu memory for storing cpu states in case of system crash. */
note_buf_t * crash_notes ;
2007-10-17 10:27:27 +04:00
/* vmcoreinfo stuff */
unsigned char vmcoreinfo_data [ VMCOREINFO_BYTES ] ;
u32 vmcoreinfo_note [ VMCOREINFO_NOTE_SIZE / 4 ] ;
2007-10-17 10:27:28 +04:00
size_t vmcoreinfo_size ;
size_t vmcoreinfo_max_size = sizeof ( vmcoreinfo_data ) ;
2007-10-17 10:27:27 +04:00
2005-06-26 01:57:52 +04:00
/* Location of the reserved area for the crash kernel */
struct resource crashk_res = {
. name = " Crash kernel " ,
. start = 0 ,
. end = 0 ,
. flags = IORESOURCE_BUSY | IORESOURCE_MEM
} ;
2005-06-26 01:58:26 +04:00
int kexec_should_crash ( struct task_struct * p )
{
2007-10-19 10:39:52 +04:00
if ( in_interrupt ( ) | | ! p - > pid | | is_global_init ( p ) | | panic_on_oops )
2005-06-26 01:58:26 +04:00
return 1 ;
return 0 ;
}
2005-06-26 01:57:52 +04:00
/*
* When kexec transitions to the new kernel there is a one - to - one
* mapping between physical and virtual addresses . On processors
* where you can disable the MMU this is trivial , and easy . For
* others it is still a simple predictable page table to setup .
*
* In that environment kexec copies the new kernel to its final
* resting place . This means I can only support memory whose
* physical address can fit in an unsigned long . In particular
* addresses where ( pfn < < PAGE_SHIFT ) > ULONG_MAX cannot be handled .
* If the assembly stub has more restrictive requirements
* KEXEC_SOURCE_MEMORY_LIMIT and KEXEC_DEST_MEMORY_LIMIT can be
* defined more restrictively in < asm / kexec . h > .
*
* The code for the transition from the current kernel to the
* the new kernel is placed in the control_code_buffer , whose size
* is given by KEXEC_CONTROL_CODE_SIZE . In the best case only a single
* page of memory is necessary , but some architectures require more .
* Because this memory must be identity mapped in the transition from
* virtual to physical addresses it must live in the range
* 0 - TASK_SIZE , as only the user space mappings are arbitrarily
* modifiable .
*
* The assembly stub in the control code buffer is passed a linked list
* of descriptor pages detailing the source pages of the new kernel ,
* and the destination addresses of those source pages . As this data
* structure is not used in the context of the current OS , it must
* be self - contained .
*
* The code has been made to work with highmem pages and will use a
* destination page in its final resting place ( if it happens
* to allocate it ) . The end product of this is that most of the
* physical address space , and most of RAM can be used .
*
* Future directions include :
* - allocating a page table with the control code buffer identity
* mapped , to simplify machine_kexec and make kexec_on_panic more
* reliable .
*/
/*
* KIMAGE_NO_DEST is an impossible destination address . . . , for
* allocating pages whose destination address we do not care about .
*/
# define KIMAGE_NO_DEST (-1UL)
2005-06-26 01:58:28 +04:00
static int kimage_is_destination_range ( struct kimage * image ,
unsigned long start , unsigned long end ) ;
static struct page * kimage_alloc_page ( struct kimage * image ,
2005-10-21 11:22:03 +04:00
gfp_t gfp_mask ,
2005-06-26 01:58:28 +04:00
unsigned long dest ) ;
2005-06-26 01:57:52 +04:00
static int do_kimage_alloc ( struct kimage * * rimage , unsigned long entry ,
2005-06-26 01:58:28 +04:00
unsigned long nr_segments ,
struct kexec_segment __user * segments )
2005-06-26 01:57:52 +04:00
{
size_t segment_bytes ;
struct kimage * image ;
unsigned long i ;
int result ;
/* Allocate a controlling structure */
result = - ENOMEM ;
2006-12-07 07:38:51 +03:00
image = kzalloc ( sizeof ( * image ) , GFP_KERNEL ) ;
2005-06-26 01:58:28 +04:00
if ( ! image )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
image - > head = 0 ;
image - > entry = & image - > head ;
image - > last_entry = & image - > head ;
image - > control_page = ~ 0 ; /* By default this does not apply */
image - > start = entry ;
image - > type = KEXEC_TYPE_DEFAULT ;
/* Initialize the list of control pages */
INIT_LIST_HEAD ( & image - > control_pages ) ;
/* Initialize the list of destination pages */
INIT_LIST_HEAD ( & image - > dest_pages ) ;
/* Initialize the list of unuseable pages */
INIT_LIST_HEAD ( & image - > unuseable_pages ) ;
/* Read in the segments */
image - > nr_segments = nr_segments ;
segment_bytes = nr_segments * sizeof ( * segments ) ;
result = copy_from_user ( image - > segment , segments , segment_bytes ) ;
if ( result )
goto out ;
/*
* Verify we have good destination addresses . The caller is
* responsible for making certain we don ' t attempt to load
* the new image into invalid or reserved areas of RAM . This
* just verifies it is an address we can use .
*
* Since the kernel does everything in page size chunks ensure
* the destination addreses are page aligned . Too many
* special cases crop of when we don ' t do this . The most
* insidious is getting overlapping destination addresses
* simply because addresses are changed to page size
* granularity .
*/
result = - EADDRNOTAVAIL ;
for ( i = 0 ; i < nr_segments ; i + + ) {
unsigned long mstart , mend ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
mstart = image - > segment [ i ] . mem ;
mend = mstart + image - > segment [ i ] . memsz ;
if ( ( mstart & ~ PAGE_MASK ) | | ( mend & ~ PAGE_MASK ) )
goto out ;
if ( mend > = KEXEC_DESTINATION_MEMORY_LIMIT )
goto out ;
}
/* Verify our destination addresses do not overlap.
* If we alloed overlapping destination addresses
* through very weird things can happen with no
* easy explanation as one segment stops on another .
*/
result = - EINVAL ;
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < nr_segments ; i + + ) {
2005-06-26 01:57:52 +04:00
unsigned long mstart , mend ;
unsigned long j ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
mstart = image - > segment [ i ] . mem ;
mend = mstart + image - > segment [ i ] . memsz ;
2005-06-26 01:58:28 +04:00
for ( j = 0 ; j < i ; j + + ) {
2005-06-26 01:57:52 +04:00
unsigned long pstart , pend ;
pstart = image - > segment [ j ] . mem ;
pend = pstart + image - > segment [ j ] . memsz ;
/* Do the segments overlap ? */
if ( ( mend > pstart ) & & ( mstart < pend ) )
goto out ;
}
}
/* Ensure our buffer sizes are strictly less than
* our memory sizes . This should always be the case ,
* and it is easier to check up front than to be surprised
* later on .
*/
result = - EINVAL ;
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < nr_segments ; i + + ) {
2005-06-26 01:57:52 +04:00
if ( image - > segment [ i ] . bufsz > image - > segment [ i ] . memsz )
goto out ;
}
result = 0 ;
2005-06-26 01:58:28 +04:00
out :
if ( result = = 0 )
2005-06-26 01:57:52 +04:00
* rimage = image ;
2005-06-26 01:58:28 +04:00
else
2005-06-26 01:57:52 +04:00
kfree ( image ) ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
static int kimage_normal_alloc ( struct kimage * * rimage , unsigned long entry ,
2005-06-26 01:58:28 +04:00
unsigned long nr_segments ,
struct kexec_segment __user * segments )
2005-06-26 01:57:52 +04:00
{
int result ;
struct kimage * image ;
/* Allocate and initialize a controlling structure */
image = NULL ;
result = do_kimage_alloc ( & image , entry , nr_segments , segments ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
* rimage = image ;
/*
* Find a location for the control code buffer , and add it
* the vector of segments so that it ' s pages will also be
* counted as destination pages .
*/
result = - ENOMEM ;
image - > control_code_page = kimage_alloc_control_pages ( image ,
2005-06-26 01:58:28 +04:00
get_order ( KEXEC_CONTROL_CODE_SIZE ) ) ;
2005-06-26 01:57:52 +04:00
if ( ! image - > control_code_page ) {
printk ( KERN_ERR " Could not allocate control_code_buffer \n " ) ;
goto out ;
}
result = 0 ;
out :
2005-06-26 01:58:28 +04:00
if ( result = = 0 )
2005-06-26 01:57:52 +04:00
* rimage = image ;
2005-06-26 01:58:28 +04:00
else
2005-06-26 01:57:52 +04:00
kfree ( image ) ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
static int kimage_crash_alloc ( struct kimage * * rimage , unsigned long entry ,
2005-06-26 01:58:28 +04:00
unsigned long nr_segments ,
2005-06-28 09:29:33 +04:00
struct kexec_segment __user * segments )
2005-06-26 01:57:52 +04:00
{
int result ;
struct kimage * image ;
unsigned long i ;
image = NULL ;
/* Verify we have a valid entry point */
if ( ( entry < crashk_res . start ) | | ( entry > crashk_res . end ) ) {
result = - EADDRNOTAVAIL ;
goto out ;
}
/* Allocate and initialize a controlling structure */
result = do_kimage_alloc ( & image , entry , nr_segments , segments ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
/* Enable the special crash kernel control page
* allocation policy .
*/
image - > control_page = crashk_res . start ;
image - > type = KEXEC_TYPE_CRASH ;
/*
* Verify we have good destination addresses . Normally
* the caller is responsible for making certain we don ' t
* attempt to load the new image into invalid or reserved
* areas of RAM . But crash kernels are preloaded into a
* reserved area of ram . We must ensure the addresses
* are in the reserved area otherwise preloading the
* kernel could corrupt things .
*/
result = - EADDRNOTAVAIL ;
for ( i = 0 ; i < nr_segments ; i + + ) {
unsigned long mstart , mend ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
mstart = image - > segment [ i ] . mem ;
2005-06-26 01:57:55 +04:00
mend = mstart + image - > segment [ i ] . memsz - 1 ;
2005-06-26 01:57:52 +04:00
/* Ensure we are within the crash kernel limits */
if ( ( mstart < crashk_res . start ) | | ( mend > crashk_res . end ) )
goto out ;
}
/*
* Find a location for the control code buffer , and add
* the vector of segments so that it ' s pages will also be
* counted as destination pages .
*/
result = - ENOMEM ;
image - > control_code_page = kimage_alloc_control_pages ( image ,
2005-06-26 01:58:28 +04:00
get_order ( KEXEC_CONTROL_CODE_SIZE ) ) ;
2005-06-26 01:57:52 +04:00
if ( ! image - > control_code_page ) {
printk ( KERN_ERR " Could not allocate control_code_buffer \n " ) ;
goto out ;
}
result = 0 ;
2005-06-26 01:58:28 +04:00
out :
if ( result = = 0 )
2005-06-26 01:57:52 +04:00
* rimage = image ;
2005-06-26 01:58:28 +04:00
else
2005-06-26 01:57:52 +04:00
kfree ( image ) ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
2005-06-26 01:58:28 +04:00
static int kimage_is_destination_range ( struct kimage * image ,
unsigned long start ,
unsigned long end )
2005-06-26 01:57:52 +04:00
{
unsigned long i ;
for ( i = 0 ; i < image - > nr_segments ; i + + ) {
unsigned long mstart , mend ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
mstart = image - > segment [ i ] . mem ;
2005-06-26 01:58:28 +04:00
mend = mstart + image - > segment [ i ] . memsz ;
if ( ( end > mstart ) & & ( start < mend ) )
2005-06-26 01:57:52 +04:00
return 1 ;
}
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return 0 ;
}
2005-10-21 11:22:03 +04:00
static struct page * kimage_alloc_pages ( gfp_t gfp_mask , unsigned int order )
2005-06-26 01:57:52 +04:00
{
struct page * pages ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
pages = alloc_pages ( gfp_mask , order ) ;
if ( pages ) {
unsigned int count , i ;
pages - > mapping = NULL ;
[PATCH] mm: split page table lock
Christoph Lameter demonstrated very poor scalability on the SGI 512-way, with
a many-threaded application which concurrently initializes different parts of
a large anonymous area.
This patch corrects that, by using a separate spinlock per page table page, to
guard the page table entries in that page, instead of using the mm's single
page_table_lock. (But even then, page_table_lock is still used to guard page
table allocation, and anon_vma allocation.)
In this implementation, the spinlock is tucked inside the struct page of the
page table page: with a BUILD_BUG_ON in case it overflows - which it would in
the case of 32-bit PA-RISC with spinlock debugging enabled.
Splitting the lock is not quite for free: another cacheline access. Ideally,
I suppose we would use split ptlock only for multi-threaded processes on
multi-cpu machines; but deciding that dynamically would have its own costs.
So for now enable it by config, at some number of cpus - since the Kconfig
language doesn't support inequalities, let preprocessor compare that with
NR_CPUS. But I don't think it's worth being user-configurable: for good
testing of both split and unsplit configs, split now at 4 cpus, and perhaps
change that to 8 later.
There is a benefit even for singly threaded processes: kswapd can be attacking
one part of the mm while another part is busy faulting.
Signed-off-by: Hugh Dickins <hugh@veritas.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-10-30 04:16:40 +03:00
set_page_private ( pages , order ) ;
2005-06-26 01:57:52 +04:00
count = 1 < < order ;
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < count ; i + + )
2005-06-26 01:57:52 +04:00
SetPageReserved ( pages + i ) ;
}
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return pages ;
}
static void kimage_free_pages ( struct page * page )
{
unsigned int order , count , i ;
2005-06-26 01:58:28 +04:00
[PATCH] mm: split page table lock
Christoph Lameter demonstrated very poor scalability on the SGI 512-way, with
a many-threaded application which concurrently initializes different parts of
a large anonymous area.
This patch corrects that, by using a separate spinlock per page table page, to
guard the page table entries in that page, instead of using the mm's single
page_table_lock. (But even then, page_table_lock is still used to guard page
table allocation, and anon_vma allocation.)
In this implementation, the spinlock is tucked inside the struct page of the
page table page: with a BUILD_BUG_ON in case it overflows - which it would in
the case of 32-bit PA-RISC with spinlock debugging enabled.
Splitting the lock is not quite for free: another cacheline access. Ideally,
I suppose we would use split ptlock only for multi-threaded processes on
multi-cpu machines; but deciding that dynamically would have its own costs.
So for now enable it by config, at some number of cpus - since the Kconfig
language doesn't support inequalities, let preprocessor compare that with
NR_CPUS. But I don't think it's worth being user-configurable: for good
testing of both split and unsplit configs, split now at 4 cpus, and perhaps
change that to 8 later.
There is a benefit even for singly threaded processes: kswapd can be attacking
one part of the mm while another part is busy faulting.
Signed-off-by: Hugh Dickins <hugh@veritas.com>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-10-30 04:16:40 +03:00
order = page_private ( page ) ;
2005-06-26 01:57:52 +04:00
count = 1 < < order ;
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < count ; i + + )
2005-06-26 01:57:52 +04:00
ClearPageReserved ( page + i ) ;
__free_pages ( page , order ) ;
}
static void kimage_free_page_list ( struct list_head * list )
{
struct list_head * pos , * next ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
list_for_each_safe ( pos , next , list ) {
struct page * page ;
page = list_entry ( pos , struct page , lru ) ;
list_del ( & page - > lru ) ;
kimage_free_pages ( page ) ;
}
}
2005-06-26 01:58:28 +04:00
static struct page * kimage_alloc_normal_control_pages ( struct kimage * image ,
unsigned int order )
2005-06-26 01:57:52 +04:00
{
/* Control pages are special, they are the intermediaries
* that are needed while we copy the rest of the pages
* to their final resting place . As such they must
* not conflict with either the destination addresses
* or memory the kernel is already using .
*
* The only case where we really need more than one of
* these are for architectures where we cannot disable
* the MMU and must instead generate an identity mapped
* page table for all of the memory .
*
* At worst this runs in O ( N ) of the image size .
*/
struct list_head extra_pages ;
struct page * pages ;
unsigned int count ;
count = 1 < < order ;
INIT_LIST_HEAD ( & extra_pages ) ;
/* Loop while I can allocate a page and the page allocated
* is a destination page .
*/
do {
unsigned long pfn , epfn , addr , eaddr ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
pages = kimage_alloc_pages ( GFP_KERNEL , order ) ;
if ( ! pages )
break ;
pfn = page_to_pfn ( pages ) ;
epfn = pfn + count ;
addr = pfn < < PAGE_SHIFT ;
eaddr = epfn < < PAGE_SHIFT ;
if ( ( epfn > = ( KEXEC_CONTROL_MEMORY_LIMIT > > PAGE_SHIFT ) ) | |
2005-06-26 01:58:28 +04:00
kimage_is_destination_range ( image , addr , eaddr ) ) {
2005-06-26 01:57:52 +04:00
list_add ( & pages - > lru , & extra_pages ) ;
pages = NULL ;
}
2005-06-26 01:58:28 +04:00
} while ( ! pages ) ;
2005-06-26 01:57:52 +04:00
if ( pages ) {
/* Remember the allocated page... */
list_add ( & pages - > lru , & image - > control_pages ) ;
/* Because the page is already in it's destination
* location we will never allocate another page at
* that address . Therefore kimage_alloc_pages
* will not return it ( again ) and we don ' t need
* to give it an entry in image - > segment [ ] .
*/
}
/* Deal with the destination pages I have inadvertently allocated.
*
* Ideally I would convert multi - page allocations into single
* page allocations , and add everyting to image - > dest_pages .
*
* For now it is simpler to just free the pages .
*/
kimage_free_page_list ( & extra_pages ) ;
2005-06-26 01:58:28 +04:00
return pages ;
2005-06-26 01:57:52 +04:00
}
2005-06-26 01:58:28 +04:00
static struct page * kimage_alloc_crash_control_pages ( struct kimage * image ,
unsigned int order )
2005-06-26 01:57:52 +04:00
{
/* Control pages are special, they are the intermediaries
* that are needed while we copy the rest of the pages
* to their final resting place . As such they must
* not conflict with either the destination addresses
* or memory the kernel is already using .
*
* Control pages are also the only pags we must allocate
* when loading a crash kernel . All of the other pages
* are specified by the segments and we just memcpy
* into them directly .
*
* The only case where we really need more than one of
* these are for architectures where we cannot disable
* the MMU and must instead generate an identity mapped
* page table for all of the memory .
*
* Given the low demand this implements a very simple
* allocator that finds the first hole of the appropriate
* size in the reserved memory region , and allocates all
* of the memory up to and including the hole .
*/
unsigned long hole_start , hole_end , size ;
struct page * pages ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
pages = NULL ;
size = ( 1 < < order ) < < PAGE_SHIFT ;
hole_start = ( image - > control_page + ( size - 1 ) ) & ~ ( size - 1 ) ;
hole_end = hole_start + size - 1 ;
2005-06-26 01:58:28 +04:00
while ( hole_end < = crashk_res . end ) {
2005-06-26 01:57:52 +04:00
unsigned long i ;
2005-06-26 01:58:28 +04:00
if ( hole_end > KEXEC_CONTROL_MEMORY_LIMIT )
2005-06-26 01:57:52 +04:00
break ;
2005-06-26 01:58:28 +04:00
if ( hole_end > crashk_res . end )
2005-06-26 01:57:52 +04:00
break ;
/* See if I overlap any of the segments */
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < image - > nr_segments ; i + + ) {
2005-06-26 01:57:52 +04:00
unsigned long mstart , mend ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
mstart = image - > segment [ i ] . mem ;
mend = mstart + image - > segment [ i ] . memsz - 1 ;
if ( ( hole_end > = mstart ) & & ( hole_start < = mend ) ) {
/* Advance the hole to the end of the segment */
hole_start = ( mend + ( size - 1 ) ) & ~ ( size - 1 ) ;
hole_end = hole_start + size - 1 ;
break ;
}
}
/* If I don't overlap any segments I have found my hole! */
if ( i = = image - > nr_segments ) {
pages = pfn_to_page ( hole_start > > PAGE_SHIFT ) ;
break ;
}
}
2005-06-26 01:58:28 +04:00
if ( pages )
2005-06-26 01:57:52 +04:00
image - > control_page = hole_end ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return pages ;
}
2005-06-26 01:58:28 +04:00
struct page * kimage_alloc_control_pages ( struct kimage * image ,
unsigned int order )
2005-06-26 01:57:52 +04:00
{
struct page * pages = NULL ;
2005-06-26 01:58:28 +04:00
switch ( image - > type ) {
2005-06-26 01:57:52 +04:00
case KEXEC_TYPE_DEFAULT :
pages = kimage_alloc_normal_control_pages ( image , order ) ;
break ;
case KEXEC_TYPE_CRASH :
pages = kimage_alloc_crash_control_pages ( image , order ) ;
break ;
}
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return pages ;
}
static int kimage_add_entry ( struct kimage * image , kimage_entry_t entry )
{
2005-06-26 01:58:28 +04:00
if ( * image - > entry ! = 0 )
2005-06-26 01:57:52 +04:00
image - > entry + + ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
if ( image - > entry = = image - > last_entry ) {
kimage_entry_t * ind_page ;
struct page * page ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
page = kimage_alloc_page ( image , GFP_KERNEL , KIMAGE_NO_DEST ) ;
2005-06-26 01:58:28 +04:00
if ( ! page )
2005-06-26 01:57:52 +04:00
return - ENOMEM ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
ind_page = page_address ( page ) ;
* image - > entry = virt_to_phys ( ind_page ) | IND_INDIRECTION ;
image - > entry = ind_page ;
2005-06-26 01:58:28 +04:00
image - > last_entry = ind_page +
( ( PAGE_SIZE / sizeof ( kimage_entry_t ) ) - 1 ) ;
2005-06-26 01:57:52 +04:00
}
* image - > entry = entry ;
image - > entry + + ;
* image - > entry = 0 ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return 0 ;
}
2005-06-26 01:58:28 +04:00
static int kimage_set_destination ( struct kimage * image ,
unsigned long destination )
2005-06-26 01:57:52 +04:00
{
int result ;
destination & = PAGE_MASK ;
result = kimage_add_entry ( image , destination | IND_DESTINATION ) ;
2005-06-26 01:58:28 +04:00
if ( result = = 0 )
2005-06-26 01:57:52 +04:00
image - > destination = destination ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
static int kimage_add_page ( struct kimage * image , unsigned long page )
{
int result ;
page & = PAGE_MASK ;
result = kimage_add_entry ( image , page | IND_SOURCE ) ;
2005-06-26 01:58:28 +04:00
if ( result = = 0 )
2005-06-26 01:57:52 +04:00
image - > destination + = PAGE_SIZE ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
static void kimage_free_extra_pages ( struct kimage * image )
{
/* Walk through and free any extra destination pages I may have */
kimage_free_page_list ( & image - > dest_pages ) ;
/* Walk through and free any unuseable pages I have cached */
kimage_free_page_list ( & image - > unuseable_pages ) ;
}
static int kimage_terminate ( struct kimage * image )
{
2005-06-26 01:58:28 +04:00
if ( * image - > entry ! = 0 )
2005-06-26 01:57:52 +04:00
image - > entry + + ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
* image - > entry = IND_DONE ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return 0 ;
}
# define for_each_kimage_entry(image, ptr, entry) \
for ( ptr = & image - > head ; ( entry = * ptr ) & & ! ( entry & IND_DONE ) ; \
ptr = ( entry & IND_INDIRECTION ) ? \
phys_to_virt ( ( entry & PAGE_MASK ) ) : ptr + 1 )
static void kimage_free_entry ( kimage_entry_t entry )
{
struct page * page ;
page = pfn_to_page ( entry > > PAGE_SHIFT ) ;
kimage_free_pages ( page ) ;
}
static void kimage_free ( struct kimage * image )
{
kimage_entry_t * ptr , entry ;
kimage_entry_t ind = 0 ;
if ( ! image )
return ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
kimage_free_extra_pages ( image ) ;
for_each_kimage_entry ( image , ptr , entry ) {
if ( entry & IND_INDIRECTION ) {
/* Free the previous indirection page */
2005-06-26 01:58:28 +04:00
if ( ind & IND_INDIRECTION )
2005-06-26 01:57:52 +04:00
kimage_free_entry ( ind ) ;
/* Save this indirection page until we are
* done with it .
*/
ind = entry ;
}
2005-06-26 01:58:28 +04:00
else if ( entry & IND_SOURCE )
2005-06-26 01:57:52 +04:00
kimage_free_entry ( entry ) ;
}
/* Free the final indirection page */
2005-06-26 01:58:28 +04:00
if ( ind & IND_INDIRECTION )
2005-06-26 01:57:52 +04:00
kimage_free_entry ( ind ) ;
/* Handle any machine specific cleanup */
machine_kexec_cleanup ( image ) ;
/* Free the kexec control pages... */
kimage_free_page_list ( & image - > control_pages ) ;
kfree ( image ) ;
}
2005-06-26 01:58:28 +04:00
static kimage_entry_t * kimage_dst_used ( struct kimage * image ,
unsigned long page )
2005-06-26 01:57:52 +04:00
{
kimage_entry_t * ptr , entry ;
unsigned long destination = 0 ;
for_each_kimage_entry ( image , ptr , entry ) {
2005-06-26 01:58:28 +04:00
if ( entry & IND_DESTINATION )
2005-06-26 01:57:52 +04:00
destination = entry & PAGE_MASK ;
else if ( entry & IND_SOURCE ) {
2005-06-26 01:58:28 +04:00
if ( page = = destination )
2005-06-26 01:57:52 +04:00
return ptr ;
destination + = PAGE_SIZE ;
}
}
2005-06-26 01:58:28 +04:00
2005-06-28 09:29:33 +04:00
return NULL ;
2005-06-26 01:57:52 +04:00
}
2005-06-26 01:58:28 +04:00
static struct page * kimage_alloc_page ( struct kimage * image ,
2005-10-21 11:22:03 +04:00
gfp_t gfp_mask ,
2005-06-26 01:58:28 +04:00
unsigned long destination )
2005-06-26 01:57:52 +04:00
{
/*
* Here we implement safeguards to ensure that a source page
* is not copied to its destination page before the data on
* the destination page is no longer useful .
*
* To do this we maintain the invariant that a source page is
* either its own destination page , or it is not a
* destination page at all .
*
* That is slightly stronger than required , but the proof
* that no problems will not occur is trivial , and the
* implementation is simply to verify .
*
* When allocating all pages normally this algorithm will run
* in O ( N ) time , but in the worst case it will run in O ( N ^ 2 )
* time . If the runtime is a problem the data structures can
* be fixed .
*/
struct page * page ;
unsigned long addr ;
/*
* Walk through the list of destination pages , and see if I
* have a match .
*/
list_for_each_entry ( page , & image - > dest_pages , lru ) {
addr = page_to_pfn ( page ) < < PAGE_SHIFT ;
if ( addr = = destination ) {
list_del ( & page - > lru ) ;
return page ;
}
}
page = NULL ;
while ( 1 ) {
kimage_entry_t * old ;
/* Allocate a page, if we run out of memory give up */
page = kimage_alloc_pages ( gfp_mask , 0 ) ;
2005-06-26 01:58:28 +04:00
if ( ! page )
2005-06-28 09:29:33 +04:00
return NULL ;
2005-06-26 01:57:52 +04:00
/* If the page cannot be used file it away */
2005-06-26 01:58:28 +04:00
if ( page_to_pfn ( page ) >
( KEXEC_SOURCE_MEMORY_LIMIT > > PAGE_SHIFT ) ) {
2005-06-26 01:57:52 +04:00
list_add ( & page - > lru , & image - > unuseable_pages ) ;
continue ;
}
addr = page_to_pfn ( page ) < < PAGE_SHIFT ;
/* If it is the destination page we want use it */
if ( addr = = destination )
break ;
/* If the page is not a destination page use it */
2005-06-26 01:58:28 +04:00
if ( ! kimage_is_destination_range ( image , addr ,
addr + PAGE_SIZE ) )
2005-06-26 01:57:52 +04:00
break ;
/*
* I know that the page is someones destination page .
* See if there is already a source page for this
* destination page . And if so swap the source pages .
*/
old = kimage_dst_used ( image , addr ) ;
if ( old ) {
/* If so move it */
unsigned long old_addr ;
struct page * old_page ;
old_addr = * old & PAGE_MASK ;
old_page = pfn_to_page ( old_addr > > PAGE_SHIFT ) ;
copy_highpage ( page , old_page ) ;
* old = addr | ( * old & ~ PAGE_MASK ) ;
/* The old page I have found cannot be a
* destination page , so return it .
*/
addr = old_addr ;
page = old_page ;
break ;
}
else {
/* Place the page on the destination list I
* will use it later .
*/
list_add ( & page - > lru , & image - > dest_pages ) ;
}
}
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return page ;
}
static int kimage_load_normal_segment ( struct kimage * image ,
2005-06-26 01:58:28 +04:00
struct kexec_segment * segment )
2005-06-26 01:57:52 +04:00
{
unsigned long maddr ;
unsigned long ubytes , mbytes ;
int result ;
2005-06-28 09:29:33 +04:00
unsigned char __user * buf ;
2005-06-26 01:57:52 +04:00
result = 0 ;
buf = segment - > buf ;
ubytes = segment - > bufsz ;
mbytes = segment - > memsz ;
maddr = segment - > mem ;
result = kimage_set_destination ( image , maddr ) ;
2005-06-26 01:58:28 +04:00
if ( result < 0 )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
while ( mbytes ) {
2005-06-26 01:57:52 +04:00
struct page * page ;
char * ptr ;
size_t uchunk , mchunk ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
page = kimage_alloc_page ( image , GFP_HIGHUSER , maddr ) ;
2007-10-18 14:07:05 +04:00
if ( ! page ) {
2005-06-26 01:57:52 +04:00
result = - ENOMEM ;
goto out ;
}
2005-06-26 01:58:28 +04:00
result = kimage_add_page ( image , page_to_pfn ( page )
< < PAGE_SHIFT ) ;
if ( result < 0 )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
ptr = kmap ( page ) ;
/* Start with a clear page */
memset ( ptr , 0 , PAGE_SIZE ) ;
ptr + = maddr & ~ PAGE_MASK ;
mchunk = PAGE_SIZE - ( maddr & ~ PAGE_MASK ) ;
2005-06-26 01:58:28 +04:00
if ( mchunk > mbytes )
2005-06-26 01:57:52 +04:00
mchunk = mbytes ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
uchunk = mchunk ;
2005-06-26 01:58:28 +04:00
if ( uchunk > ubytes )
2005-06-26 01:57:52 +04:00
uchunk = ubytes ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
result = copy_from_user ( ptr , buf , uchunk ) ;
kunmap ( page ) ;
if ( result ) {
result = ( result < 0 ) ? result : - EIO ;
goto out ;
}
ubytes - = uchunk ;
maddr + = mchunk ;
buf + = mchunk ;
mbytes - = mchunk ;
}
2005-06-26 01:58:28 +04:00
out :
2005-06-26 01:57:52 +04:00
return result ;
}
static int kimage_load_crash_segment ( struct kimage * image ,
2005-06-26 01:58:28 +04:00
struct kexec_segment * segment )
2005-06-26 01:57:52 +04:00
{
/* For crash dumps kernels we simply copy the data from
* user space to it ' s destination .
* We do things a page at a time for the sake of kmap .
*/
unsigned long maddr ;
unsigned long ubytes , mbytes ;
int result ;
2005-06-28 09:29:33 +04:00
unsigned char __user * buf ;
2005-06-26 01:57:52 +04:00
result = 0 ;
buf = segment - > buf ;
ubytes = segment - > bufsz ;
mbytes = segment - > memsz ;
maddr = segment - > mem ;
2005-06-26 01:58:28 +04:00
while ( mbytes ) {
2005-06-26 01:57:52 +04:00
struct page * page ;
char * ptr ;
size_t uchunk , mchunk ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
page = pfn_to_page ( maddr > > PAGE_SHIFT ) ;
2007-10-18 14:07:05 +04:00
if ( ! page ) {
2005-06-26 01:57:52 +04:00
result = - ENOMEM ;
goto out ;
}
ptr = kmap ( page ) ;
ptr + = maddr & ~ PAGE_MASK ;
mchunk = PAGE_SIZE - ( maddr & ~ PAGE_MASK ) ;
2005-06-26 01:58:28 +04:00
if ( mchunk > mbytes )
2005-06-26 01:57:52 +04:00
mchunk = mbytes ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
uchunk = mchunk ;
if ( uchunk > ubytes ) {
uchunk = ubytes ;
/* Zero the trailing part of the page */
memset ( ptr + uchunk , 0 , mchunk - uchunk ) ;
}
result = copy_from_user ( ptr , buf , uchunk ) ;
2006-12-07 20:51:35 +03:00
kexec_flush_icache_page ( page ) ;
2005-06-26 01:57:52 +04:00
kunmap ( page ) ;
if ( result ) {
result = ( result < 0 ) ? result : - EIO ;
goto out ;
}
ubytes - = uchunk ;
maddr + = mchunk ;
buf + = mchunk ;
mbytes - = mchunk ;
}
2005-06-26 01:58:28 +04:00
out :
2005-06-26 01:57:52 +04:00
return result ;
}
static int kimage_load_segment ( struct kimage * image ,
2005-06-26 01:58:28 +04:00
struct kexec_segment * segment )
2005-06-26 01:57:52 +04:00
{
int result = - ENOMEM ;
2005-06-26 01:58:28 +04:00
switch ( image - > type ) {
2005-06-26 01:57:52 +04:00
case KEXEC_TYPE_DEFAULT :
result = kimage_load_normal_segment ( image , segment ) ;
break ;
case KEXEC_TYPE_CRASH :
result = kimage_load_crash_segment ( image , segment ) ;
break ;
}
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
/*
* Exec Kernel system call : for obvious reasons only root may call it .
*
* This call breaks up into three pieces .
* - A generic part which loads the new kernel from the current
* address space , and very carefully places the data in the
* allocated pages .
*
* - A generic part that interacts with the kernel and tells all of
* the devices to shut down . Preventing on - going dmas , and placing
* the devices in a consistent state so a later kernel can
* reinitialize them .
*
* - A machine specific part that includes the syscall number
* and the copies the image to it ' s final destination . And
* jumps into the image at entry .
*
* kexec does not sync , or unmount filesystems so if you need
* that to happen you need to do that yourself .
*/
2006-06-23 13:05:07 +04:00
struct kimage * kexec_image ;
struct kimage * kexec_crash_image ;
2005-06-26 01:57:52 +04:00
/*
* A home grown binary mutex .
* Nothing can wait so this mutex is safe to use
* in interrupt context : )
*/
2006-06-23 13:05:07 +04:00
static int kexec_lock ;
2005-06-26 01:57:52 +04:00
2005-06-26 01:58:28 +04:00
asmlinkage long sys_kexec_load ( unsigned long entry , unsigned long nr_segments ,
struct kexec_segment __user * segments ,
unsigned long flags )
2005-06-26 01:57:52 +04:00
{
struct kimage * * dest_image , * image ;
int locked ;
int result ;
/* We only trust the superuser with rebooting the system. */
if ( ! capable ( CAP_SYS_BOOT ) )
return - EPERM ;
/*
* Verify we have a legal set of flags
* This leaves us room for future extensions .
*/
if ( ( flags & KEXEC_FLAGS ) ! = ( flags & ~ KEXEC_ARCH_MASK ) )
return - EINVAL ;
/* Verify we are on the appropriate architecture */
if ( ( ( flags & KEXEC_ARCH_MASK ) ! = KEXEC_ARCH ) & &
( ( flags & KEXEC_ARCH_MASK ) ! = KEXEC_ARCH_DEFAULT ) )
return - EINVAL ;
/* Put an artificial cap on the number
* of segments passed to kexec_load .
*/
if ( nr_segments > KEXEC_SEGMENT_MAX )
return - EINVAL ;
image = NULL ;
result = 0 ;
/* Because we write directly to the reserved memory
* region when loading crash kernels we need a mutex here to
* prevent multiple crash kernels from attempting to load
* simultaneously , and to prevent a crash kernel from loading
* over the top of a in use crash kernel .
*
* KISS : always take the mutex .
*/
locked = xchg ( & kexec_lock , 1 ) ;
2005-06-26 01:58:28 +04:00
if ( locked )
2005-06-26 01:57:52 +04:00
return - EBUSY ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
dest_image = & kexec_image ;
2005-06-26 01:58:28 +04:00
if ( flags & KEXEC_ON_CRASH )
2005-06-26 01:57:52 +04:00
dest_image = & kexec_crash_image ;
if ( nr_segments > 0 ) {
unsigned long i ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
/* Loading another kernel to reboot into */
2005-06-26 01:58:28 +04:00
if ( ( flags & KEXEC_ON_CRASH ) = = 0 )
result = kimage_normal_alloc ( & image , entry ,
nr_segments , segments ) ;
2005-06-26 01:57:52 +04:00
/* Loading another kernel to switch to if this one crashes */
else if ( flags & KEXEC_ON_CRASH ) {
/* Free any current crash dump kernel before
* we corrupt it .
*/
kimage_free ( xchg ( & kexec_crash_image , NULL ) ) ;
2005-06-26 01:58:28 +04:00
result = kimage_crash_alloc ( & image , entry ,
nr_segments , segments ) ;
2005-06-26 01:57:52 +04:00
}
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
result = machine_kexec_prepare ( image ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
2005-06-26 01:58:28 +04:00
for ( i = 0 ; i < nr_segments ; i + + ) {
2005-06-26 01:57:52 +04:00
result = kimage_load_segment ( image , & image - > segment [ i ] ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
}
result = kimage_terminate ( image ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
goto out ;
}
/* Install the new kernel, and Uninstall the old */
image = xchg ( dest_image , image ) ;
2005-06-26 01:58:28 +04:00
out :
2006-09-29 13:00:39 +04:00
locked = xchg ( & kexec_lock , 0 ) ; /* Release the mutex */
BUG_ON ( ! locked ) ;
2005-06-26 01:57:52 +04:00
kimage_free ( image ) ;
2005-06-26 01:58:28 +04:00
2005-06-26 01:57:52 +04:00
return result ;
}
# ifdef CONFIG_COMPAT
asmlinkage long compat_sys_kexec_load ( unsigned long entry ,
2005-06-26 01:58:28 +04:00
unsigned long nr_segments ,
struct compat_kexec_segment __user * segments ,
unsigned long flags )
2005-06-26 01:57:52 +04:00
{
struct compat_kexec_segment in ;
struct kexec_segment out , __user * ksegments ;
unsigned long i , result ;
/* Don't allow clients that don't understand the native
* architecture to do anything .
*/
2005-06-26 01:58:28 +04:00
if ( ( flags & KEXEC_ARCH_MASK ) = = KEXEC_ARCH_DEFAULT )
2005-06-26 01:57:52 +04:00
return - EINVAL ;
2005-06-26 01:58:28 +04:00
if ( nr_segments > KEXEC_SEGMENT_MAX )
2005-06-26 01:57:52 +04:00
return - EINVAL ;
ksegments = compat_alloc_user_space ( nr_segments * sizeof ( out ) ) ;
for ( i = 0 ; i < nr_segments ; i + + ) {
result = copy_from_user ( & in , & segments [ i ] , sizeof ( in ) ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
return - EFAULT ;
out . buf = compat_ptr ( in . buf ) ;
out . bufsz = in . bufsz ;
out . mem = in . mem ;
out . memsz = in . memsz ;
result = copy_to_user ( & ksegments [ i ] , & out , sizeof ( out ) ) ;
2005-06-26 01:58:28 +04:00
if ( result )
2005-06-26 01:57:52 +04:00
return - EFAULT ;
}
return sys_kexec_load ( entry , nr_segments , ksegments , flags ) ;
}
# endif
2005-06-26 01:58:26 +04:00
void crash_kexec ( struct pt_regs * regs )
2005-06-26 01:57:52 +04:00
{
int locked ;
/* Take the kexec_lock here to prevent sys_kexec_load
* running on one cpu from replacing the crash kernel
* we are using after a panic on a different cpu .
*
* If the crash kernel was not located in a fixed area
* of memory the xchg ( & kexec_crash_image ) would be
* sufficient . But since I reuse the memory . . .
*/
locked = xchg ( & kexec_lock , 1 ) ;
if ( ! locked ) {
2006-06-24 02:29:34 +04:00
if ( kexec_crash_image ) {
2006-01-10 07:51:44 +03:00
struct pt_regs fixed_regs ;
crash_setup_regs ( & fixed_regs , regs ) ;
2007-10-17 10:27:27 +04:00
crash_save_vmcoreinfo ( ) ;
2006-01-10 07:51:44 +03:00
machine_crash_shutdown ( & fixed_regs ) ;
2006-06-24 02:29:34 +04:00
machine_kexec ( kexec_crash_image ) ;
2005-06-26 01:57:52 +04:00
}
2006-09-29 13:00:39 +04:00
locked = xchg ( & kexec_lock , 0 ) ;
BUG_ON ( ! locked ) ;
2005-06-26 01:57:52 +04:00
}
}
2006-01-10 07:51:41 +03:00
2006-12-07 07:40:41 +03:00
static u32 * append_elf_note ( u32 * buf , char * name , unsigned type , void * data ,
size_t data_len )
{
struct elf_note note ;
note . n_namesz = strlen ( name ) + 1 ;
note . n_descsz = data_len ;
note . n_type = type ;
memcpy ( buf , & note , sizeof ( note ) ) ;
buf + = ( sizeof ( note ) + 3 ) / 4 ;
memcpy ( buf , name , note . n_namesz ) ;
buf + = ( note . n_namesz + 3 ) / 4 ;
memcpy ( buf , data , note . n_descsz ) ;
buf + = ( note . n_descsz + 3 ) / 4 ;
return buf ;
}
static void final_note ( u32 * buf )
{
struct elf_note note ;
note . n_namesz = 0 ;
note . n_descsz = 0 ;
note . n_type = 0 ;
memcpy ( buf , & note , sizeof ( note ) ) ;
}
void crash_save_cpu ( struct pt_regs * regs , int cpu )
{
struct elf_prstatus prstatus ;
u32 * buf ;
if ( ( cpu < 0 ) | | ( cpu > = NR_CPUS ) )
return ;
/* Using ELF notes here is opportunistic.
* I need a well defined structure format
* for the data I pass , and I need tags
* on the data to indicate what information I have
* squirrelled away . ELF notes happen to provide
* all of that , so there is no need to invent something new .
*/
buf = ( u32 * ) per_cpu_ptr ( crash_notes , cpu ) ;
if ( ! buf )
return ;
memset ( & prstatus , 0 , sizeof ( prstatus ) ) ;
prstatus . pr_pid = current - > pid ;
elf_core_copy_regs ( & prstatus . pr_reg , regs ) ;
2007-05-08 11:28:22 +04:00
buf = append_elf_note ( buf , KEXEC_CORE_NOTE_NAME , NT_PRSTATUS ,
& prstatus , sizeof ( prstatus ) ) ;
2006-12-07 07:40:41 +03:00
final_note ( buf ) ;
}
2006-01-10 07:51:41 +03:00
static int __init crash_notes_memory_init ( void )
{
/* Allocate memory for saving cpu registers. */
crash_notes = alloc_percpu ( note_buf_t ) ;
if ( ! crash_notes ) {
printk ( " Kexec: Memory allocation for saving cpu register "
" states failed \n " ) ;
return - ENOMEM ;
}
return 0 ;
}
module_init ( crash_notes_memory_init )
2007-10-17 10:27:27 +04:00
void crash_save_vmcoreinfo ( void )
{
u32 * buf ;
if ( ! vmcoreinfo_size )
return ;
2007-10-17 10:27:28 +04:00
vmcoreinfo_append_str ( " CRASHTIME=%ld " , get_seconds ( ) ) ;
2007-10-17 10:27:27 +04:00
buf = ( u32 * ) vmcoreinfo_note ;
buf = append_elf_note ( buf , VMCOREINFO_NOTE_NAME , 0 , vmcoreinfo_data ,
vmcoreinfo_size ) ;
final_note ( buf ) ;
}
void vmcoreinfo_append_str ( const char * fmt , . . . )
{
va_list args ;
char buf [ 0x50 ] ;
int r ;
va_start ( args , fmt ) ;
r = vsnprintf ( buf , sizeof ( buf ) , fmt , args ) ;
va_end ( args ) ;
if ( r + vmcoreinfo_size > vmcoreinfo_max_size )
r = vmcoreinfo_max_size - vmcoreinfo_size ;
memcpy ( & vmcoreinfo_data [ vmcoreinfo_size ] , buf , r ) ;
vmcoreinfo_size + = r ;
}
/*
* provide an empty default implementation here - - architecture
* code may override this
*/
void __attribute__ ( ( weak ) ) arch_crash_save_vmcoreinfo ( void )
{ }
unsigned long __attribute__ ( ( weak ) ) paddr_vmcoreinfo_note ( void )
{
return __pa ( ( unsigned long ) ( char * ) & vmcoreinfo_note ) ;
}
static int __init crash_save_vmcoreinfo_init ( void )
{
2007-10-17 10:27:28 +04:00
vmcoreinfo_append_str ( " OSRELEASE=%s \n " , init_uts_ns . name . release ) ;
vmcoreinfo_append_str ( " PAGESIZE=%ld \n " , PAGE_SIZE ) ;
2007-10-17 10:27:27 +04:00
2007-10-17 10:27:30 +04:00
VMCOREINFO_SYMBOL ( init_uts_ns ) ;
VMCOREINFO_SYMBOL ( node_online_map ) ;
VMCOREINFO_SYMBOL ( swapper_pg_dir ) ;
VMCOREINFO_SYMBOL ( _stext ) ;
2007-10-17 10:27:27 +04:00
# ifndef CONFIG_NEED_MULTIPLE_NODES
2007-10-17 10:27:30 +04:00
VMCOREINFO_SYMBOL ( mem_map ) ;
VMCOREINFO_SYMBOL ( contig_page_data ) ;
2007-10-17 10:27:27 +04:00
# endif
# ifdef CONFIG_SPARSEMEM
2007-10-17 10:27:30 +04:00
VMCOREINFO_SYMBOL ( mem_section ) ;
VMCOREINFO_LENGTH ( mem_section , NR_SECTION_ROOTS ) ;
VMCOREINFO_SIZE ( mem_section ) ;
VMCOREINFO_OFFSET ( mem_section , section_mem_map ) ;
2007-10-17 10:27:27 +04:00
# endif
2007-10-17 10:27:30 +04:00
VMCOREINFO_SIZE ( page ) ;
VMCOREINFO_SIZE ( pglist_data ) ;
VMCOREINFO_SIZE ( zone ) ;
VMCOREINFO_SIZE ( free_area ) ;
VMCOREINFO_SIZE ( list_head ) ;
VMCOREINFO_TYPEDEF_SIZE ( nodemask_t ) ;
VMCOREINFO_OFFSET ( page , flags ) ;
VMCOREINFO_OFFSET ( page , _count ) ;
VMCOREINFO_OFFSET ( page , mapping ) ;
VMCOREINFO_OFFSET ( page , lru ) ;
VMCOREINFO_OFFSET ( pglist_data , node_zones ) ;
VMCOREINFO_OFFSET ( pglist_data , nr_zones ) ;
2007-10-17 10:27:27 +04:00
# ifdef CONFIG_FLAT_NODE_MEM_MAP
2007-10-17 10:27:30 +04:00
VMCOREINFO_OFFSET ( pglist_data , node_mem_map ) ;
2007-10-17 10:27:27 +04:00
# endif
2007-10-17 10:27:30 +04:00
VMCOREINFO_OFFSET ( pglist_data , node_start_pfn ) ;
VMCOREINFO_OFFSET ( pglist_data , node_spanned_pages ) ;
VMCOREINFO_OFFSET ( pglist_data , node_id ) ;
VMCOREINFO_OFFSET ( zone , free_area ) ;
VMCOREINFO_OFFSET ( zone , vm_stat ) ;
VMCOREINFO_OFFSET ( zone , spanned_pages ) ;
VMCOREINFO_OFFSET ( free_area , free_list ) ;
VMCOREINFO_OFFSET ( list_head , next ) ;
VMCOREINFO_OFFSET ( list_head , prev ) ;
VMCOREINFO_LENGTH ( zone . free_area , MAX_ORDER ) ;
VMCOREINFO_NUMBER ( NR_FREE_PAGES ) ;
2007-10-17 10:27:27 +04:00
arch_crash_save_vmcoreinfo ( ) ;
return 0 ;
}
module_init ( crash_save_vmcoreinfo_init )