2005-04-17 02:20:36 +04:00
/*
* Program to hack in a PT_NOTE program header entry in an ELF file .
* This is needed for OF on RS / 6000 s to load an image correctly .
* Note that OF needs a program header entry for the note , not an
* ELF section .
*
* Copyright 2000 Paul Mackerras .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
2008-10-31 14:27:17 +03:00
* Usage : addnote zImage
2005-04-17 02:20:36 +04:00
*/
# include <stdio.h>
# include <stdlib.h>
# include <fcntl.h>
# include <unistd.h>
# include <string.h>
2005-05-01 19:58:45 +04:00
/* CHRP note section */
2010-09-13 13:47:40 +04:00
static const char arch [ ] = " PowerPC " ;
2005-04-17 02:20:36 +04:00
# define N_DESCR 6
unsigned int descr [ N_DESCR ] = {
0xffffffff , /* real-mode = true */
2008-06-24 08:20:29 +04:00
0x02000000 , /* real-base, i.e. where we expect OF to be */
2005-04-17 02:20:36 +04:00
0xffffffff , /* real-size */
0xffffffff , /* virt-base */
0xffffffff , /* virt-size */
0x4000 , /* load-base */
} ;
2005-05-01 19:58:45 +04:00
/* RPA note section */
2010-09-13 13:47:40 +04:00
static const char rpaname [ ] = " IBM,RPA-Client-Config " ;
2005-05-01 19:58:45 +04:00
/*
* Note : setting ignore_my_client_config * should * mean that OF ignores
* all the other fields , but there is a firmware bug which means that
* it looks at the splpar field at least . So these values need to be
* reasonable .
*/
# define N_RPA_DESCR 8
unsigned int rpanote [ N_RPA_DESCR ] = {
2008-10-31 14:27:17 +03:00
0 , /* lparaffinity */
64 , /* min_rmo_size */
2005-05-01 19:58:45 +04:00
0 , /* min_rmo_percent */
2008-10-31 14:27:17 +03:00
40 , /* max_pft_size */
2005-05-01 19:58:45 +04:00
1 , /* splpar */
- 1 , /* min_load */
2008-10-31 14:27:17 +03:00
0 , /* new_mem_def */
1 , /* ignore_my_client_config */
2005-05-01 19:58:45 +04:00
} ;
# define ROUNDUP(len) (((len) + 3) & ~3)
2005-04-17 02:20:36 +04:00
unsigned char buf [ 512 ] ;
2008-10-31 14:27:17 +03:00
# define GET_16BE(off) ((buf[off] << 8) + (buf[(off)+1]))
# define GET_32BE(off) ((GET_16BE(off) << 16) + GET_16BE((off)+2))
2005-04-17 02:20:36 +04:00
2008-10-31 14:27:17 +03:00
# define PUT_16BE(off, v) (buf[off] = ((v) >> 8) & 0xff, \
buf [ ( off ) + 1 ] = ( v ) & 0xff )
# define PUT_32BE(off, v) (PUT_16BE((off), (v) >> 16), \
PUT_16BE ( ( off ) + 2 , ( v ) ) )
2005-04-17 02:20:36 +04:00
/* Structure of an ELF file */
# define E_IDENT 0 /* ELF header */
# define E_PHOFF 28
# define E_PHENTSIZE 42
# define E_PHNUM 44
# define E_HSIZE 52 /* size of ELF header */
# define EI_MAGIC 0 /* offsets in E_IDENT area */
# define EI_CLASS 4
# define EI_DATA 5
# define PH_TYPE 0 /* ELF program header */
# define PH_OFFSET 4
# define PH_FILESZ 16
# define PH_HSIZE 32 /* size of program header */
# define PT_NOTE 4 /* Program header type = note */
# define ELFCLASS32 1
# define ELFDATA2MSB 2
unsigned char elf_magic [ 4 ] = { 0x7f , ' E ' , ' L ' , ' F ' } ;
int
main ( int ac , char * * av )
{
2008-10-31 14:27:17 +03:00
int fd , n , i ;
2005-04-17 02:20:36 +04:00
int ph , ps , np ;
2005-05-01 19:58:45 +04:00
int nnote , nnote2 , ns ;
2008-10-31 14:27:17 +03:00
if ( ac ! = 2 ) {
fprintf ( stderr , " Usage: %s elf-file \n " , av [ 0 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
}
2008-10-31 14:27:17 +03:00
fd = open ( av [ 1 ] , O_RDWR ) ;
2005-04-17 02:20:36 +04:00
if ( fd < 0 ) {
2008-10-31 14:27:17 +03:00
perror ( av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
}
2005-05-01 19:58:45 +04:00
nnote = 12 + ROUNDUP ( strlen ( arch ) + 1 ) + sizeof ( descr ) ;
nnote2 = 12 + ROUNDUP ( strlen ( rpaname ) + 1 ) + sizeof ( rpanote ) ;
2005-04-17 02:20:36 +04:00
n = read ( fd , buf , sizeof ( buf ) ) ;
if ( n < 0 ) {
perror ( " read " ) ;
exit ( 1 ) ;
}
if ( n < E_HSIZE | | memcmp ( & buf [ E_IDENT + EI_MAGIC ] , elf_magic , 4 ) ! = 0 )
goto notelf ;
if ( buf [ E_IDENT + EI_CLASS ] ! = ELFCLASS32
| | buf [ E_IDENT + EI_DATA ] ! = ELFDATA2MSB ) {
fprintf ( stderr , " %s is not a big-endian 32-bit ELF image \n " ,
2008-10-31 14:27:17 +03:00
av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
}
2008-10-31 14:27:17 +03:00
ph = GET_32BE ( E_PHOFF ) ;
ps = GET_16BE ( E_PHENTSIZE ) ;
np = GET_16BE ( E_PHNUM ) ;
2005-04-17 02:20:36 +04:00
if ( ph < E_HSIZE | | ps < PH_HSIZE | | np < 1 )
goto notelf ;
2005-05-01 19:58:45 +04:00
if ( ph + ( np + 2 ) * ps + nnote + nnote2 > n )
2005-04-17 02:20:36 +04:00
goto nospace ;
for ( i = 0 ; i < np ; + + i ) {
2008-10-31 14:27:17 +03:00
if ( GET_32BE ( ph + PH_TYPE ) = = PT_NOTE ) {
2005-04-17 02:20:36 +04:00
fprintf ( stderr , " %s already has a note entry \n " ,
2008-10-31 14:27:17 +03:00
av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 0 ) ;
}
ph + = ps ;
}
/* XXX check that the area we want to use is all zeroes */
2005-05-01 19:58:45 +04:00
for ( i = 0 ; i < 2 * ps + nnote + nnote2 ; + + i )
2005-04-17 02:20:36 +04:00
if ( buf [ ph + i ] ! = 0 )
goto nospace ;
/* fill in the program header entry */
2005-05-01 19:58:45 +04:00
ns = ph + 2 * ps ;
2008-10-31 14:27:17 +03:00
PUT_32BE ( ph + PH_TYPE , PT_NOTE ) ;
PUT_32BE ( ph + PH_OFFSET , ns ) ;
PUT_32BE ( ph + PH_FILESZ , nnote ) ;
2005-04-17 02:20:36 +04:00
/* fill in the note area we point to */
/* XXX we should probably make this a proper section */
2008-10-31 14:27:17 +03:00
PUT_32BE ( ns , strlen ( arch ) + 1 ) ;
PUT_32BE ( ns + 4 , N_DESCR * 4 ) ;
PUT_32BE ( ns + 8 , 0x1275 ) ;
2005-08-08 07:24:38 +04:00
strcpy ( ( char * ) & buf [ ns + 12 ] , arch ) ;
2005-04-17 02:20:36 +04:00
ns + = 12 + strlen ( arch ) + 1 ;
2005-05-01 19:58:45 +04:00
for ( i = 0 ; i < N_DESCR ; + + i , ns + = 4 )
2008-10-31 14:27:17 +03:00
PUT_32BE ( ns , descr [ i ] ) ;
2005-05-01 19:58:45 +04:00
/* fill in the second program header entry and the RPA note area */
ph + = ps ;
2008-10-31 14:27:17 +03:00
PUT_32BE ( ph + PH_TYPE , PT_NOTE ) ;
PUT_32BE ( ph + PH_OFFSET , ns ) ;
PUT_32BE ( ph + PH_FILESZ , nnote2 ) ;
2005-05-01 19:58:45 +04:00
/* fill in the note area we point to */
2008-10-31 14:27:17 +03:00
PUT_32BE ( ns , strlen ( rpaname ) + 1 ) ;
PUT_32BE ( ns + 4 , sizeof ( rpanote ) ) ;
PUT_32BE ( ns + 8 , 0x12759999 ) ;
strcpy ( ( char * ) & buf [ ns + 12 ] , rpaname ) ;
ns + = 12 + ROUNDUP ( strlen ( rpaname ) + 1 ) ;
for ( i = 0 ; i < N_RPA_DESCR ; + + i , ns + = 4 )
PUT_32BE ( ns , rpanote [ i ] ) ;
2005-04-17 02:20:36 +04:00
/* Update the number of program headers */
2008-10-31 14:27:17 +03:00
PUT_16BE ( E_PHNUM , np + 2 ) ;
2005-04-17 02:20:36 +04:00
/* write back */
lseek ( fd , ( long ) 0 , SEEK_SET ) ;
i = write ( fd , buf , n ) ;
if ( i < 0 ) {
perror ( " write " ) ;
exit ( 1 ) ;
}
if ( i < n ) {
2008-10-31 14:27:17 +03:00
fprintf ( stderr , " %s: write truncated \n " , av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
}
exit ( 0 ) ;
notelf :
2008-10-31 14:27:17 +03:00
fprintf ( stderr , " %s does not appear to be an ELF file \n " , av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
nospace :
fprintf ( stderr , " sorry, I can't find space in %s to put the note \n " ,
2008-10-31 14:27:17 +03:00
av [ 1 ] ) ;
2005-04-17 02:20:36 +04:00
exit ( 1 ) ;
}