19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2006 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.os;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1990619816d99154d504a14774c6f2d5f4254ff780Guang Zhuimport java.io.BufferedInputStream;
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.ByteArrayOutputStream;
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.File;
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.FileInputStream;
236d25a990afffd5eb385aba3043d5dfad36f1539aWink Savilleimport java.io.FileNotFoundException;
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.FileOutputStream;
25da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwoodimport java.io.FileWriter;
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.IOException;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.InputStream;
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.regex.Pattern;
296d25a990afffd5eb385aba3043d5dfad36f1539aWink Savilleimport java.util.zip.CRC32;
306d25a990afffd5eb385aba3043d5dfad36f1539aWink Savilleimport java.util.zip.CheckedInputStream;
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Tools for managing files.  Not for public consumption.
349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @hide
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
366d25a990afffd5eb385aba3043d5dfad36f1539aWink Savillepublic class FileUtils {
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IRWXU = 00700;
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IRUSR = 00400;
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IWUSR = 00200;
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IXUSR = 00100;
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IRWXG = 00070;
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IRGRP = 00040;
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IWGRP = 00020;
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IXGRP = 00010;
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IRWXO = 00007;
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IROTH = 00004;
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IWOTH = 00002;
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int S_IXOTH = 00001;
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** Regular expression for safe filenames: no spaces or metacharacters */
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private static final Pattern SAFE_FILENAME_PATTERN = Pattern.compile("[\\w%+,./=_-]+");
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static native int setPermissions(String file, int mode, int uid, int gid);
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /** returns the FAT file system volume ID for the volume mounted
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * at the given mount point, or -1 for failure
596d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     * @param mountPoint point for FAT volume
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return volume ID or -1
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static native int getFatVolumeId(String mountPoint);
638bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn
648bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn    /**
658bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn     * Perform an fsync on the given FileOutputStream.  The stream at this
668bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn     * point must be flushed but not yet closed.
678bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn     */
688bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn    public static boolean sync(FileOutputStream stream) {
698bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn        try {
708bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn            if (stream != null) {
718bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                stream.getFD().sync();
728bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn            }
738bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn            return true;
748bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn        } catch (IOException e) {
758bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn        }
768bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn        return false;
778bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn    }
788bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // copy a file from srcFile to destFile, return true if succeed, return
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // false if fail
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static boolean copyFile(File srcFile, File destFile) {
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        boolean result = false;
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        try {
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            InputStream in = new FileInputStream(srcFile);
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            try {
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                result = copyToFile(in, destFile);
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } finally  {
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                in.close();
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } catch (IOException e) {
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            result = false;
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return result;
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9590619816d99154d504a14774c6f2d5f4254ff780Guang Zhu
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Copy data from a source stream to destFile.
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Return true if succeed, return false if failed.
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static boolean copyToFile(InputStream inputStream, File destFile) {
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        try {
1021afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn            if (destFile.exists()) {
1031afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn                destFile.delete();
1041afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn            }
1058bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn            FileOutputStream out = new FileOutputStream(destFile);
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            try {
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                byte[] buffer = new byte[4096];
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int bytesRead;
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                while ((bytesRead = inputStream.read(buffer)) >= 0) {
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    out.write(buffer, 0, bytesRead);
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } finally {
1138bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                out.flush();
1148bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                try {
1158bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                    out.getFD().sync();
1168bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                } catch (IOException e) {
1178bdf5935c0db4a66ab33a10b43398d2523cfa15dDianne Hackborn                }
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                out.close();
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } catch (IOException e) {
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return false;
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Check if a filename is "safe" (no metacharacters or spaces).
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param file  The file to check
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static boolean isFilenameSafe(File file) {
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Note, we check whether it matches what's known to be safe,
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // rather than what's known to be unsafe.  Non-ASCII, control
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // characters, etc. are all unsafe by default.
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return SAFE_FILENAME_PATTERN.matcher(file.getPath()).matches();
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Read a text file into a String, optionally limiting the length.
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param file to read (will not seek, so things like /proc files are OK)
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param max length (positive for head, negative of tail, 0 for no limit)
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param ellipsis to add of the file was truncated (can be null)
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return the contents of the file, possibly truncated
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @throws IOException if something goes wrong reading the file
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static String readTextFile(File file, int max, String ellipsis) throws IOException {
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        InputStream input = new FileInputStream(file);
14790619816d99154d504a14774c6f2d5f4254ff780Guang Zhu        // wrapping a BufferedInputStream around it because when reading /proc with unbuffered
14890619816d99154d504a14774c6f2d5f4254ff780Guang Zhu        // input stream, bytes read not equal to buffer size is not necessarily the correct
14990619816d99154d504a14774c6f2d5f4254ff780Guang Zhu        // indication for EOF; but it is true for BufferedInputStream due to its implementation.
15090619816d99154d504a14774c6f2d5f4254ff780Guang Zhu        BufferedInputStream bis = new BufferedInputStream(input);
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        try {
15242471dd5552a346dd82a58a663159875ccc4fb79Dan Egnor            long size = file.length();
15342471dd5552a346dd82a58a663159875ccc4fb79Dan Egnor            if (max > 0 || (size > 0 && max == 0)) {  // "head" mode: read the first N bytes
15442471dd5552a346dd82a58a663159875ccc4fb79Dan Egnor                if (size > 0 && (max == 0 || size < max)) max = (int) size;
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                byte[] data = new byte[max + 1];
15690619816d99154d504a14774c6f2d5f4254ff780Guang Zhu                int length = bis.read(data);
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (length <= 0) return "";
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (length <= max) return new String(data, 0, length);
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (ellipsis == null) return new String(data, 0, max);
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return new String(data, 0, max) + ellipsis;
16142471dd5552a346dd82a58a663159875ccc4fb79Dan Egnor            } else if (max < 0) {  // "tail" mode: keep the last N
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int len;
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                boolean rolled = false;
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                byte[] last = null, data = null;
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                do {
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (last != null) rolled = true;
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    byte[] tmp = last; last = data; data = tmp;
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (data == null) data = new byte[-max];
16990619816d99154d504a14774c6f2d5f4254ff780Guang Zhu                    len = bis.read(data);
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } while (len == data.length);
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (last == null && len <= 0) return "";
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (last == null) return new String(data, 0, len);
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (len > 0) {
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    rolled = true;
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    System.arraycopy(last, len, last, 0, last.length - len);
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    System.arraycopy(data, 0, last, last.length - len, len);
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (ellipsis == null || !rolled) return new String(last);
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return ellipsis + new String(last);
18142471dd5552a346dd82a58a663159875ccc4fb79Dan Egnor            } else {  // "cat" mode: size unknown, read it all in streaming fashion
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                ByteArrayOutputStream contents = new ByteArrayOutputStream();
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int len;
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                byte[] data = new byte[1024];
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                do {
18690619816d99154d504a14774c6f2d5f4254ff780Guang Zhu                    len = bis.read(data);
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (len > 0) contents.write(data, 0, len);
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } while (len == data.length);
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return contents.toString();
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } finally {
19290619816d99154d504a14774c6f2d5f4254ff780Guang Zhu            bis.close();
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            input.close();
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
196da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood
197da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood   /**
198da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     * Writes string to file. Basically same as "echo -n $string > $filename"
199da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     *
200da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     * @param filename
201da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     * @param string
202da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     * @throws IOException
203da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood     */
204da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood    public static void stringToFile(String filename, String string) throws IOException {
205da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood        FileWriter out = new FileWriter(filename);
206da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood        try {
207da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood            out.write(string);
208da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood        } finally {
209da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood            out.close();
210da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood        }
211da8bb74b9d9ffcb095815db800d0816c411f1fbaMike Lockwood    }
2121b9a6a6e58fd73b5d1b6a434d17f0a69806858ecWink Saville
2136d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville    /**
2146d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     * Computes the checksum of a file using the CRC32 checksum routine.
2156d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     * The value of the checksum is returned.
2166d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     *
2176d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     * @param file  the file to checksum, must not be null
2186d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     * @return the checksum value or an exception is thrown.
2196d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville     */
2206d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville    public static long checksumCrc32(File file) throws FileNotFoundException, IOException {
2216d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville        CRC32 checkSummer = new CRC32();
2226d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville        CheckedInputStream cis = null;
2236d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville
2246d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville        try {
2256d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            cis = new CheckedInputStream( new FileInputStream(file), checkSummer);
2266d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            byte[] buf = new byte[128];
2276d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            while(cis.read(buf) >= 0) {
2286d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville                // Just read for checksum to get calculated.
2296d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            }
2306d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            return checkSummer.getValue();
2316d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville        } finally {
2326d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            if (cis != null) {
2336d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville                try {
2346d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville                    cis.close();
2356d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville                } catch (IOException e) {
2366d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville                }
2376d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville            }
2386d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville        }
2396d25a990afffd5eb385aba3043d5dfad36f1539aWink Saville    }
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
241