/*
* Copyright (C) 2006 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.content.res;
import static android.system.OsConstants.S_ISFIFO;
import static android.system.OsConstants.S_ISSOCK;
import android.compat.annotation.UnsupportedAppUsage;
import android.os.Bundle;
import android.os.Parcel;
import android.os.ParcelFileDescriptor;
import android.os.Parcelable;
import android.system.ErrnoException;
import android.system.Os;
import android.system.StructStat;
import java.io.Closeable;
import java.io.FileDescriptor;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.nio.channels.ReadableByteChannel;
import java.nio.channels.WritableByteChannel;
/**
* File descriptor of an entry in the AssetManager. This provides your own
* opened FileDescriptor that can be used to read the data, as well as the
* offset and length of that entry's data in the file.
*/
public class AssetFileDescriptor implements Parcelable, Closeable {
/**
* Length used with {@link #AssetFileDescriptor(ParcelFileDescriptor, long, long)}
* and {@link #getDeclaredLength} when a length has not been declared. This means
* the data extends to the end of the file.
*/
public static final long UNKNOWN_LENGTH = -1;
@UnsupportedAppUsage
private final ParcelFileDescriptor mFd;
@UnsupportedAppUsage
private final long mStartOffset;
@UnsupportedAppUsage
private final long mLength;
private final Bundle mExtras;
/**
* Create a new AssetFileDescriptor from the given values.
*
* @param fd The underlying file descriptor.
* @param startOffset The location within the file that the asset starts.
* This must be 0 if length is UNKNOWN_LENGTH.
* @param length The number of bytes of the asset, or
* {@link #UNKNOWN_LENGTH} if it extends to the end of the file.
*/
public AssetFileDescriptor(ParcelFileDescriptor fd, long startOffset,
long length) {
this(fd, startOffset, length, null);
}
/**
* Create a new AssetFileDescriptor from the given values.
*
* @param fd The underlying file descriptor.
* @param startOffset The location within the file that the asset starts.
* This must be 0 if length is UNKNOWN_LENGTH.
* @param length The number of bytes of the asset, or
* {@link #UNKNOWN_LENGTH} if it extends to the end of the file.
* @param extras additional details that can be used to interpret the
* underlying file descriptor. May be null.
*/
public AssetFileDescriptor(ParcelFileDescriptor fd, long startOffset,
long length, Bundle extras) {
if (fd == null) {
throw new IllegalArgumentException("fd must not be null");
}
if (length < 0 && startOffset != 0) {
throw new IllegalArgumentException(
"startOffset must be 0 when using UNKNOWN_LENGTH");
}
mFd = fd;
mStartOffset = startOffset;
mLength = length;
mExtras = extras;
}
/**
* The AssetFileDescriptor contains its own ParcelFileDescriptor, which
* in addition to the normal FileDescriptor object also allows you to close
* the descriptor when you are done with it.
*/
public ParcelFileDescriptor getParcelFileDescriptor() {
return mFd;
}
/**
* Returns the FileDescriptor that can be used to read the data in the
* file.
*/
public FileDescriptor getFileDescriptor() {
return mFd.getFileDescriptor();
}
/**
* Returns the byte offset where this asset entry's data starts.
*/
public long getStartOffset() {
return mStartOffset;
}
/**
* Returns any additional details that can be used to interpret the
* underlying file descriptor. May be null.
*/
public Bundle getExtras() {
return mExtras;
}
/**
* Returns the total number of bytes of this asset entry's data. May be
* {@link #UNKNOWN_LENGTH} if the asset extends to the end of the file.
* If the AssetFileDescriptor was constructed with {@link #UNKNOWN_LENGTH},
* this will use {@link ParcelFileDescriptor#getStatSize()
* ParcelFileDescriptor.getStatSize()} to find the total size of the file,
* returning that number if found or {@link #UNKNOWN_LENGTH} if it could
* not be determined.
*
* @see #getDeclaredLength()
*/
public long getLength() {
if (mLength >= 0) {
return mLength;
}
long len = mFd.getStatSize();
return len >= 0 ? len : UNKNOWN_LENGTH;
}
/**
* Return the actual number of bytes that were declared when the
* AssetFileDescriptor was constructed. Will be
* {@link #UNKNOWN_LENGTH} if the length was not declared, meaning data
* should be read to the end of the file.
*
* @see #getDeclaredLength()
*/
public long getDeclaredLength() {
return mLength;
}
/**
* Convenience for calling getParcelFileDescriptor().close()
.
*/
@Override
public void close() throws IOException {
mFd.close();
}
/**
* Create and return a new auto-close input stream for this asset. This
* will either return a full asset {@link AutoCloseInputStream}, or
* an underlying {@link ParcelFileDescriptor.AutoCloseInputStream
* ParcelFileDescriptor.AutoCloseInputStream} depending on whether the
* the object represents a complete file or sub-section of a file. You
* should only call this once for a particular asset.
*/
public FileInputStream createInputStream() throws IOException {
if (mLength < 0) {
return new ParcelFileDescriptor.AutoCloseInputStream(mFd);
}
return new AutoCloseInputStream(this);
}
/**
* Create and return a new auto-close output stream for this asset. This
* will either return a full asset {@link AutoCloseOutputStream}, or
* an underlying {@link ParcelFileDescriptor.AutoCloseOutputStream
* ParcelFileDescriptor.AutoCloseOutputStream} depending on whether the
* the object represents a complete file or sub-section of a file. You
* should only call this once for a particular asset.
*/
public FileOutputStream createOutputStream() throws IOException {
if (mLength < 0) {
return new ParcelFileDescriptor.AutoCloseOutputStream(mFd);
}
return new AutoCloseOutputStream(this);
}
@Override
public String toString() {
return "{AssetFileDescriptor: " + mFd
+ " start=" + mStartOffset + " len=" + mLength + "}";
}
/**
* An InputStream you can create on a ParcelFileDescriptor, which will
* take care of calling {@link ParcelFileDescriptor#close
* ParcelFileDescriptor.close()} for you when the stream is closed.
* It has a ParcelFileDescriptor.AutoCloseInputStream member to make delegate calls
* and during definition it will create seekable or non seekable child object
* AssetFileDescriptor.AutoCloseInputStream depends on the type of file descriptor
* to provide different solution.
*/
public static class AutoCloseInputStream
extends ParcelFileDescriptor.AutoCloseInputStream {
private ParcelFileDescriptor.AutoCloseInputStream mDelegateInputStream;
public AutoCloseInputStream(AssetFileDescriptor fd) throws IOException {
super(fd.getParcelFileDescriptor());
StructStat ss;
try {
ss = Os.fstat(fd.getParcelFileDescriptor().getFileDescriptor());
} catch (ErrnoException e) {
throw new IOException(e);
}
if (S_ISSOCK(ss.st_mode) || S_ISFIFO(ss.st_mode)) {
mDelegateInputStream = new NonSeekableAutoCloseInputStream(fd);
} else {
mDelegateInputStream = new SeekableAutoCloseInputStream(fd);
}
}
@Override
public int available() throws IOException {
return mDelegateInputStream.available();
}
@Override
public int read() throws IOException {
return mDelegateInputStream.read();
}
@Override
public int read(byte[] buffer, int offset, int count) throws IOException {
return mDelegateInputStream.read(buffer, offset, count);
}
@Override
public int read(byte[] buffer) throws IOException {
return mDelegateInputStream.read(buffer);
}
@Override
public long skip(long count) throws IOException {
return mDelegateInputStream.skip(count);
}
@Override
public void mark(int readlimit) {
mDelegateInputStream.mark(readlimit);
}
@Override
public boolean markSupported() {
return mDelegateInputStream.markSupported();
}
@Override
public synchronized void reset() throws IOException {
mDelegateInputStream.reset();
}
@Override
public FileChannel getChannel() {
return mDelegateInputStream.getChannel();
}
@Override
public void close() throws IOException {
// Make the mDelegateInputStream own file descriptor and super.close()
// is not needed here to avoid double close the file descriptor.
mDelegateInputStream.close();
}
}
/**
* An InputStream you can create on a non seekable file descriptor,
* like PIPE, SOCKET and FIFO, which will take care of calling
* {@link ParcelFileDescriptor#close ParcelFileDescriptor.close()}
* for you when the stream is closed.
*/
private static class NonSeekableAutoCloseInputStream
extends ParcelFileDescriptor.AutoCloseInputStream {
private long mRemaining;
NonSeekableAutoCloseInputStream(AssetFileDescriptor fd) throws IOException {
super(fd.getParcelFileDescriptor());
super.skip(fd.getStartOffset());
mRemaining = (int) fd.getLength();
}
@Override
public int available() throws IOException {
return mRemaining >= 0
? (mRemaining < 0x7fffffff ? (int) mRemaining : 0x7fffffff)
: super.available();
}
@Override
public int read() throws IOException {
byte[] buffer = new byte[1];
int result = read(buffer, 0, 1);
return result == -1 ? -1 : buffer[0] & 0xff;
}
@Override
public int read(byte[] buffer, int offset, int count) throws IOException {
if (mRemaining >= 0) {
if (mRemaining == 0) return -1;
if (count > mRemaining) count = (int) mRemaining;
int res = super.read(buffer, offset, count);
if (res >= 0) mRemaining -= res;
return res;
}
return super.read(buffer, offset, count);
}
@Override
public int read(byte[] buffer) throws IOException {
return read(buffer, 0, buffer.length);
}
@Override
public long skip(long count) throws IOException {
if (mRemaining >= 0) {
if (mRemaining == 0) return -1;
if (count > mRemaining) count = mRemaining;
long res = super.skip(count);
if (res >= 0) mRemaining -= res;
return res;
}
return super.skip(count);
}
@Override
public void mark(int readlimit) {
if (mRemaining >= 0) {
// Not supported.
return;
}
super.mark(readlimit);
}
@Override
public boolean markSupported() {
if (mRemaining >= 0) {
return false;
}
return super.markSupported();
}
@Override
public synchronized void reset() throws IOException {
if (mRemaining >= 0) {
// Not supported.
return;
}
super.reset();
}
}
/**
* An InputStream you can create on a seekable file descriptor, which means
* you can use pread to read from a specific offset, this will take care of
* calling {@link ParcelFileDescriptor#close ParcelFileDescriptor.close()}
* for you when the stream is closed.
*/
private static class SeekableAutoCloseInputStream
extends ParcelFileDescriptor.AutoCloseInputStream {
/** Size of current file. */
private long mTotalSize;
/** The absolute position of current file start point. */
private final long mFileOffset;
/** The relative position where input stream is against mFileOffset. */
private long mOffset;
private OffsetCorrectFileChannel mOffsetCorrectFileChannel;
SeekableAutoCloseInputStream(AssetFileDescriptor fd) throws IOException {
super(fd.getParcelFileDescriptor());
mTotalSize = fd.getLength();
mFileOffset = fd.getStartOffset();
}
@Override
public int available() throws IOException {
long available = mTotalSize - mOffset;
return available >= 0
? (available < 0x7fffffff ? (int) available : 0x7fffffff)
: 0;
}
@Override
public int read() throws IOException {
byte[] buffer = new byte[1];
int result = read(buffer, 0, 1);
return result == -1 ? -1 : buffer[0] & 0xff;
}
@Override
public int read(byte[] buffer, int offset, int count) throws IOException {
int available = available();
if (available <= 0) {
return -1;
}
if (count == 0) {
// Java's InputStream explicitly specifies that this returns zero.
return 0;
}
if (count > available) count = available;
try {
int res = Os.pread(getFD(), buffer, offset, count, mFileOffset + mOffset);
// pread returns 0 at end of file, while java's InputStream interface requires -1
if (res == 0) res = -1;
if (res > 0) {
mOffset += res;
updateChannelPosition(mOffset + mFileOffset);
}
return res;
} catch (ErrnoException e) {
throw new IOException(e);
}
}
@Override
public int read(byte[] buffer) throws IOException {
return read(buffer, 0, buffer.length);
}
@Override
public long skip(long count) throws IOException {
int available = available();
if (available <= 0) {
return -1;
}
if (count > available) count = available;
mOffset += count;
updateChannelPosition(mOffset + mFileOffset);
return count;
}
@Override
public void mark(int readlimit) {
// Not supported.
return;
}
@Override
public boolean markSupported() {
return false;
}
@Override
public synchronized void reset() throws IOException {
// Not supported.
return;
}
@Override
public FileChannel getChannel() {
if (mOffsetCorrectFileChannel == null) {
mOffsetCorrectFileChannel = new OffsetCorrectFileChannel(super.getChannel());
}
try {
updateChannelPosition(mOffset + mFileOffset);
} catch (IOException e) {
throw new RuntimeException(e);
}
return mOffsetCorrectFileChannel;
}
/**
* Update the position of mOffsetCorrectFileChannel only after it is constructed.
*
* @param newPosition The absolute position mOffsetCorrectFileChannel needs to be moved to.
*/
private void updateChannelPosition(long newPosition) throws IOException {
if (mOffsetCorrectFileChannel != null) {
mOffsetCorrectFileChannel.position(newPosition);
}
}
/**
* A FileChannel wrapper that will update mOffset of the AutoCloseInputStream
* to correct position when using FileChannel to read. All occurrence of position
* should be using absolute solution and each override method just do Delegation
* besides additional check. All methods related to write mode have been disabled
* and will throw UnsupportedOperationException with customized message.
*/
private class OffsetCorrectFileChannel extends FileChannel {
private final FileChannel mDelegate;
private static final String METHOD_NOT_SUPPORTED_MESSAGE =
"This Method is not supported in AutoCloseInputStream FileChannel.";
OffsetCorrectFileChannel(FileChannel fc) {
mDelegate = fc;
}
@Override
public int read(ByteBuffer dst) throws IOException {
if (available() <= 0) return -1;
int bytesRead = mDelegate.read(dst);
if (bytesRead != -1) mOffset += bytesRead;
return bytesRead;
}
@Override
public long read(ByteBuffer[] dsts, int offset, int length) throws IOException {
if (available() <= 0) return -1;
if (mOffset + length > mTotalSize) {
length = (int) (mTotalSize - mOffset);
}
long bytesRead = mDelegate.read(dsts, offset, length);
if (bytesRead != -1) mOffset += bytesRead;
return bytesRead;
}
@Override
/**The only read method that does not move channel position*/
public int read(ByteBuffer dst, long position) throws IOException {
if (position - mFileOffset > mTotalSize) return -1;
return mDelegate.read(dst, position);
}
@Override
public long position() throws IOException {
return mDelegate.position();
}
@Override
public FileChannel position(long newPosition) throws IOException {
mOffset = newPosition - mFileOffset;
return mDelegate.position(newPosition);
}
@Override
public long size() throws IOException {
return mTotalSize;
}
@Override
public long transferTo(long position, long count, WritableByteChannel target)
throws IOException {
if (position - mFileOffset > mTotalSize) {
return 0;
}
if (position - mFileOffset + count > mTotalSize) {
count = mTotalSize - (position - mFileOffset);
}
return mDelegate.transferTo(position, count, target);
}
@Override
public MappedByteBuffer map(MapMode mode, long position, long size) throws IOException {
if (position - mFileOffset > mTotalSize) {
throw new IOException(
"Cannot map to buffer because position exceed current file size.");
}
if (position - mFileOffset + size > mTotalSize) {
size = mTotalSize - (position - mFileOffset);
}
return mDelegate.map(mode, position, size);
}
@Override
protected void implCloseChannel() throws IOException {
mDelegate.close();
}
@Override
public int write(ByteBuffer src) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public long write(ByteBuffer[] srcs, int offset, int length) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public int write(ByteBuffer src, long position) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public long transferFrom(ReadableByteChannel src, long position, long count)
throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public FileChannel truncate(long size) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public void force(boolean metaData) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public FileLock lock(long position, long size, boolean shared) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
@Override
public FileLock tryLock(long position, long size, boolean shared) throws IOException {
throw new UnsupportedOperationException(METHOD_NOT_SUPPORTED_MESSAGE);
}
}
}
/**
* An OutputStream you can create on a ParcelFileDescriptor, which will
* take care of calling {@link ParcelFileDescriptor#close
* ParcelFileDescriptor.close()} for you when the stream is closed.
*/
public static class AutoCloseOutputStream
extends ParcelFileDescriptor.AutoCloseOutputStream {
private long mRemaining;
public AutoCloseOutputStream(AssetFileDescriptor fd) throws IOException {
super(fd.getParcelFileDescriptor());
if (fd.getParcelFileDescriptor().seekTo(fd.getStartOffset()) < 0) {
throw new IOException("Unable to seek");
}
mRemaining = (int) fd.getLength();
}
@Override
public void write(byte[] buffer, int offset, int count) throws IOException {
if (mRemaining >= 0) {
if (mRemaining == 0) return;
if (count > mRemaining) count = (int) mRemaining;
super.write(buffer, offset, count);
mRemaining -= count;
return;
}
super.write(buffer, offset, count);
}
@Override
public void write(byte[] buffer) throws IOException {
if (mRemaining >= 0) {
if (mRemaining == 0) return;
int count = buffer.length;
if (count > mRemaining) count = (int) mRemaining;
super.write(buffer);
mRemaining -= count;
return;
}
super.write(buffer);
}
@Override
public void write(int oneByte) throws IOException {
if (mRemaining >= 0) {
if (mRemaining == 0) return;
super.write(oneByte);
mRemaining--;
return;
}
super.write(oneByte);
}
}
/* Parcelable interface */
@Override
public int describeContents() {
return mFd.describeContents();
}
@Override
public void writeToParcel(Parcel out, int flags) {
mFd.writeToParcel(out, flags);
out.writeLong(mStartOffset);
out.writeLong(mLength);
if (mExtras != null) {
out.writeInt(1);
out.writeBundle(mExtras);
} else {
out.writeInt(0);
}
}
AssetFileDescriptor(Parcel src) {
mFd = ParcelFileDescriptor.CREATOR.createFromParcel(src);
mStartOffset = src.readLong();
mLength = src.readLong();
if (src.readInt() != 0) {
mExtras = src.readBundle();
} else {
mExtras = null;
}
}
public static final @android.annotation.NonNull Parcelable.Creator CREATOR
= new Parcelable.Creator() {
public AssetFileDescriptor createFromParcel(Parcel in) {
return new AssetFileDescriptor(in);
}
public AssetFileDescriptor[] newArray(int size) {
return new AssetFileDescriptor[size];
}
};
}