2018-11-15 14:52:50 +09:00
// SPDX-License-Identifier: GPL-2.0
/*
* Kexec image loader
* Copyright ( C ) 2018 Linaro Limited
* Author : AKASHI Takahiro < takahiro . akashi @ linaro . org >
*/
# define pr_fmt(fmt) "kexec_file(Image): " fmt
# include <linux/err.h>
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/kexec.h>
2018-11-15 14:52:54 +09:00
# include <linux/pe.h>
2018-11-15 14:52:50 +09:00
# include <linux/string.h>
2018-11-15 14:52:54 +09:00
# include <linux/verification.h>
2018-11-15 14:52:50 +09:00
# include <asm/byteorder.h>
# include <asm/cpufeature.h>
# include <asm/image.h>
# include <asm/memory.h>
static int image_probe ( const char * kernel_buf , unsigned long kernel_len )
{
2018-11-15 14:52:54 +09:00
const struct arm64_image_header * h =
( const struct arm64_image_header * ) ( kernel_buf ) ;
2018-11-15 14:52:50 +09:00
2018-11-15 14:52:54 +09:00
if ( ! h | | ( kernel_len < sizeof ( * h ) ) )
return - EINVAL ;
2018-11-15 14:52:50 +09:00
2018-11-15 14:52:54 +09:00
if ( memcmp ( & h - > magic , ARM64_IMAGE_MAGIC , sizeof ( h - > magic ) ) )
2018-11-15 14:52:50 +09:00
return - EINVAL ;
return 0 ;
}
static void * image_load ( struct kimage * image ,
char * kernel , unsigned long kernel_len ,
char * initrd , unsigned long initrd_len ,
char * cmdline , unsigned long cmdline_len )
{
struct arm64_image_header * h ;
u64 flags , value ;
bool be_image , be_kernel ;
struct kexec_buf kbuf ;
2020-11-03 12:11:06 -08:00
unsigned long text_offset , kernel_segment_number ;
2018-11-15 14:52:50 +09:00
struct kexec_segment * kernel_segment ;
int ret ;
/*
* We require a kernel with an unambiguous Image header . Per
2019-06-12 14:52:38 -03:00
* Documentation / arm64 / booting . rst , this is the case when image_size
2018-11-15 14:52:50 +09:00
* is non - zero ( practically speaking , since v3 .17 ) .
*/
h = ( struct arm64_image_header * ) kernel ;
if ( ! h - > image_size )
return ERR_PTR ( - EINVAL ) ;
/* Check cpu features */
flags = le64_to_cpu ( h - > flags ) ;
be_image = arm64_image_flag_field ( flags , ARM64_IMAGE_FLAG_BE ) ;
be_kernel = IS_ENABLED ( CONFIG_CPU_BIG_ENDIAN ) ;
if ( ( be_image ! = be_kernel ) & & ! system_supports_mixed_endian ( ) )
return ERR_PTR ( - EINVAL ) ;
value = arm64_image_flag_field ( flags , ARM64_IMAGE_FLAG_PAGE_SIZE ) ;
if ( ( ( value = = ARM64_IMAGE_FLAG_PAGE_SIZE_4K ) & &
! system_supports_4kb_granule ( ) ) | |
( ( value = = ARM64_IMAGE_FLAG_PAGE_SIZE_64K ) & &
! system_supports_64kb_granule ( ) ) | |
( ( value = = ARM64_IMAGE_FLAG_PAGE_SIZE_16K ) & &
! system_supports_16kb_granule ( ) ) )
return ERR_PTR ( - EINVAL ) ;
/* Load the kernel */
kbuf . image = image ;
kbuf . buf_min = 0 ;
kbuf . buf_max = ULONG_MAX ;
kbuf . top_down = false ;
kbuf . buffer = kernel ;
kbuf . bufsz = kernel_len ;
2019-07-11 17:27:32 +05:30
kbuf . mem = KEXEC_BUF_MEM_UNKNOWN ;
2018-11-15 14:52:50 +09:00
kbuf . memsz = le64_to_cpu ( h - > image_size ) ;
text_offset = le64_to_cpu ( h - > text_offset ) ;
kbuf . buf_align = MIN_KIMG_ALIGN ;
/* Adjust kernel segment with TEXT_OFFSET */
kbuf . memsz + = text_offset ;
2020-11-03 12:11:06 -08:00
kernel_segment_number = image - > nr_segments ;
/*
* The location of the kernel segment may make it impossible to satisfy
* the other segment requirements , so we try repeatedly to find a
* location that will work .
*/
while ( ( ret = kexec_add_buffer ( & kbuf ) ) = = 0 ) {
/* Try to load additional data */
kernel_segment = & image - > segment [ kernel_segment_number ] ;
ret = load_other_segments ( image , kernel_segment - > mem ,
kernel_segment - > memsz , initrd ,
initrd_len , cmdline ) ;
if ( ! ret )
break ;
/*
* We couldn ' t find space for the other segments ; erase the
* kernel segment and try the next available hole .
*/
image - > nr_segments - = 1 ;
kbuf . buf_min = kernel_segment - > mem + kernel_segment - > memsz ;
kbuf . mem = KEXEC_BUF_MEM_UNKNOWN ;
}
if ( ret ) {
pr_err ( " Could not find any suitable kernel location! " ) ;
2018-11-15 14:52:50 +09:00
return ERR_PTR ( ret ) ;
2020-11-03 12:11:06 -08:00
}
2018-11-15 14:52:50 +09:00
2020-11-03 12:11:06 -08:00
kernel_segment = & image - > segment [ kernel_segment_number ] ;
2018-11-15 14:52:50 +09:00
kernel_segment - > mem + = text_offset ;
kernel_segment - > memsz - = text_offset ;
image - > start = kernel_segment - > mem ;
pr_debug ( " Loaded kernel at 0x%lx bufsz=0x%lx memsz=0x%lx \n " ,
kernel_segment - > mem , kbuf . bufsz ,
kernel_segment - > memsz ) ;
2020-11-09 10:49:23 +00:00
return NULL ;
2018-11-15 14:52:50 +09:00
}
2018-11-15 14:52:54 +09:00
# ifdef CONFIG_KEXEC_IMAGE_VERIFY_SIG
static int image_verify_sig ( const char * kernel , unsigned long kernel_len )
{
return verify_pefile_signature ( kernel , kernel_len , NULL ,
VERIFYING_KEXEC_PE_SIGNATURE ) ;
}
# endif
2018-11-15 14:52:50 +09:00
const struct kexec_file_ops kexec_image_ops = {
. probe = image_probe ,
. load = image_load ,
2018-11-15 14:52:54 +09:00
# ifdef CONFIG_KEXEC_IMAGE_VERIFY_SIG
. verify_sig = image_verify_sig ,
# endif
2018-11-15 14:52:50 +09:00
} ;