2006-10-18 17:14:55 +04:00
/*
* machine_kexec . c for kexec
* Created by < nschichan @ corp . free . fr > on Thu Oct 12 15 : 15 : 06 2006
*
* This source code is licensed under the GNU General Public License ,
* Version 2. See the file COPYING for more details .
*/
# include <linux/kexec.h>
# include <linux/mm.h>
# include <linux/delay.h>
# include <asm/cacheflush.h>
# include <asm/page.h>
2007-02-17 22:11:19 +03:00
extern const unsigned char relocate_new_kernel [ ] ;
2007-07-31 18:16:32 +04:00
extern const size_t relocate_new_kernel_size ;
2006-10-18 17:14:55 +04:00
extern unsigned long kexec_start_address ;
extern unsigned long kexec_indirection_page ;
int
machine_kexec_prepare ( struct kimage * kimage )
{
return 0 ;
}
void
machine_kexec_cleanup ( struct kimage * kimage )
{
}
void
machine_shutdown ( void )
{
}
void
machine_crash_shutdown ( struct pt_regs * regs )
{
}
2007-07-31 18:16:32 +04:00
typedef void ( * noretfun_t ) ( void ) __attribute__ ( ( noreturn ) ) ;
2006-10-18 17:14:55 +04:00
void
machine_kexec ( struct kimage * image )
{
unsigned long reboot_code_buffer ;
unsigned long entry ;
unsigned long * ptr ;
reboot_code_buffer =
( unsigned long ) page_address ( image - > control_code_page ) ;
kexec_start_address = image - > start ;
2007-07-31 18:16:32 +04:00
kexec_indirection_page =
( unsigned long ) phys_to_virt ( image - > head & PAGE_MASK ) ;
2006-10-18 17:14:55 +04:00
memcpy ( ( void * ) reboot_code_buffer , relocate_new_kernel ,
relocate_new_kernel_size ) ;
/*
* The generic kexec code builds a page list with physical
* addresses . they are directly accessible through KSEG0 ( or
* CKSEG0 or XPHYS if on 64 bit system ) , hence the
* pys_to_virt ( ) call .
*/
for ( ptr = & image - > head ; ( entry = * ptr ) & & ! ( entry & IND_DONE ) ;
ptr = ( entry & IND_INDIRECTION ) ?
phys_to_virt ( entry & PAGE_MASK ) : ptr + 1 ) {
if ( * ptr & IND_SOURCE | | * ptr & IND_INDIRECTION | |
* ptr & IND_DESTINATION )
2007-07-31 18:16:32 +04:00
* ptr = ( unsigned long ) phys_to_virt ( * ptr ) ;
2006-10-18 17:14:55 +04:00
}
/*
* we do not want to be bothered .
*/
local_irq_disable ( ) ;
2007-07-31 18:16:32 +04:00
printk ( " Will call new kernel at %08lx \n " , image - > start ) ;
2006-10-18 17:14:55 +04:00
printk ( " Bye ... \n " ) ;
2007-08-20 17:57:38 +04:00
__flush_cache_all ( ) ;
2007-07-31 18:16:32 +04:00
( ( noretfun_t ) reboot_code_buffer ) ( ) ;
2006-10-18 17:14:55 +04:00
}