v2.4.9.9 -> v2.4.9.10
[opensuse:kernel.git] / fs / jffs2 / compr.c
1 /*
2  * JFFS2 -- Journalling Flash File System, Version 2.
3  *
4  * Copyright (C) 2001 Red Hat, Inc.
5  *
6  * Created by Arjan van de Ven <arjanv@redhat.com>
7  *
8  * The original JFFS, from which the design for JFFS2 was derived,
9  * was designed and implemented by Axis Communications AB.
10  *
11  * The contents of this file are subject to the Red Hat eCos Public
12  * License Version 1.1 (the "Licence"); you may not use this file
13  * except in compliance with the Licence.  You may obtain a copy of
14  * the Licence at http://www.redhat.com/
15  *
16  * Software distributed under the Licence is distributed on an "AS IS"
17  * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied.
18  * See the Licence for the specific language governing rights and
19  * limitations under the Licence.
20  *
21  * The Original Code is JFFS2 - Journalling Flash File System, version 2
22  *
23  * Alternatively, the contents of this file may be used under the
24  * terms of the GNU General Public License version 2 (the "GPL"), in
25  * which case the provisions of the GPL are applicable instead of the
26  * above.  If you wish to allow the use of your version of this file
27  * only under the terms of the GPL and not to allow others to use your
28  * version of this file under the RHEPL, indicate your decision by
29  * deleting the provisions above and replace them with the notice and
30  * other provisions required by the GPL.  If you do not delete the
31  * provisions above, a recipient may use your version of this file
32  * under either the RHEPL or the GPL.
33  *
34  * $Id: compr.c,v 1.16 2001/03/15 15:38:23 dwmw2 Exp $
35  *
36  */
37
38 #include <linux/kernel.h>
39 #include <linux/string.h>
40 #include <linux/types.h>
41 #include <linux/errno.h>
42 #include <linux/jffs2.h>
43
44 int zlib_compress(unsigned char *data_in, unsigned char *cpage_out, __u32 *sourcelen, __u32 *dstlen);
45 void zlib_decompress(unsigned char *data_in, unsigned char *cpage_out, __u32 srclen, __u32 destlen);
46 int rtime_compress(unsigned char *data_in, unsigned char *cpage_out, __u32 *sourcelen, __u32 *dstlen);
47 void rtime_decompress(unsigned char *data_in, unsigned char *cpage_out, __u32 srclen, __u32 destlen);
48 int rubinmips_compress(unsigned char *data_in, unsigned char *cpage_out, __u32 *sourcelen, __u32 *dstlen);
49 void rubinmips_decompress(unsigned char *data_in, unsigned char *cpage_out, __u32 srclen, __u32 destlen);
50 int dynrubin_compress(unsigned char *data_in, unsigned char *cpage_out, __u32 *sourcelen, __u32 *dstlen);
51 void dynrubin_decompress(unsigned char *data_in, unsigned char *cpage_out, __u32 srclen, __u32 destlen);
52
53
54 /* jffs2_compress:
55  * @data: Pointer to uncompressed data
56  * @cdata: Pointer to buffer for compressed data
57  * @datalen: On entry, holds the amount of data available for compression.
58  *      On exit, expected to hold the amount of data actually compressed.
59  * @cdatalen: On entry, holds the amount of space available for compressed
60  *      data. On exit, expected to hold the actual size of the compressed
61  *      data.
62  *
63  * Returns: Byte to be stored with data indicating compression type used.
64  * Zero is used to show that the data could not be compressed - the 
65  * compressed version was actually larger than the original.
66  *
67  * If the cdata buffer isn't large enough to hold all the uncompressed data,
68  * jffs2_compress should compress as much as will fit, and should set 
69  * *datalen accordingly to show the amount of data which were compressed.
70  */
71 unsigned char jffs2_compress(unsigned char *data_in, unsigned char *cpage_out, 
72                     __u32 *datalen, __u32 *cdatalen)
73 {
74         int ret;
75
76         ret = zlib_compress(data_in, cpage_out, datalen, cdatalen);
77         if (!ret) {
78                 return JFFS2_COMPR_ZLIB;
79         }
80
81         ret = dynrubin_compress(data_in, cpage_out, datalen, cdatalen);
82         if (!ret) {
83                 return JFFS2_COMPR_DYNRUBIN;
84         }
85
86 #if 0 /* Phase this one out */
87         ret = rubinmips_compress(data_in, cpage_out, datalen, cdatalen);
88         if (!ret) {
89                 return JFFS2_COMPR_RUBINMIPS;
90         }
91 #endif
92         ret = rtime_compress(data_in, cpage_out, datalen, cdatalen);
93         if (!ret) {
94                 return JFFS2_COMPR_RTIME;
95         }
96 #if 0
97         /* We don't need to copy. Let the caller special-case the COMPR_NONE case. */
98         /* If we get here, no compression is going to work */
99         /* But we might want to use the fragmentation part -- Arjan */
100         memcpy(cpage_out,data_in,min(*datalen,*cdatalen));
101         if (*datalen > *cdatalen)
102                 *datalen = *cdatalen;
103 #endif          
104         return JFFS2_COMPR_NONE; /* We failed to compress */
105
106 }
107
108
109 int jffs2_decompress(unsigned char comprtype, unsigned char *cdata_in, 
110                      unsigned char *data_out, __u32 cdatalen, __u32 datalen)
111 {
112         switch (comprtype) {
113         case JFFS2_COMPR_NONE:
114                 /* This should be special-cased elsewhere, but we might as well deal with it */
115                 memcpy(data_out, cdata_in, datalen);
116                 break;
117
118         case JFFS2_COMPR_ZERO:
119                 memset(data_out, 0, datalen);
120                 break;
121
122         case JFFS2_COMPR_ZLIB:
123                 zlib_decompress(cdata_in, data_out, cdatalen, datalen);
124                 break;
125
126         case JFFS2_COMPR_RTIME:
127                 rtime_decompress(cdata_in, data_out, cdatalen, datalen);
128                 break;
129 #if 1 /* Phase this one out */
130         case JFFS2_COMPR_RUBINMIPS:
131                 rubinmips_decompress(cdata_in, data_out, cdatalen, datalen);
132                 break;
133 #endif
134         case JFFS2_COMPR_DYNRUBIN:
135                 dynrubin_decompress(cdata_in, data_out, cdatalen, datalen);
136                 break;
137
138         default:
139                 printk(KERN_NOTICE "Unknown JFFS2 compression type 0x%02x\n", comprtype);
140                 return -EIO;
141         }
142         return 0;
143 }