Add a library to search through a cpio or initramfs buffer for a specified path name. No canocalization of the path is performed.
Signed-off-by: Milton Miller <[EMAIL PROTECTED]> --- vs 12177 rediff Makefile cpio.c should be reusable by a stand alone user space applicaton. Index: kernel/arch/powerpc/boot/cpio.c =================================================================== --- /dev/null 1970-01-01 00:00:00.000000000 +0000 +++ kernel/arch/powerpc/boot/cpio.c 2007-08-22 20:51:29.000000000 -0500 @@ -0,0 +1,306 @@ +/* + * 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. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + * Copyright 2007 IBM Corporation. + * + * Authors: Milton Miller <[EMAIL PROTECTED]> + * + */ + +#include "cpio.h" +#include "string.h" + +struct cpio_header { + char ino[8]; + char mode[8]; + char uid[8]; + char gid[8]; + char nlink[8]; + char mtime[8]; + char filesize[8]; + char maj[8]; + char min[8]; + char rmaj[8]; + char rmin[8]; + char namesize[8]; + char chksum[8]; +} cpio_header_buf; + +static int check_magic(char magic[6]) +{ + return !memcmp(magic,"070701",6) || !memcmp(magic,"070702",6); +} + +static int read_magic(struct gunzip_state *stream) +{ + int len; + char magic[6]; + + len = gunzip_partial(stream, magic, sizeof(magic)); + if (len == 0) + return 0; + + if (len == sizeof(magic) && check_magic(magic)) + return len; + + + /* Not the right magic or short read. We might have stumbled + * onto a compressed archive immediately following an + * uncompressed one, or just some NUL bytes at the end of the + * archive. Inform the higher layers by the negative length. + */ + return -len; +} + +static int get_cpio_header(struct gunzip_state *stream) +{ + int len; + + len = read_magic(stream); + if (len <= 0) + return len; + + gunzip_exactly(stream, &cpio_header_buf, sizeof(cpio_header_buf)); + len += sizeof(cpio_header_buf); + + return len; +} + +static unsigned int cpio_str_to_num(char hexascii[8]) +{ + unsigned int num = 0; + char c; + int d; + + for (d=0; d < 8; d++) { + c = hexascii[d]; + num <<= 4; + if (c >= '0' && c <= '9') { + num += c - '0'; + } else if (c >= 'A' && c <= 'F') { + num += c - 'A' + 10; + } else if (c >= 'a' && c <= 'f') { + num += c - 'a' + 10; + } else { + cpio_error("bad cpio archive header: " + "invalid a hex digit"); + } + } + + return num; +} + +static char name_buf[MAX_PATH+1]; +static const char cpio_end[] = "TRAILER!!!"; +#define CPIO_END_LEN sizeof(cpio_end) + +/* check_next_file + * Look for @path in @stream. Set @consumed to the number of bytes + * succesfully read and processed. return 1 on match, 0 for discarding + * an unmatched file, -1 on end of archive (either detected trailer or + * EOF on stream), or -(1 + bytes read) for a short read or bad magic + * number. (For the short or bad read, the consumed is not changed). + */ +static int check_next_file(char *path, int pathlen, + struct gunzip_state *stream, int *consumed) +{ + int len, total, match; + + if (pathlen > MAX_PATH) { + cpio_error("path too long to search\n"); + } + total = get_cpio_header(stream); + if (total <= 0) + return total - 1; + + len = cpio_str_to_num(cpio_header_buf.namesize); + + if (len == pathlen || len == CPIO_END_LEN) { + gunzip_exactly(stream, name_buf, len); + total += len; + match = !strcmp(name_buf, path); + if (!match && !cpio_str_to_num(cpio_header_buf.filesize)) + match = -!strcmp(name_buf, cpio_end); + } else { + gunzip_discard(stream, len); + total += len; + name_buf[0] = '\0'; + match = 0; + } + + len = total % 4; + if (len) { + gunzip_discard(stream, 4 - len); + total += 4 - len; + } + + if (!match) { + len = cpio_str_to_num(cpio_header_buf.filesize); + gunzip_discard(stream, len); + total += len; + + len = total % 4; + if (len) { + gunzip_discard(stream, 4 - len); + total += 4 - len; + } + } + + *consumed += total; + return match; +} + +static char *this_buf; +static int this_archive; + +/* find_in_cpio. + * find a pathname @path in a single cpio archive described by @stream. + * Return is the same as check_next_file. + */ +int find_in_cpio(char *path, struct gunzip_state *stream) +{ + int found; + int pathlen = strlen(path) + 1; + + this_archive = 0; + do { + found = check_next_file(path, pathlen, stream, &this_archive); + } while (found == 0); + + return found; +} + +/* find_in_initramfs + * Search a initramfs buffer for a given path name. Returns 0 on + * not found, or 1 with @state ready to read the file. + * + * Note: this returns the first match, not the last. The kernel + * decompressor effectivly uses the last match. This code also + * doesn't worry about the directories in the path existing. + */ +int find_in_initramfs(char *path, char *buf, int len, + struct gunzip_state *stream) +{ + int found, total = 0; + int pathlen = strlen(path) + 1; + int *ibuf; + + do { + /* get to word boundary, but stop if not NUL */ + for (; total % 4 && total < len - 4; total++) + if (buf[total]) + break; + + if ((total % 4) == 0) { + /* fast forward over NUL words. */ + for (ibuf = (int *)&buf[total]; + total < len - sizeof(*ibuf) && !*ibuf; + ibuf++) + total += sizeof(*ibuf); + } + + /* check remainder of a short archive -- it must all be + * zero as both gzip header and cpio header are bigger + * than this. + */ + if (total >= len - 4) { + for (;total < len; total++) { + if (buf[len]) { + cpio_error("Junk at end of buffer"); + } + } + break; + } else if (total % 4) { + /* + * If we are unalinged and not at the end of the buffer + * we must be following a compressed archive. Only + * NUL and gzip headers are allowed. We skipped NUL. + */ + if (!(buf[total] == 0x1b && buf[total + 1] == 0x8b)) + cpio_error("unalinged junk in buffer"); + } + + this_buf = buf + total; + this_archive = 0; + + gunzip_start(stream, this_buf, len - total); + + do { + found = check_next_file(path, pathlen, stream, + &this_archive); + } while (!found); + + if (found > 0) + return found; + + if (stream->s.workspace) { + int discard; + + if (found < -1) { + cpio_error("Junk in compressed archive"); + } + + /* we either found EOF or TRAILER!!!. In the later + * case we need to discard to the end of the gzip + * contents. + */ + do { + discard = gunzip_partial(stream, name_buf, + MAX_PATH); + this_archive += discard; + } while (discard); + + /* + * Peek at how many input bytes were consumed. + * reset our consumed input. + */ + total += stream->s.total_in + 8; + + /* clean up zlib */ + discard = gunzip_finish(stream, name_buf, 0); + } else { + if (this_archive % 4) { + cpio_error("Archive not multiple of 4?"); + } + total += this_archive; + + if (!this_archive) { + cpio_error("junk between archives"); + } + /* don't check the < -1 of found, it might be an + * archive. This will be caught by the !this_archive + * check on the next loop. + */ + } + } while (total < len); + + return 0; +} + +void get_cpio_info(void **archive_start, int *consumed) +{ + *archive_start = this_buf; + *consumed = this_archive; +} + +int get_cpio_file_mode(void) +{ + return cpio_str_to_num(cpio_header_buf.mode); +} + +int get_cpio_file_size(void) +{ + return cpio_str_to_num(cpio_header_buf.filesize); +} Index: kernel/arch/powerpc/boot/cpio.h =================================================================== --- /dev/null 1970-01-01 00:00:00.000000000 +0000 +++ kernel/arch/powerpc/boot/cpio.h 2007-08-22 20:51:29.000000000 -0500 @@ -0,0 +1,12 @@ +#include "gunzip_util.h" + +extern int find_in_cpio(char *path, struct gunzip_state *stream); +extern int find_in_initramfs(char *path, char *buf, int len, + struct gunzip_state *state); +extern void get_cpio_info(void **archive_start, int *consumed); +extern int get_cpio_file_size(void); +extern int get_cpio_file_mode(void); + +#define MAX_PATH 256 + +extern void cpio_error(char *msg); Index: kernel/arch/powerpc/boot/Makefile =================================================================== --- kernel.orig/arch/powerpc/boot/Makefile 2007-08-22 20:51:06.000000000 -0500 +++ kernel/arch/powerpc/boot/Makefile 2007-08-22 20:52:46.000000000 -0500 @@ -36,13 +36,13 @@ $(obj)/ebony.o: BOOTCFLAGS += -mcpu=440 zlib := inffast.c inflate.c inftrees.c zlibheader := inffast.h inffixed.h inflate.h inftrees.h infutil.h -zliblinuxheader := zlib.h zconf.h zutil.h +zliblinuxheader := zlib.h zconf.h zutil.h stat.h $(addprefix $(obj)/,$(zlib) gunzip_util.o main.o): \ $(addprefix $(obj)/,$(zliblinuxheader)) $(addprefix $(obj)/,$(zlibheader)) src-wlib := string.S crt0.S stdio.c main.c flatdevtree.c flatdevtree_misc.c \ - flatdevtree_conv.c marshal.c memranges.c kexec.c rtas.c \ + flatdevtree_conv.c marshal.c memranges.c kexec.c rtas.c cpio.c \ ns16550.c serial.c simple_alloc.c div64.S util.S \ gunzip_util.c elf_util.c $(zlib) devtree.c oflib.c ofconsole.c \ 4xx.c ebony.c mv64x60.c mpsc.c mv64x60_i2c.c cuboot.c bamboo.c \ _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev