blob: b5b3febc10cc9d12a8ddd9ef01aaf022711b7198 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * addinitrd - program to add a initrd image to an ecoff kernel
3 *
4 * (C) 1999 Thomas Bogendoerfer
5 * minor modifications, cleanup: Guido Guenther <agx@sigxcpu.org>
6 * further cleanup: Maciej W. Rozycki
7 */
8
9#include <sys/types.h>
10#include <sys/stat.h>
11#include <fcntl.h>
12#include <unistd.h>
13#include <stdio.h>
14#include <netinet/in.h>
15
16#include "ecoff.h"
17
18#define MIPS_PAGE_SIZE 4096
19#define MIPS_PAGE_MASK (MIPS_PAGE_SIZE-1)
20
21#define swab16(x) \
22 ((unsigned short)( \
23 (((unsigned short)(x) & (unsigned short)0x00ffU) << 8) | \
24 (((unsigned short)(x) & (unsigned short)0xff00U) >> 8) ))
25
26#define swab32(x) \
27 ((unsigned int)( \
28 (((unsigned int)(x) & (unsigned int)0x000000ffUL) << 24) | \
29 (((unsigned int)(x) & (unsigned int)0x0000ff00UL) << 8) | \
30 (((unsigned int)(x) & (unsigned int)0x00ff0000UL) >> 8) | \
31 (((unsigned int)(x) & (unsigned int)0xff000000UL) >> 24) ))
32
33#define SWAB(a) (swab ? swab32(a) : (a))
34
Ralf Baechle49a89ef2007-10-11 23:46:15 +010035void die(char *s)
Linus Torvalds1da177e2005-04-16 15:20:36 -070036{
Ralf Baechle49a89ef2007-10-11 23:46:15 +010037 perror(s);
38 exit(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -070039}
40
Ralf Baechle49a89ef2007-10-11 23:46:15 +010041int main(int argc, char *argv[])
Linus Torvalds1da177e2005-04-16 15:20:36 -070042{
Ralf Baechle21a151d2007-10-11 23:46:15 +010043 int fd_vmlinux, fd_initrd, fd_outfile;
Linus Torvalds1da177e2005-04-16 15:20:36 -070044 FILHDR efile;
45 AOUTHDR eaout;
46 SCNHDR esecs[3];
47 struct stat st;
48 char buf[1024];
49 unsigned long loadaddr;
50 unsigned long initrd_header[2];
Ralf Baechle21a151d2007-10-11 23:46:15 +010051 int i, cnt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070052 int swab = 0;
53
54 if (argc != 4) {
Ralf Baechle21a151d2007-10-11 23:46:15 +010055 printf("Usage: %s <vmlinux> <initrd> <outfile>\n", argv[0]);
Ralf Baechle49a89ef2007-10-11 23:46:15 +010056 exit(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -070057 }
58
Ralf Baechle21a151d2007-10-11 23:46:15 +010059 if ((fd_vmlinux = open (argv[1], O_RDONLY)) < 0)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010060 die("open vmlinux");
Linus Torvalds1da177e2005-04-16 15:20:36 -070061 if (read (fd_vmlinux, &efile, sizeof efile) != sizeof efile)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010062 die("read file header");
Linus Torvalds1da177e2005-04-16 15:20:36 -070063 if (read (fd_vmlinux, &eaout, sizeof eaout) != sizeof eaout)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010064 die("read aout header");
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 if (read (fd_vmlinux, esecs, sizeof esecs) != sizeof esecs)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010066 die("read section headers");
Linus Torvalds1da177e2005-04-16 15:20:36 -070067 /*
68 * check whether the file is good for us
69 */
70 /* TBD */
71
72 /*
73 * check, if we have to swab words
74 */
75 if (ntohs(0xaa55) == 0xaa55) {
76 if (efile.f_magic == swab16(MIPSELMAGIC))
77 swab = 1;
78 } else {
79 if (efile.f_magic == swab16(MIPSEBMAGIC))
80 swab = 1;
81 }
82
83 /* make sure we have an empty data segment for the initrd */
84 if (eaout.dsize || esecs[1].s_size) {
Ralf Baechle49a89ef2007-10-11 23:46:15 +010085 fprintf(stderr, "Data segment not empty. Giving up!\n");
86 exit(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -070087 }
88 if ((fd_initrd = open (argv[2], O_RDONLY)) < 0)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010089 die("open initrd");
Linus Torvalds1da177e2005-04-16 15:20:36 -070090 if (fstat (fd_initrd, &st) < 0)
Ralf Baechle49a89ef2007-10-11 23:46:15 +010091 die("fstat initrd");
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 loadaddr = ((SWAB(esecs[2].s_vaddr) + SWAB(esecs[2].s_size)
93 + MIPS_PAGE_SIZE-1) & ~MIPS_PAGE_MASK) - 8;
94 if (loadaddr < (SWAB(esecs[2].s_vaddr) + SWAB(esecs[2].s_size)))
95 loadaddr += MIPS_PAGE_SIZE;
96 initrd_header[0] = SWAB(0x494E5244);
97 initrd_header[1] = SWAB(st.st_size);
98 eaout.dsize = esecs[1].s_size = initrd_header[1] = SWAB(st.st_size+8);
99 eaout.data_start = esecs[1].s_vaddr = esecs[1].s_paddr = SWAB(loadaddr);
100
Ralf Baechle21a151d2007-10-11 23:46:15 +0100101 if ((fd_outfile = open (argv[3], O_RDWR|O_CREAT|O_TRUNC, 0666)) < 0)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100102 die("open outfile");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103 if (write (fd_outfile, &efile, sizeof efile) != sizeof efile)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100104 die("write file header");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700105 if (write (fd_outfile, &eaout, sizeof eaout) != sizeof eaout)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100106 die("write aout header");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700107 if (write (fd_outfile, esecs, sizeof esecs) != sizeof esecs)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100108 die("write section headers");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109 /* skip padding */
110 if(lseek(fd_vmlinux, SWAB(esecs[0].s_scnptr), SEEK_SET) == (off_t)-1)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100111 die("lseek vmlinux");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112 if(lseek(fd_outfile, SWAB(esecs[0].s_scnptr), SEEK_SET) == (off_t)-1)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100113 die("lseek outfile");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700114 /* copy text segment */
115 cnt = SWAB(eaout.tsize);
116 while (cnt) {
117 if ((i = read (fd_vmlinux, buf, sizeof buf)) <= 0)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100118 die("read vmlinux");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700119 if (write (fd_outfile, buf, i) != i)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100120 die("write vmlinux");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121 cnt -= i;
122 }
123 if (write (fd_outfile, initrd_header, sizeof initrd_header) != sizeof initrd_header)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100124 die("write initrd header");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700125 while ((i = read (fd_initrd, buf, sizeof buf)) > 0)
126 if (write (fd_outfile, buf, i) != i)
Ralf Baechle49a89ef2007-10-11 23:46:15 +0100127 die("write initrd");
128 close(fd_vmlinux);
129 close(fd_initrd);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700130 return 0;
131}