diff options
author | Denys Vlasenko <vda.linux@googlemail.com> | 2007-09-30 20:56:49 -0400 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2007-10-10 19:53:57 -0400 |
commit | 8336793baf962163c9fab5a3f39614295fdbab27 (patch) | |
tree | 6940426f8880928444e227d43085ca3d2fbfba80 | |
parent | b3448b0bde5f1a858397fe791f76632e978a1dc8 (diff) |
[ZLIB]: Move bnx2 driver gzip unpacker into zlib.
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
Acked-by: Michael Chan <mchan@broadcom.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | drivers/net/bnx2.c | 48 | ||||
-rw-r--r-- | include/linux/zlib.h | 6 | ||||
-rw-r--r-- | lib/zlib_inflate/inffast.c | 18 | ||||
-rw-r--r-- | lib/zlib_inflate/inflate.c | 55 | ||||
-rw-r--r-- | lib/zlib_inflate/inflate_syms.c | 1 |
5 files changed, 69 insertions, 59 deletions
diff --git a/drivers/net/bnx2.c b/drivers/net/bnx2.c index 73d4a579790c..6d6ea56fe384 100644 --- a/drivers/net/bnx2.c +++ b/drivers/net/bnx2.c | |||
@@ -2761,48 +2761,6 @@ bnx2_set_rx_mode(struct net_device *dev) | |||
2761 | spin_unlock_bh(&bp->phy_lock); | 2761 | spin_unlock_bh(&bp->phy_lock); |
2762 | } | 2762 | } |
2763 | 2763 | ||
2764 | /* To be moved to generic lib/ */ | ||
2765 | static int | ||
2766 | bnx2_gunzip(void *gunzip_buf, unsigned sz, u8 *zbuf, int len) | ||
2767 | { | ||
2768 | struct z_stream_s *strm; | ||
2769 | int rc; | ||
2770 | |||
2771 | /* gzip header (1f,8b,08... 10 bytes total + possible asciz filename) | ||
2772 | * is stripped */ | ||
2773 | |||
2774 | rc = -ENOMEM; | ||
2775 | strm = kmalloc(sizeof(*strm), GFP_KERNEL); | ||
2776 | if (strm == NULL) | ||
2777 | goto gunzip_nomem2; | ||
2778 | strm->workspace = kmalloc(zlib_inflate_workspacesize(), GFP_KERNEL); | ||
2779 | if (strm->workspace == NULL) | ||
2780 | goto gunzip_nomem3; | ||
2781 | |||
2782 | strm->next_in = zbuf; | ||
2783 | strm->avail_in = len; | ||
2784 | strm->next_out = gunzip_buf; | ||
2785 | strm->avail_out = sz; | ||
2786 | |||
2787 | rc = zlib_inflateInit2(strm, -MAX_WBITS); | ||
2788 | if (rc == Z_OK) { | ||
2789 | rc = zlib_inflate(strm, Z_FINISH); | ||
2790 | /* after Z_FINISH, only Z_STREAM_END is "we unpacked it all" */ | ||
2791 | if (rc == Z_STREAM_END) | ||
2792 | rc = sz - strm->avail_out; | ||
2793 | else | ||
2794 | rc = -EINVAL; | ||
2795 | zlib_inflateEnd(strm); | ||
2796 | } else | ||
2797 | rc = -EINVAL; | ||
2798 | |||
2799 | kfree(strm->workspace); | ||
2800 | gunzip_nomem3: | ||
2801 | kfree(strm); | ||
2802 | gunzip_nomem2: | ||
2803 | return rc; | ||
2804 | } | ||
2805 | |||
2806 | static void | 2764 | static void |
2807 | load_rv2p_fw(struct bnx2 *bp, u32 *rv2p_code, u32 rv2p_code_len, | 2765 | load_rv2p_fw(struct bnx2 *bp, u32 *rv2p_code, u32 rv2p_code_len, |
2808 | u32 rv2p_proc) | 2766 | u32 rv2p_proc) |
@@ -2858,7 +2816,7 @@ load_cpu_fw(struct bnx2 *bp, struct cpu_reg *cpu_reg, struct fw_info *fw) | |||
2858 | text = vmalloc(FW_BUF_SIZE); | 2816 | text = vmalloc(FW_BUF_SIZE); |
2859 | if (!text) | 2817 | if (!text) |
2860 | return -ENOMEM; | 2818 | return -ENOMEM; |
2861 | rc = bnx2_gunzip(text, FW_BUF_SIZE, fw->gz_text, fw->gz_text_len); | 2819 | rc = zlib_inflate_blob(text, FW_BUF_SIZE, fw->gz_text, fw->gz_text_len); |
2862 | if (rc < 0) { | 2820 | if (rc < 0) { |
2863 | vfree(text); | 2821 | vfree(text); |
2864 | return rc; | 2822 | return rc; |
@@ -2935,14 +2893,14 @@ bnx2_init_cpus(struct bnx2 *bp) | |||
2935 | text = vmalloc(FW_BUF_SIZE); | 2893 | text = vmalloc(FW_BUF_SIZE); |
2936 | if (!text) | 2894 | if (!text) |
2937 | return -ENOMEM; | 2895 | return -ENOMEM; |
2938 | rc = bnx2_gunzip(text, FW_BUF_SIZE, bnx2_rv2p_proc1, sizeof(bnx2_rv2p_proc1)); | 2896 | rc = zlib_inflate_blob(text, FW_BUF_SIZE, bnx2_rv2p_proc1, sizeof(bnx2_rv2p_proc1)); |
2939 | if (rc < 0) { | 2897 | if (rc < 0) { |
2940 | vfree(text); | 2898 | vfree(text); |
2941 | goto init_cpu_err; | 2899 | goto init_cpu_err; |
2942 | } | 2900 | } |
2943 | load_rv2p_fw(bp, text, rc /* == len */, RV2P_PROC1); | 2901 | load_rv2p_fw(bp, text, rc /* == len */, RV2P_PROC1); |
2944 | 2902 | ||
2945 | rc = bnx2_gunzip(text, FW_BUF_SIZE, bnx2_rv2p_proc2, sizeof(bnx2_rv2p_proc2)); | 2903 | rc = zlib_inflate_blob(text, FW_BUF_SIZE, bnx2_rv2p_proc2, sizeof(bnx2_rv2p_proc2)); |
2946 | if (rc < 0) { | 2904 | if (rc < 0) { |
2947 | vfree(text); | 2905 | vfree(text); |
2948 | goto init_cpu_err; | 2906 | goto init_cpu_err; |
diff --git a/include/linux/zlib.h b/include/linux/zlib.h index 9e3192a7dc6f..40c49cb3eb51 100644 --- a/include/linux/zlib.h +++ b/include/linux/zlib.h | |||
@@ -82,7 +82,7 @@ | |||
82 | struct internal_state; | 82 | struct internal_state; |
83 | 83 | ||
84 | typedef struct z_stream_s { | 84 | typedef struct z_stream_s { |
85 | Byte *next_in; /* next input byte */ | 85 | const Byte *next_in; /* next input byte */ |
86 | uInt avail_in; /* number of bytes available at next_in */ | 86 | uInt avail_in; /* number of bytes available at next_in */ |
87 | uLong total_in; /* total nb of input bytes read so far */ | 87 | uLong total_in; /* total nb of input bytes read so far */ |
88 | 88 | ||
@@ -699,4 +699,8 @@ extern int zlib_inflateInit2(z_streamp strm, int windowBits); | |||
699 | struct internal_state {int dummy;}; /* hack for buggy compilers */ | 699 | struct internal_state {int dummy;}; /* hack for buggy compilers */ |
700 | #endif | 700 | #endif |
701 | 701 | ||
702 | /* Utility function: initialize zlib, unpack binary blob, clean up zlib, | ||
703 | * return len or negative error code. */ | ||
704 | extern int zlib_inflate_blob(void *dst, unsigned dst_sz, const void *src, unsigned src_sz); | ||
705 | |||
702 | #endif /* _ZLIB_H */ | 706 | #endif /* _ZLIB_H */ |
diff --git a/lib/zlib_inflate/inffast.c b/lib/zlib_inflate/inffast.c index d84560c076d8..8550b0c05d00 100644 --- a/lib/zlib_inflate/inffast.c +++ b/lib/zlib_inflate/inffast.c | |||
@@ -69,22 +69,22 @@ | |||
69 | void inflate_fast(z_streamp strm, unsigned start) | 69 | void inflate_fast(z_streamp strm, unsigned start) |
70 | { | 70 | { |
71 | struct inflate_state *state; | 71 | struct inflate_state *state; |
72 | unsigned char *in; /* local strm->next_in */ | 72 | const unsigned char *in; /* local strm->next_in */ |
73 | unsigned char *last; /* while in < last, enough input available */ | 73 | const unsigned char *last; /* while in < last, enough input available */ |
74 | unsigned char *out; /* local strm->next_out */ | 74 | unsigned char *out; /* local strm->next_out */ |
75 | unsigned char *beg; /* inflate()'s initial strm->next_out */ | 75 | unsigned char *beg; /* inflate()'s initial strm->next_out */ |
76 | unsigned char *end; /* while out < end, enough space available */ | 76 | unsigned char *end; /* while out < end, enough space available */ |
77 | #ifdef INFLATE_STRICT | 77 | #ifdef INFLATE_STRICT |
78 | unsigned dmax; /* maximum distance from zlib header */ | 78 | unsigned dmax; /* maximum distance from zlib header */ |
79 | #endif | 79 | #endif |
80 | unsigned wsize; /* window size or zero if not using window */ | 80 | unsigned wsize; /* window size or zero if not using window */ |
81 | unsigned whave; /* valid bytes in the window */ | 81 | unsigned whave; /* valid bytes in the window */ |
82 | unsigned write; /* window write index */ | 82 | unsigned write; /* window write index */ |
83 | unsigned char *window; /* allocated sliding window, if wsize != 0 */ | 83 | unsigned char *window; /* allocated sliding window, if wsize != 0 */ |
84 | unsigned long hold; /* local strm->hold */ | 84 | unsigned long hold; /* local strm->hold */ |
85 | unsigned bits; /* local strm->bits */ | 85 | unsigned bits; /* local strm->bits */ |
86 | code const *lcode; /* local strm->lencode */ | 86 | code const *lcode; /* local strm->lencode */ |
87 | code const *dcode; /* local strm->distcode */ | 87 | code const *dcode; /* local strm->distcode */ |
88 | unsigned lmask; /* mask for first level of length codes */ | 88 | unsigned lmask; /* mask for first level of length codes */ |
89 | unsigned dmask; /* mask for first level of distance codes */ | 89 | unsigned dmask; /* mask for first level of distance codes */ |
90 | code this; /* retrieved table entry */ | 90 | code this; /* retrieved table entry */ |
@@ -92,7 +92,7 @@ void inflate_fast(z_streamp strm, unsigned start) | |||
92 | /* window position, window bytes to copy */ | 92 | /* window position, window bytes to copy */ |
93 | unsigned len; /* match length, unused bytes */ | 93 | unsigned len; /* match length, unused bytes */ |
94 | unsigned dist; /* match distance */ | 94 | unsigned dist; /* match distance */ |
95 | unsigned char *from; /* where to copy match from */ | 95 | unsigned char *from; /* where to copy match from */ |
96 | 96 | ||
97 | /* copy state to local variables */ | 97 | /* copy state to local variables */ |
98 | state = (struct inflate_state *)strm->state; | 98 | state = (struct inflate_state *)strm->state; |
diff --git a/lib/zlib_inflate/inflate.c b/lib/zlib_inflate/inflate.c index 7e1e3114a73e..0ad1ebf00947 100644 --- a/lib/zlib_inflate/inflate.c +++ b/lib/zlib_inflate/inflate.c | |||
@@ -332,14 +332,14 @@ static int zlib_inflateSyncPacket(z_streamp strm) | |||
332 | int zlib_inflate(z_streamp strm, int flush) | 332 | int zlib_inflate(z_streamp strm, int flush) |
333 | { | 333 | { |
334 | struct inflate_state *state; | 334 | struct inflate_state *state; |
335 | unsigned char *next; /* next input */ | 335 | const unsigned char *next; /* next input */ |
336 | unsigned char *put; /* next output */ | 336 | unsigned char *put; /* next output */ |
337 | unsigned have, left; /* available input and output */ | 337 | unsigned have, left; /* available input and output */ |
338 | unsigned long hold; /* bit buffer */ | 338 | unsigned long hold; /* bit buffer */ |
339 | unsigned bits; /* bits in bit buffer */ | 339 | unsigned bits; /* bits in bit buffer */ |
340 | unsigned in, out; /* save starting available input and output */ | 340 | unsigned in, out; /* save starting available input and output */ |
341 | unsigned copy; /* number of stored or match bytes to copy */ | 341 | unsigned copy; /* number of stored or match bytes to copy */ |
342 | unsigned char *from; /* where to copy match bytes from */ | 342 | unsigned char *from; /* where to copy match bytes from */ |
343 | code this; /* current decoding table entry */ | 343 | code this; /* current decoding table entry */ |
344 | code last; /* parent table entry */ | 344 | code last; /* parent table entry */ |
345 | unsigned len; /* length to copy for repeats, bits to drop */ | 345 | unsigned len; /* length to copy for repeats, bits to drop */ |
@@ -897,7 +897,7 @@ int zlib_inflateIncomp(z_stream *z) | |||
897 | 897 | ||
898 | /* Setup some variables to allow misuse of updateWindow */ | 898 | /* Setup some variables to allow misuse of updateWindow */ |
899 | z->avail_out = 0; | 899 | z->avail_out = 0; |
900 | z->next_out = z->next_in + z->avail_in; | 900 | z->next_out = (unsigned char*)z->next_in + z->avail_in; |
901 | 901 | ||
902 | zlib_updatewindow(z, z->avail_in); | 902 | zlib_updatewindow(z, z->avail_in); |
903 | 903 | ||
@@ -916,3 +916,50 @@ int zlib_inflateIncomp(z_stream *z) | |||
916 | 916 | ||
917 | return Z_OK; | 917 | return Z_OK; |
918 | } | 918 | } |
919 | |||
920 | #include <linux/errno.h> | ||
921 | #include <linux/slab.h> | ||
922 | #include <linux/vmalloc.h> | ||
923 | |||
924 | /* Utility function: initialize zlib, unpack binary blob, clean up zlib, | ||
925 | * return len or negative error code. */ | ||
926 | int zlib_inflate_blob(void *gunzip_buf, unsigned sz, const void *buf, unsigned len) | ||
927 | { | ||
928 | const u8 *zbuf = buf; | ||
929 | struct z_stream_s *strm; | ||
930 | int rc; | ||
931 | |||
932 | rc = -ENOMEM; | ||
933 | strm = kmalloc(sizeof(*strm), GFP_KERNEL); | ||
934 | if (strm == NULL) | ||
935 | goto gunzip_nomem1; | ||
936 | strm->workspace = kmalloc(zlib_inflate_workspacesize(), GFP_KERNEL); | ||
937 | if (strm->workspace == NULL) | ||
938 | goto gunzip_nomem2; | ||
939 | |||
940 | /* gzip header (1f,8b,08... 10 bytes total + possible asciz filename) | ||
941 | * expected to be stripped from input */ | ||
942 | |||
943 | strm->next_in = zbuf; | ||
944 | strm->avail_in = len; | ||
945 | strm->next_out = gunzip_buf; | ||
946 | strm->avail_out = sz; | ||
947 | |||
948 | rc = zlib_inflateInit2(strm, -MAX_WBITS); | ||
949 | if (rc == Z_OK) { | ||
950 | rc = zlib_inflate(strm, Z_FINISH); | ||
951 | /* after Z_FINISH, only Z_STREAM_END is "we unpacked it all" */ | ||
952 | if (rc == Z_STREAM_END) | ||
953 | rc = sz - strm->avail_out; | ||
954 | else | ||
955 | rc = -EINVAL; | ||
956 | zlib_inflateEnd(strm); | ||
957 | } else | ||
958 | rc = -EINVAL; | ||
959 | |||
960 | kfree(strm->workspace); | ||
961 | gunzip_nomem2: | ||
962 | kfree(strm); | ||
963 | gunzip_nomem1: | ||
964 | return rc; /* returns Z_OK (0) if successful */ | ||
965 | } | ||
diff --git a/lib/zlib_inflate/inflate_syms.c b/lib/zlib_inflate/inflate_syms.c index 2061d4f06765..67329fe9907e 100644 --- a/lib/zlib_inflate/inflate_syms.c +++ b/lib/zlib_inflate/inflate_syms.c | |||
@@ -16,4 +16,5 @@ EXPORT_SYMBOL(zlib_inflateInit2); | |||
16 | EXPORT_SYMBOL(zlib_inflateEnd); | 16 | EXPORT_SYMBOL(zlib_inflateEnd); |
17 | EXPORT_SYMBOL(zlib_inflateReset); | 17 | EXPORT_SYMBOL(zlib_inflateReset); |
18 | EXPORT_SYMBOL(zlib_inflateIncomp); | 18 | EXPORT_SYMBOL(zlib_inflateIncomp); |
19 | EXPORT_SYMBOL(zlib_inflate_blob); | ||
19 | MODULE_LICENSE("GPL"); | 20 | MODULE_LICENSE("GPL"); |