diff options
Diffstat (limited to 'arch/powerpc')
-rw-r--r-- | arch/powerpc/boot/Makefile | 3 | ||||
-rw-r--r-- | arch/powerpc/boot/gunzip_util.c | 140 | ||||
-rw-r--r-- | arch/powerpc/boot/gunzip_util.h | 30 | ||||
-rw-r--r-- | arch/powerpc/boot/main.c | 113 |
4 files changed, 192 insertions, 94 deletions
diff --git a/arch/powerpc/boot/Makefile b/arch/powerpc/boot/Makefile index dc779407de14..3628d8681844 100644 --- a/arch/powerpc/boot/Makefile +++ b/arch/powerpc/boot/Makefile | |||
@@ -41,7 +41,8 @@ $(addprefix $(obj)/,$(zlib) main.o): $(addprefix $(obj)/,$(zliblinuxheader)) \ | |||
41 | $(addprefix $(obj)/,$(zlibheader)) | 41 | $(addprefix $(obj)/,$(zlibheader)) |
42 | 42 | ||
43 | src-wlib := string.S stdio.c main.c flatdevtree.c flatdevtree_misc.c \ | 43 | src-wlib := string.S stdio.c main.c flatdevtree.c flatdevtree_misc.c \ |
44 | ns16550.c serial.c simple_alloc.c div64.S util.S $(zlib) | 44 | ns16550.c serial.c simple_alloc.c div64.S util.S \ |
45 | gunzip_util.c $(zlib) | ||
45 | src-plat := of.c | 46 | src-plat := of.c |
46 | src-boot := crt0.S $(src-wlib) $(src-plat) empty.c | 47 | src-boot := crt0.S $(src-wlib) $(src-plat) empty.c |
47 | 48 | ||
diff --git a/arch/powerpc/boot/gunzip_util.c b/arch/powerpc/boot/gunzip_util.c new file mode 100644 index 000000000000..3d9ff8f081c6 --- /dev/null +++ b/arch/powerpc/boot/gunzip_util.c | |||
@@ -0,0 +1,140 @@ | |||
1 | /* | ||
2 | * Copyright 2007 David Gibson, IBM Corporation. | ||
3 | * Based on earlier work, Copyright (C) Paul Mackerras 1997. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or | ||
6 | * modify it under the terms of the GNU General Public License | ||
7 | * as published by the Free Software Foundation; either version | ||
8 | * 2 of the License, or (at your option) any later version. | ||
9 | */ | ||
10 | |||
11 | #include <stddef.h> | ||
12 | #include "string.h" | ||
13 | #include "stdio.h" | ||
14 | #include "ops.h" | ||
15 | #include "gunzip_util.h" | ||
16 | |||
17 | struct gunzip_state state; | ||
18 | |||
19 | #define HEAD_CRC 2 | ||
20 | #define EXTRA_FIELD 4 | ||
21 | #define ORIG_NAME 8 | ||
22 | #define COMMENT 0x10 | ||
23 | #define RESERVED 0xe0 | ||
24 | |||
25 | void gunzip_start(struct gunzip_state *state, void *src, int srclen) | ||
26 | { | ||
27 | char *hdr = src; | ||
28 | int hdrlen = 0; | ||
29 | |||
30 | memset(state, 0, sizeof(*state)); | ||
31 | |||
32 | /* Check for gzip magic number */ | ||
33 | if ((hdr[0] == 0x1f) && (hdr[1] == 0x8b)) { | ||
34 | /* gzip data, initialize zlib parameters */ | ||
35 | int r, flags; | ||
36 | |||
37 | state->s.workspace = state->scratch; | ||
38 | if (zlib_inflate_workspacesize() > sizeof(state->scratch)) { | ||
39 | printf("insufficient scratch space for gunzip\n\r"); | ||
40 | exit(); | ||
41 | } | ||
42 | |||
43 | /* skip header */ | ||
44 | hdrlen = 10; | ||
45 | flags = hdr[3]; | ||
46 | if (hdr[2] != Z_DEFLATED || (flags & RESERVED) != 0) { | ||
47 | printf("bad gzipped data\n\r"); | ||
48 | exit(); | ||
49 | } | ||
50 | if ((flags & EXTRA_FIELD) != 0) | ||
51 | hdrlen = 12 + hdr[10] + (hdr[11] << 8); | ||
52 | if ((flags & ORIG_NAME) != 0) | ||
53 | while (hdr[hdrlen++] != 0) | ||
54 | ; | ||
55 | if ((flags & COMMENT) != 0) | ||
56 | while (hdr[hdrlen++] != 0) | ||
57 | ; | ||
58 | if ((flags & HEAD_CRC) != 0) | ||
59 | hdrlen += 2; | ||
60 | if (hdrlen >= srclen) { | ||
61 | printf("gunzip_start: ran out of data in header\n\r"); | ||
62 | exit(); | ||
63 | } | ||
64 | |||
65 | r = zlib_inflateInit2(&state->s, -MAX_WBITS); | ||
66 | if (r != Z_OK) { | ||
67 | printf("inflateInit2 returned %d\n\r", r); | ||
68 | exit(); | ||
69 | } | ||
70 | } | ||
71 | |||
72 | state->s.next_in = src + hdrlen; | ||
73 | state->s.avail_in = srclen - hdrlen; | ||
74 | } | ||
75 | |||
76 | int gunzip_partial(struct gunzip_state *state, void *dst, int dstlen) | ||
77 | { | ||
78 | int len; | ||
79 | |||
80 | if (state->s.workspace) { | ||
81 | /* gunzipping */ | ||
82 | int r; | ||
83 | |||
84 | state->s.next_out = dst; | ||
85 | state->s.avail_out = dstlen; | ||
86 | r = zlib_inflate(&state->s, Z_FULL_FLUSH); | ||
87 | if (r != Z_OK && r != Z_STREAM_END) { | ||
88 | printf("inflate returned %d msg: %s\n\r", r, state->s.msg); | ||
89 | exit(); | ||
90 | } | ||
91 | len = state->s.next_out - (unsigned char *)dst; | ||
92 | } else { | ||
93 | /* uncompressed image */ | ||
94 | len = min(state->s.avail_in, (unsigned)dstlen); | ||
95 | memcpy(dst, state->s.next_in, len); | ||
96 | state->s.next_in += len; | ||
97 | state->s.avail_in -= len; | ||
98 | } | ||
99 | return len; | ||
100 | } | ||
101 | |||
102 | void gunzip_exactly(struct gunzip_state *state, void *dst, int dstlen) | ||
103 | { | ||
104 | int len; | ||
105 | |||
106 | len = gunzip_partial(state, dst, dstlen); | ||
107 | if (len < dstlen) { | ||
108 | printf("gunzip_block: ran out of data\n\r"); | ||
109 | exit(); | ||
110 | } | ||
111 | } | ||
112 | |||
113 | void gunzip_discard(struct gunzip_state *state, int len) | ||
114 | { | ||
115 | static char discard_buf[128]; | ||
116 | |||
117 | while (len > sizeof(discard_buf)) { | ||
118 | gunzip_exactly(state, discard_buf, sizeof(discard_buf)); | ||
119 | len -= sizeof(discard_buf); | ||
120 | } | ||
121 | |||
122 | if (len > 0) | ||
123 | gunzip_exactly(state, discard_buf, len); | ||
124 | } | ||
125 | |||
126 | int gunzip_finish(struct gunzip_state *state, void *dst, int dstlen) | ||
127 | { | ||
128 | int len; | ||
129 | |||
130 | if (state->s.workspace) { | ||
131 | len = gunzip_partial(state, dst, dstlen); | ||
132 | zlib_inflateEnd(&state->s); | ||
133 | } else { | ||
134 | /* uncompressed image */ | ||
135 | len = min(state->s.avail_in, (unsigned)dstlen); | ||
136 | memcpy(dst, state->s.next_in, len); | ||
137 | } | ||
138 | |||
139 | return len; | ||
140 | } | ||
diff --git a/arch/powerpc/boot/gunzip_util.h b/arch/powerpc/boot/gunzip_util.h new file mode 100644 index 000000000000..950f62fe0a6d --- /dev/null +++ b/arch/powerpc/boot/gunzip_util.h | |||
@@ -0,0 +1,30 @@ | |||
1 | /* | ||
2 | * Decompression convenience functions | ||
3 | * | ||
4 | * Copyright 2007 David Gibson, IBM Corporation. | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | #ifndef _PPC_BOOT_GUNZIP_UTIL_H_ | ||
11 | #define _PPC_BOOT_GUNZIP_UTIL_H_ | ||
12 | |||
13 | #include "zlib.h" | ||
14 | |||
15 | /* scratch space for gunzip; 46912 is from zlib_inflate_workspacesize() */ | ||
16 | #define GUNZIP_SCRATCH_SIZE 46912 | ||
17 | |||
18 | struct gunzip_state { | ||
19 | z_stream s; | ||
20 | char scratch[46912]; | ||
21 | }; | ||
22 | |||
23 | void gunzip_start(struct gunzip_state *state, void *src, int srclen); | ||
24 | int gunzip_partial(struct gunzip_state *state, void *dst, int dstlen); | ||
25 | void gunzip_exactly(struct gunzip_state *state, void *dst, int len); | ||
26 | void gunzip_discard(struct gunzip_state *state, int len); | ||
27 | int gunzip_finish(struct gunzip_state *state, void *dst, int len); | ||
28 | |||
29 | #endif /* _PPC_BOOT_GUNZIP_UTIL_H_ */ | ||
30 | |||
diff --git a/arch/powerpc/boot/main.c b/arch/powerpc/boot/main.c index 6f6b50d238b6..404620a9e733 100644 --- a/arch/powerpc/boot/main.c +++ b/arch/powerpc/boot/main.c | |||
@@ -14,8 +14,8 @@ | |||
14 | #include "page.h" | 14 | #include "page.h" |
15 | #include "string.h" | 15 | #include "string.h" |
16 | #include "stdio.h" | 16 | #include "stdio.h" |
17 | #include "zlib.h" | ||
18 | #include "ops.h" | 17 | #include "ops.h" |
18 | #include "gunzip_util.h" | ||
19 | #include "flatdevtree.h" | 19 | #include "flatdevtree.h" |
20 | 20 | ||
21 | extern void flush_cache(void *, unsigned long); | 21 | extern void flush_cache(void *, unsigned long); |
@@ -30,6 +30,8 @@ extern char _initrd_end[]; | |||
30 | extern char _dtb_start[]; | 30 | extern char _dtb_start[]; |
31 | extern char _dtb_end[]; | 31 | extern char _dtb_end[]; |
32 | 32 | ||
33 | static struct gunzip_state gzstate; | ||
34 | |||
33 | struct addr_range { | 35 | struct addr_range { |
34 | unsigned long addr; | 36 | unsigned long addr; |
35 | unsigned long size; | 37 | unsigned long size; |
@@ -42,71 +44,12 @@ static struct addr_range initrd; | |||
42 | static unsigned long elfoffset; | 44 | static unsigned long elfoffset; |
43 | static int is_64bit; | 45 | static int is_64bit; |
44 | 46 | ||
45 | /* scratch space for gunzip; 46912 is from zlib_inflate_workspacesize() */ | ||
46 | static char scratch[46912]; | ||
47 | static char elfheader[256]; | 47 | static char elfheader[256]; |
48 | 48 | ||
49 | typedef void (*kernel_entry_t)(unsigned long, unsigned long, void *); | 49 | typedef void (*kernel_entry_t)(unsigned long, unsigned long, void *); |
50 | 50 | ||
51 | #undef DEBUG | 51 | #undef DEBUG |
52 | 52 | ||
53 | #define HEAD_CRC 2 | ||
54 | #define EXTRA_FIELD 4 | ||
55 | #define ORIG_NAME 8 | ||
56 | #define COMMENT 0x10 | ||
57 | #define RESERVED 0xe0 | ||
58 | |||
59 | static void gunzip(void *dst, int dstlen, unsigned char *src, int *lenp) | ||
60 | { | ||
61 | z_stream s; | ||
62 | int r, i, flags; | ||
63 | |||
64 | /* skip header */ | ||
65 | i = 10; | ||
66 | flags = src[3]; | ||
67 | if (src[2] != Z_DEFLATED || (flags & RESERVED) != 0) { | ||
68 | printf("bad gzipped data\n\r"); | ||
69 | exit(); | ||
70 | } | ||
71 | if ((flags & EXTRA_FIELD) != 0) | ||
72 | i = 12 + src[10] + (src[11] << 8); | ||
73 | if ((flags & ORIG_NAME) != 0) | ||
74 | while (src[i++] != 0) | ||
75 | ; | ||
76 | if ((flags & COMMENT) != 0) | ||
77 | while (src[i++] != 0) | ||
78 | ; | ||
79 | if ((flags & HEAD_CRC) != 0) | ||
80 | i += 2; | ||
81 | if (i >= *lenp) { | ||
82 | printf("gunzip: ran out of data in header\n\r"); | ||
83 | exit(); | ||
84 | } | ||
85 | |||
86 | if (zlib_inflate_workspacesize() > sizeof(scratch)) { | ||
87 | printf("gunzip needs more mem\n"); | ||
88 | exit(); | ||
89 | } | ||
90 | memset(&s, 0, sizeof(s)); | ||
91 | s.workspace = scratch; | ||
92 | r = zlib_inflateInit2(&s, -MAX_WBITS); | ||
93 | if (r != Z_OK) { | ||
94 | printf("inflateInit2 returned %d\n\r", r); | ||
95 | exit(); | ||
96 | } | ||
97 | s.next_in = src + i; | ||
98 | s.avail_in = *lenp - i; | ||
99 | s.next_out = dst; | ||
100 | s.avail_out = dstlen; | ||
101 | r = zlib_inflate(&s, Z_FULL_FLUSH); | ||
102 | if (r != Z_OK && r != Z_STREAM_END) { | ||
103 | printf("inflate returned %d msg: %s\n\r", r, s.msg); | ||
104 | exit(); | ||
105 | } | ||
106 | *lenp = s.next_out - (unsigned char *) dst; | ||
107 | zlib_inflateEnd(&s); | ||
108 | } | ||
109 | |||
110 | static int is_elf64(void *hdr) | 53 | static int is_elf64(void *hdr) |
111 | { | 54 | { |
112 | Elf64_Ehdr *elf64 = hdr; | 55 | Elf64_Ehdr *elf64 = hdr; |
@@ -132,8 +75,8 @@ static int is_elf64(void *hdr) | |||
132 | return 0; | 75 | return 0; |
133 | 76 | ||
134 | elfoffset = (unsigned long)elf64ph->p_offset; | 77 | elfoffset = (unsigned long)elf64ph->p_offset; |
135 | vmlinux.size = (unsigned long)elf64ph->p_filesz + elfoffset; | 78 | vmlinux.size = (unsigned long)elf64ph->p_filesz; |
136 | vmlinux.memsize = (unsigned long)elf64ph->p_memsz + elfoffset; | 79 | vmlinux.memsize = (unsigned long)elf64ph->p_memsz; |
137 | 80 | ||
138 | is_64bit = 1; | 81 | is_64bit = 1; |
139 | return 1; | 82 | return 1; |
@@ -164,8 +107,8 @@ static int is_elf32(void *hdr) | |||
164 | return 0; | 107 | return 0; |
165 | 108 | ||
166 | elfoffset = elf32ph->p_offset; | 109 | elfoffset = elf32ph->p_offset; |
167 | vmlinux.size = elf32ph->p_filesz + elf32ph->p_offset; | 110 | vmlinux.size = elf32ph->p_filesz; |
168 | vmlinux.memsize = elf32ph->p_memsz + elf32ph->p_offset; | 111 | vmlinux.memsize = elf32ph->p_memsz; |
169 | return 1; | 112 | return 1; |
170 | } | 113 | } |
171 | 114 | ||
@@ -177,13 +120,8 @@ static void prep_kernel(unsigned long a1, unsigned long a2) | |||
177 | vmlinuz.size = (unsigned long)(_vmlinux_end - _vmlinux_start); | 120 | vmlinuz.size = (unsigned long)(_vmlinux_end - _vmlinux_start); |
178 | 121 | ||
179 | /* gunzip the ELF header of the kernel */ | 122 | /* gunzip the ELF header of the kernel */ |
180 | if (*(unsigned short *)vmlinuz.addr == 0x1f8b) { | 123 | gunzip_start(&gzstate, (void *)vmlinuz.addr, vmlinuz.size); |
181 | len = vmlinuz.size; | 124 | gunzip_exactly(&gzstate, elfheader, sizeof(elfheader)); |
182 | gunzip(elfheader, sizeof(elfheader), | ||
183 | (unsigned char *)vmlinuz.addr, &len); | ||
184 | } else | ||
185 | memcpy(elfheader, (const void *)vmlinuz.addr, | ||
186 | sizeof(elfheader)); | ||
187 | 125 | ||
188 | if (!is_elf64(elfheader) && !is_elf32(elfheader)) { | 126 | if (!is_elf64(elfheader) && !is_elf32(elfheader)) { |
189 | printf("Error: not a valid PPC32 or PPC64 ELF file!\n\r"); | 127 | printf("Error: not a valid PPC32 or PPC64 ELF file!\n\r"); |
@@ -192,10 +130,10 @@ static void prep_kernel(unsigned long a1, unsigned long a2) | |||
192 | if (platform_ops.image_hdr) | 130 | if (platform_ops.image_hdr) |
193 | platform_ops.image_hdr(elfheader); | 131 | platform_ops.image_hdr(elfheader); |
194 | 132 | ||
195 | /* We need to alloc the memsize plus the file offset since gzip | 133 | /* We need to alloc the memsize: gzip will expand the kernel |
196 | * will expand the header (file offset), then the kernel, then | 134 | * text/data, then possible rubbish we don't care about. But |
197 | * possible rubbish we don't care about. But the kernel bss must | 135 | * the kernel bss must be claimed (it will be zero'd by the |
198 | * be claimed (it will be zero'd by the kernel itself) | 136 | * kernel itself) |
199 | */ | 137 | */ |
200 | printf("Allocating 0x%lx bytes for kernel ...\n\r", vmlinux.memsize); | 138 | printf("Allocating 0x%lx bytes for kernel ...\n\r", vmlinux.memsize); |
201 | vmlinux.addr = (unsigned long)malloc(vmlinux.memsize); | 139 | vmlinux.addr = (unsigned long)malloc(vmlinux.memsize); |
@@ -237,24 +175,13 @@ static void prep_kernel(unsigned long a1, unsigned long a2) | |||
237 | } | 175 | } |
238 | 176 | ||
239 | /* Eventually gunzip the kernel */ | 177 | /* Eventually gunzip the kernel */ |
240 | if (*(unsigned short *)vmlinuz.addr == 0x1f8b) { | 178 | printf("gunzipping (0x%lx <- 0x%lx:0x%0lx)...", |
241 | printf("gunzipping (0x%lx <- 0x%lx:0x%0lx)...", | 179 | vmlinux.addr, vmlinuz.addr, vmlinuz.addr+vmlinuz.size); |
242 | vmlinux.addr, vmlinuz.addr, vmlinuz.addr+vmlinuz.size); | 180 | /* discard up to the actual load data */ |
243 | len = vmlinuz.size; | 181 | gunzip_discard(&gzstate, elfoffset - sizeof(elfheader)); |
244 | gunzip((void *)vmlinux.addr, vmlinux.memsize, | 182 | len = gunzip_finish(&gzstate, (void *)vmlinux.addr, |
245 | (unsigned char *)vmlinuz.addr, &len); | 183 | vmlinux.memsize); |
246 | printf("done 0x%lx bytes\n\r", len); | 184 | printf("done 0x%lx bytes\n\r", len); |
247 | } else { | ||
248 | memmove((void *)vmlinux.addr,(void *)vmlinuz.addr, | ||
249 | vmlinuz.size); | ||
250 | } | ||
251 | |||
252 | /* Skip over the ELF header */ | ||
253 | #ifdef DEBUG | ||
254 | printf("... skipping 0x%lx bytes of ELF header\n\r", | ||
255 | elfoffset); | ||
256 | #endif | ||
257 | vmlinux.addr += elfoffset; | ||
258 | 185 | ||
259 | flush_cache((void *)vmlinux.addr, vmlinux.size); | 186 | flush_cache((void *)vmlinux.addr, vmlinux.size); |
260 | } | 187 | } |