/*
 * Copyright (C) 2010 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.mtp;

import android.annotation.NonNull;
import android.annotation.Nullable;
import android.content.Context;
import android.hardware.usb.UsbDevice;
import android.hardware.usb.UsbDeviceConnection;
import android.os.CancellationSignal;
import android.os.ParcelFileDescriptor;

import android.os.UserManager;
import com.android.internal.annotations.GuardedBy;
import com.android.internal.util.Preconditions;
import dalvik.system.CloseGuard;

import java.io.IOException;

This class represents an MTP or PTP device connected on the USB host bus. An application can instantiate an object of this type, by referencing an attached UsbDevice and then use methods in this class to get information about the device and objects stored on it, as well as open the connection and transfer data.
/** * This class represents an MTP or PTP device connected on the USB host bus. An application can * instantiate an object of this type, by referencing an attached {@link * android.hardware.usb.UsbDevice} and then use methods in this class to get information about the * device and objects stored on it, as well as open the connection and transfer data. */
public final class MtpDevice { private static final String TAG = "MtpDevice"; private final UsbDevice mDevice; static { System.loadLibrary("media_jni"); }
Make sure that MTP device is closed properly
/** Make sure that MTP device is closed properly */
@GuardedBy("mLock") private CloseGuard mCloseGuard = CloseGuard.get();
Current connection to the mDevice, or null if device is not connected
/** Current connection to the {@link #mDevice}, or null if device is not connected */
@GuardedBy("mLock") private UsbDeviceConnection mConnection; private final Object mLock = new Object();
MtpClient constructor
Params:
  • device – the UsbDevice for the MTP or PTP device
/** * MtpClient constructor * * @param device the {@link android.hardware.usb.UsbDevice} for the MTP or PTP device */
public MtpDevice(@NonNull UsbDevice device) { Preconditions.checkNotNull(device); mDevice = device; }
Opens the MTP device. Once the device is open it takes ownership of the UsbDeviceConnection. The connection will be closed when you call close() The connection will also be closed if this method fails.
Params:
Returns:true if the device was successfully opened.
/** * Opens the MTP device. Once the device is open it takes ownership of the * {@link android.hardware.usb.UsbDeviceConnection}. * The connection will be closed when you call {@link #close()} * The connection will also be closed if this method fails. * * @param connection an open {@link android.hardware.usb.UsbDeviceConnection} for the device * @return true if the device was successfully opened. */
public boolean open(@NonNull UsbDeviceConnection connection) { boolean result = false; Context context = connection.getContext(); synchronized (mLock) { if (context != null) { UserManager userManager = (UserManager) context .getSystemService(Context.USER_SERVICE); if (!userManager.hasUserRestriction(UserManager.DISALLOW_USB_FILE_TRANSFER)) { result = native_open(mDevice.getDeviceName(), connection.getFileDescriptor()); } } if (!result) { connection.close(); } else { mConnection = connection; mCloseGuard.open("close"); } } return result; }
Closes all resources related to the MtpDevice object. After this is called, the object can not be used until open is called again with a new UsbDeviceConnection.
/** * Closes all resources related to the MtpDevice object. * After this is called, the object can not be used until {@link #open} is called again * with a new {@link android.hardware.usb.UsbDeviceConnection}. */
public void close() { synchronized (mLock) { if (mConnection != null) { mCloseGuard.close(); native_close(); mConnection.close(); mConnection = null; } } } @Override protected void finalize() throws Throwable { try { if (mCloseGuard != null) { mCloseGuard.warnIfOpen(); } close(); } finally { super.finalize(); } }
Returns the name of the USB device This returns the same value as UsbDevice.getDeviceName for the device's UsbDevice
Returns:the device name
/** * Returns the name of the USB device * This returns the same value as {@link android.hardware.usb.UsbDevice#getDeviceName} * for the device's {@link android.hardware.usb.UsbDevice} * * @return the device name */
public @NonNull String getDeviceName() { return mDevice.getDeviceName(); }
Returns the USB ID of the USB device. This returns the same value as UsbDevice.getDeviceId for the device's UsbDevice
Returns:the device ID
/** * Returns the USB ID of the USB device. * This returns the same value as {@link android.hardware.usb.UsbDevice#getDeviceId} * for the device's {@link android.hardware.usb.UsbDevice} * * @return the device ID */
public int getDeviceId() { return mDevice.getDeviceId(); } @Override public @NonNull String toString() { return mDevice.getDeviceName(); }
Returns the MtpDeviceInfo for this device
Returns:the device info, or null if fetching device info fails
/** * Returns the {@link MtpDeviceInfo} for this device * * @return the device info, or null if fetching device info fails */
public @Nullable MtpDeviceInfo getDeviceInfo() { return native_get_device_info(); }
Returns the list of IDs for all storage units on this device Information about each storage unit can be accessed via getStorageInfo.
Returns:the list of storage IDs, or null if fetching storage IDs fails
/** * Returns the list of IDs for all storage units on this device * Information about each storage unit can be accessed via {@link #getStorageInfo}. * * @return the list of storage IDs, or null if fetching storage IDs fails */
public @Nullable int[] getStorageIds() { return native_get_storage_ids(); }
Returns the list of object handles for all objects on the given storage unit, with the given format and parent. Information about each object can be accessed via getObjectInfo.
Params:
  • storageId – the storage unit to query
  • format – the format of the object to return, or zero for all formats
  • objectHandle – the parent object to query, -1 for the storage root, or zero for all objects
Returns:the object handles, or null if fetching object handles fails
/** * Returns the list of object handles for all objects on the given storage unit, * with the given format and parent. * Information about each object can be accessed via {@link #getObjectInfo}. * * @param storageId the storage unit to query * @param format the format of the object to return, or zero for all formats * @param objectHandle the parent object to query, -1 for the storage root, * or zero for all objects * @return the object handles, or null if fetching object handles fails */
public @Nullable int[] getObjectHandles(int storageId, int format, int objectHandle) { return native_get_object_handles(storageId, format, objectHandle); }
Returns the data for an object as a byte array. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices.
Params:
Returns:the object's data, or null if reading fails
/** * Returns the data for an object as a byte array. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. * * @param objectHandle handle of the object to read * @param objectSize the size of the object (this should match * {@link MtpObjectInfo#getCompressedSize}) * @return the object's data, or null if reading fails */
public @Nullable byte[] getObject(int objectHandle, int objectSize) { Preconditions.checkArgumentNonnegative(objectSize, "objectSize should not be negative"); return native_get_object(objectHandle, objectSize); }
Obtains object bytes in the specified range and writes it to an array. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices.
Params:
  • objectHandle – handle of the object to read
  • offset – Start index of reading range. It must be a non-negative value at most 0xffffffff.
  • size – Size of reading range. It must be a non-negative value at most Integer.MAX_VALUE or 0xffffffff. If 0xffffffff is specified, the method obtains the full bytes of object.
  • buffer – Array to write data.
Returns:Size of bytes that are actually read.
/** * Obtains object bytes in the specified range and writes it to an array. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. * * @param objectHandle handle of the object to read * @param offset Start index of reading range. It must be a non-negative value at most * 0xffffffff. * @param size Size of reading range. It must be a non-negative value at most Integer.MAX_VALUE * or 0xffffffff. If 0xffffffff is specified, the method obtains the full bytes of object. * @param buffer Array to write data. * @return Size of bytes that are actually read. */
public long getPartialObject(int objectHandle, long offset, long size, @NonNull byte[] buffer) throws IOException { return native_get_partial_object(objectHandle, offset, size, buffer); }
Obtains object bytes in the specified range and writes it to an array. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices. This is a vender-extended operation supported by Android that enables us to pass unsigned 64-bit offset. Check if the MTP device supports the operation by using MtpDeviceInfo.getOperationsSupported().
Params:
  • objectHandle – handle of the object to read
  • offset – Start index of reading range. It must be a non-negative value.
  • size – Size of reading range. It must be a non-negative value at most Integer.MAX_VALUE.
  • buffer – Array to write data.
See Also:
Returns:Size of bytes that are actually read.
/** * Obtains object bytes in the specified range and writes it to an array. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. * * This is a vender-extended operation supported by Android that enables us to pass * unsigned 64-bit offset. Check if the MTP device supports the operation by using * {@link MtpDeviceInfo#getOperationsSupported()}. * * @param objectHandle handle of the object to read * @param offset Start index of reading range. It must be a non-negative value. * @param size Size of reading range. It must be a non-negative value at most Integer.MAX_VALUE. * @param buffer Array to write data. * @return Size of bytes that are actually read. * @see MtpConstants#OPERATION_GET_PARTIAL_OBJECT_64 */
public long getPartialObject64(int objectHandle, long offset, long size, @NonNull byte[] buffer) throws IOException { return native_get_partial_object_64(objectHandle, offset, size, buffer); }
Returns the thumbnail data for an object as a byte array. The size and format of the thumbnail data can be determined via MtpObjectInfo.getThumbCompressedSize and MtpObjectInfo.getThumbFormat. For typical devices the format is JPEG.
Params:
  • objectHandle – handle of the object to read
Returns:the object's thumbnail, or null if reading fails
/** * Returns the thumbnail data for an object as a byte array. * The size and format of the thumbnail data can be determined via * {@link MtpObjectInfo#getThumbCompressedSize} and * {@link MtpObjectInfo#getThumbFormat}. * For typical devices the format is JPEG. * * @param objectHandle handle of the object to read * @return the object's thumbnail, or null if reading fails */
public @Nullable byte[] getThumbnail(int objectHandle) { return native_get_thumbnail(objectHandle); }
Retrieves the MtpStorageInfo for a storage unit.
Params:
  • storageId – the ID of the storage unit
Returns:the MtpStorageInfo, or null if fetching storage info fails
/** * Retrieves the {@link MtpStorageInfo} for a storage unit. * * @param storageId the ID of the storage unit * @return the MtpStorageInfo, or null if fetching storage info fails */
public @Nullable MtpStorageInfo getStorageInfo(int storageId) { return native_get_storage_info(storageId); }
Retrieves the MtpObjectInfo for an object.
Params:
  • objectHandle – the handle of the object
Returns:the MtpObjectInfo, or null if fetching object info fails
/** * Retrieves the {@link MtpObjectInfo} for an object. * * @param objectHandle the handle of the object * @return the MtpObjectInfo, or null if fetching object info fails */
public @Nullable MtpObjectInfo getObjectInfo(int objectHandle) { return native_get_object_info(objectHandle); }
Deletes an object on the device. This call may block, since deleting a directory containing many files may take a long time on some devices.
Params:
  • objectHandle – handle of the object to delete
Returns:true if the deletion succeeds
/** * Deletes an object on the device. This call may block, since * deleting a directory containing many files may take a long time * on some devices. * * @param objectHandle handle of the object to delete * @return true if the deletion succeeds */
public boolean deleteObject(int objectHandle) { return native_delete_object(objectHandle); }
Retrieves the object handle for the parent of an object on the device.
Params:
  • objectHandle – handle of the object to query
Returns:the parent's handle, or zero if it is in the root of the storage
/** * Retrieves the object handle for the parent of an object on the device. * * @param objectHandle handle of the object to query * @return the parent's handle, or zero if it is in the root of the storage */
public long getParent(int objectHandle) { return native_get_parent(objectHandle); }
Retrieves the ID of the storage unit containing the given object on the device.
Params:
  • objectHandle – handle of the object to query
Returns:the object's storage unit ID
/** * Retrieves the ID of the storage unit containing the given object on the device. * * @param objectHandle handle of the object to query * @return the object's storage unit ID */
public long getStorageId(int objectHandle) { return native_get_storage_id(objectHandle); }
Copies the data for an object to a file in external storage. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices.
Params:
  • objectHandle – handle of the object to read
  • destPath – path to destination for the file transfer. This path should be in the external storage as defined by Environment.getExternalStorageDirectory
Returns:true if the file transfer succeeds
/** * Copies the data for an object to a file in external storage. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. * * @param objectHandle handle of the object to read * @param destPath path to destination for the file transfer. * This path should be in the external storage as defined by * {@link android.os.Environment#getExternalStorageDirectory} * @return true if the file transfer succeeds */
public boolean importFile(int objectHandle, @NonNull String destPath) { return native_import_file(objectHandle, destPath); }
Copies the data for an object to a file descriptor. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices. The file descriptor is not closed on completion, and must be done by the caller.
Params:
  • objectHandle – handle of the object to read
  • descriptor – file descriptor to write the data to for the file transfer.
Returns:true if the file transfer succeeds
/** * Copies the data for an object to a file descriptor. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. The file descriptor is not closed * on completion, and must be done by the caller. * * @param objectHandle handle of the object to read * @param descriptor file descriptor to write the data to for the file transfer. * @return true if the file transfer succeeds */
public boolean importFile(int objectHandle, @NonNull ParcelFileDescriptor descriptor) { return native_import_file(objectHandle, descriptor.getFd()); }
Copies the data for an object from a file descriptor. This call may block for an arbitrary amount of time depending on the size of the data and speed of the devices. The file descriptor is not closed on completion, and must be done by the caller.
Params:
  • objectHandle – handle of the target file
  • size – size of the file in bytes
  • descriptor – file descriptor to read the data from.
Returns:true if the file transfer succeeds
/** * Copies the data for an object from a file descriptor. * This call may block for an arbitrary amount of time depending on the size * of the data and speed of the devices. The file descriptor is not closed * on completion, and must be done by the caller. * * @param objectHandle handle of the target file * @param size size of the file in bytes * @param descriptor file descriptor to read the data from. * @return true if the file transfer succeeds */
public boolean sendObject( int objectHandle, long size, @NonNull ParcelFileDescriptor descriptor) { return native_send_object(objectHandle, size, descriptor.getFd()); }
Uploads an object metadata for a new entry. The MtpObjectInfo can be created with the Builder class. The returned MtpObjectInfo has the new object handle field filled in.
Params:
  • info – metadata of the entry
Returns:object info of the created entry, or null if sending object info fails
/** * Uploads an object metadata for a new entry. The {@link MtpObjectInfo} can be * created with the {@link MtpObjectInfo.Builder} class. * * The returned {@link MtpObjectInfo} has the new object handle field filled in. * * @param info metadata of the entry * @return object info of the created entry, or null if sending object info fails */
public @Nullable MtpObjectInfo sendObjectInfo(@NonNull MtpObjectInfo info) { return native_send_object_info(info); }
Reads an event from the device. It blocks the current thread until it gets an event. It throws OperationCanceledException if it is cancelled by signal.
Params:
  • signal – signal for cancellation
Throws:
Returns:obtained event
/** * Reads an event from the device. It blocks the current thread until it gets an event. * It throws OperationCanceledException if it is cancelled by signal. * * @param signal signal for cancellation * @return obtained event * @throws IOException */
public @NonNull MtpEvent readEvent(@Nullable CancellationSignal signal) throws IOException { final int handle = native_submit_event_request(); Preconditions.checkState(handle >= 0, "Other thread is reading an event."); if (signal != null) { signal.setOnCancelListener(new CancellationSignal.OnCancelListener() { @Override public void onCancel() { native_discard_event_request(handle); } }); } try { return native_reap_event_request(handle); } finally { if (signal != null) { signal.setOnCancelListener(null); } } }
Returns object size in 64-bit integer. Though MtpObjectInfo#getCompressedSize returns the object size in 32-bit unsigned integer, this method returns the object size in 64-bit integer from the object property. Thus it can fetch 4GB+ object size correctly. If the device does not support objectSize property, it throws IOException.
@hide
/** * Returns object size in 64-bit integer. * * Though MtpObjectInfo#getCompressedSize returns the object size in 32-bit unsigned integer, * this method returns the object size in 64-bit integer from the object property. Thus it can * fetch 4GB+ object size correctly. If the device does not support objectSize property, it * throws IOException. * @hide */
public long getObjectSizeLong(int handle, int format) throws IOException { return native_get_object_size_long(handle, format); } // used by the JNI code private long mNativeContext; private native boolean native_open(String deviceName, int fd); private native void native_close(); private native MtpDeviceInfo native_get_device_info(); private native int[] native_get_storage_ids(); private native MtpStorageInfo native_get_storage_info(int storageId); private native int[] native_get_object_handles(int storageId, int format, int objectHandle); private native MtpObjectInfo native_get_object_info(int objectHandle); private native byte[] native_get_object(int objectHandle, long objectSize); private native long native_get_partial_object( int objectHandle, long offset, long objectSize, byte[] buffer) throws IOException; private native int native_get_partial_object_64( int objectHandle, long offset, long objectSize, byte[] buffer) throws IOException; private native byte[] native_get_thumbnail(int objectHandle); private native boolean native_delete_object(int objectHandle); private native int native_get_parent(int objectHandle); private native int native_get_storage_id(int objectHandle); private native boolean native_import_file(int objectHandle, String destPath); private native boolean native_import_file(int objectHandle, int fd); private native boolean native_send_object(int objectHandle, long size, int fd); private native MtpObjectInfo native_send_object_info(MtpObjectInfo info); private native int native_submit_event_request() throws IOException; private native MtpEvent native_reap_event_request(int handle) throws IOException; private native void native_discard_event_request(int handle); private native long native_get_object_size_long(int handle, int format) throws IOException; }