MtpServer.cpp revision b14e588bec4d5e39e61b020b5b575f2ce555d316
116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood/*
216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * Copyright (C) 2010 The Android Open Source Project
316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood *
416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * Licensed under the Apache License, Version 2.0 (the "License");
516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * you may not use this file except in compliance with the License.
616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * You may obtain a copy of the License at
716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood *
816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood *      http://www.apache.org/licenses/LICENSE-2.0
916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood *
1016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * Unless required by applicable law or agreed to in writing, software
1116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * distributed under the License is distributed on an "AS IS" BASIS,
1216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * See the License for the specific language governing permissions and
1416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood * limitations under the License.
1516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood */
1616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
17b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood#define LOG_TAG "MtpServer"
18b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood
1916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <stdio.h>
2016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <stdlib.h>
2116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <sys/types.h>
2216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <sys/ioctl.h>
2316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <sys/stat.h>
2416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <fcntl.h>
2516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include <errno.h>
2616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
27c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood#include <cutils/properties.h>
28c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood
2916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "MtpDebug.h"
3016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "MtpServer.h"
3116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "MtpStorage.h"
3216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "MtpStringBuffer.h"
3316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "MtpDatabase.h"
34b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood#include "MtpDebug.h"
3516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
3616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood#include "f_mtp.h"
3716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
387850ef999740f214a1990a9c090d3f3865d435aaMike Lockwoodnamespace android {
397850ef999740f214a1990a9c090d3f3865d435aaMike Lockwood
4016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodstatic const MtpOperationCode kSupportedOperationCodes[] = {
4116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_DEVICE_INFO,
4216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_OPEN_SESSION,
4316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_CLOSE_SESSION,
4416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_STORAGE_IDS,
4516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_STORAGE_INFO,
4616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_NUM_OBJECTS,
4716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_OBJECT_HANDLES,
4816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_OBJECT_INFO,
4916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_OBJECT,
5016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_GET_THUMB,
5116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_DELETE_OBJECT,
5216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_SEND_OBJECT_INFO,
5316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_SEND_OBJECT,
5416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_INITIATE_CAPTURE,
5516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_FORMAT_STORE,
5616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_RESET_DEVICE,
5716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_SELF_TEST,
5816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_SET_OBJECT_PROTECTION,
5916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_POWER_DOWN,
6016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_DEVICE_PROP_DESC,
6116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_DEVICE_PROP_VALUE,
6216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_SET_DEVICE_PROP_VALUE,
6316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_RESET_DEVICE_PROP_VALUE,
6416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_TERMINATE_OPEN_CAPTURE,
6516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_MOVE_OBJECT,
6616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_COPY_OBJECT,
6716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_GET_PARTIAL_OBJECT,
6816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_INITIATE_OPEN_CAPTURE,
6916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_OBJECT_PROPS_SUPPORTED,
7016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_GET_OBJECT_PROP_DESC,
7116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_GET_OBJECT_PROP_VALUE,
7216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_OPERATION_SET_OBJECT_PROP_VALUE,
7316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_GET_OBJECT_REFERENCES,
7416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_SET_OBJECT_REFERENCES,
7516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood//    MTP_OPERATION_SKIP,
7616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood};
7716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
7816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodstatic const MtpObjectProperty kSupportedObjectProperties[] = {
7916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_PROPERTY_STORAGE_ID,
8016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_PROPERTY_OBJECT_FORMAT,
8116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_PROPERTY_OBJECT_SIZE,
8216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_PROPERTY_OBJECT_FILE_NAME,
8316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_PROPERTY_PARENT_OBJECT,
8416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood};
8516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
8616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodstatic const MtpObjectFormat kSupportedPlaybackFormats[] = {
87fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_UNDEFINED,
8816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_FORMAT_ASSOCIATION,
89fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_TEXT,
90fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_HTML,
9116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MTP_FORMAT_MP3,
92fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    //MTP_FORMAT_AVI,
93fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_MPEG,
94fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_ASF,
95fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_EXIF_JPEG,
96fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_TIFF_EP,
97fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_BMP,
98fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_GIF,
99fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_JFIF,
100fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_PNG,
101fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_TIFF,
102fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_WMA,
103fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_OGG,
104fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_AAC,
105fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_FLAC,
106fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_WMV,
107fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_MP4_CONTAINER,
108fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_MP2,
109fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MTP_FORMAT_3GP_CONTAINER,
110fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_ABSTRACT_AUDIO_ALBUM,
111fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_ABSTRACT_AV_PLAYLIST,
112fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_WPL_PLAYLIST,
113fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_M3U_PLAYLIST,
114fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_MPL_PLAYLIST,
115fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    // MTP_FORMAT_PLS_PLAYLIST,
11616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood};
11716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
11816864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpServer::MtpServer(int fd, const char* databasePath)
11916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    :   mFD(fd),
12016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mDatabasePath(databasePath),
12116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mDatabase(NULL),
12216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSessionID(0),
12316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSessionOpen(false),
12416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSendObjectHandle(kInvalidObjectHandle),
12516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSendObjectFileSize(0)
12616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood{
12716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mDatabase = new MtpDatabase();
12816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mDatabase->open(databasePath, true);
12916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
13016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
13116864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpServer::~MtpServer() {
13216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
13316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
13416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodvoid MtpServer::addStorage(const char* filePath) {
13516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int index = mStorages.size() + 1;
13616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    index |= index << 16;   // set high and low part to our index
13716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorage* storage = new MtpStorage(index, filePath, mDatabase);
13816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    addStorage(storage);
13916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
14016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
14116864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpStorage* MtpServer::getStorage(MtpStorageID id) {
14216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    for (int i = 0; i < mStorages.size(); i++) {
14316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        MtpStorage* storage =  mStorages[i];
14416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        if (storage->getStorageID() == id)
14516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            return storage;
14616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
14716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return NULL;
14816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
14916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
15016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodvoid MtpServer::scanStorage() {
15116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    for (int i = 0; i < mStorages.size(); i++) {
15216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        MtpStorage* storage =  mStorages[i];
15316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        storage->scanFiles();
15416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
15516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
15616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
15716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwoodvoid MtpServer::run() {
15816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int fd = mFD;
15916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
160b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood    LOGD("MtpServer::run fd: %d", fd);
16116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
16216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    while (1) {
16316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        int ret = mRequest.read(fd);
16416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        if (ret < 0) {
165b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            LOGE("request read returned %d, errno: %d", ret, errno);
166916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            if (errno == ECANCELED) {
167916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                // return to top of loop and wait for next command
168916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                continue;
169916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            }
17016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
17116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        }
17216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        MtpOperationCode operation = mRequest.getOperationCode();
17316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        MtpTransactionID transaction = mRequest.getTransactionID();
17416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
175b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood        LOGV("operation: %s", MtpDebug::getOperationCodeName(operation));
17616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mRequest.dump();
17716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
17816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        // FIXME need to generalize this
17916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        bool dataIn = (operation == MTP_OPERATION_SEND_OBJECT_INFO);
18016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        if (dataIn) {
18116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            int ret = mData.read(fd);
18216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            if (ret < 0) {
183b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood                LOGE("data read returned %d, errno: %d", ret, errno);
184916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                if (errno == ECANCELED) {
185916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    // return to top of loop and wait for next command
186916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    continue;
187916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                }
18816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood                break;
18916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            }
190b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            LOGV("received data:");
19116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            mData.dump();
19216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        } else {
19316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            mData.reset();
19416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        }
19516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
196916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        if (handleRequest()) {
197916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            if (!dataIn && mData.hasData()) {
198916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                mData.setOperationCode(operation);
199916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                mData.setTransactionID(transaction);
200b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood                LOGV("sending data:");
201916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                mData.dump();
202916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                ret = mData.write(fd);
203916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                if (ret < 0) {
204b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood                    LOGE("request write returned %d, errno: %d", ret, errno);
205916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    if (errno == ECANCELED) {
206916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                        // return to top of loop and wait for next command
207916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                        continue;
208916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    }
209916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    break;
210916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                }
211916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            }
21216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
213916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            mResponse.setTransactionID(transaction);
214b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            LOGV("sending response %04X", mResponse.getResponseCode());
215916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            ret = mResponse.write(fd);
21616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            if (ret < 0) {
217b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood                LOGE("request write returned %d, errno: %d", ret, errno);
218916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                if (errno == ECANCELED) {
219916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    // return to top of loop and wait for next command
220916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                    continue;
221916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood                }
22216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood                break;
22316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            }
224916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        } else {
225b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            LOGV("skipping response");
22616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        }
22716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
22816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
22916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
230916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwoodbool MtpServer::handleRequest() {
23116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpOperationCode operation = mRequest.getOperationCode();
23216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpResponseCode response;
23316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
23416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mResponse.reset();
23516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
23616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (mSendObjectHandle != kInvalidObjectHandle && operation != MTP_OPERATION_SEND_OBJECT) {
23716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        // FIXME - need to delete mSendObjectHandle from the database
238b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood        LOGE("expected SendObject after SendObjectInfo");
23916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSendObjectHandle = kInvalidObjectHandle;
24016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
24116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
24216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    switch (operation) {
24316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_DEVICE_INFO:
24416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetDeviceInfo();
24516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
24616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_OPEN_SESSION:
24716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doOpenSession();
24816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
24916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_CLOSE_SESSION:
25016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doCloseSession();
25116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
25216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_STORAGE_IDS:
25316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetStorageIDs();
25416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
25516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood         case MTP_OPERATION_GET_STORAGE_INFO:
25616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetStorageInfo();
25716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
25816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT_PROPS_SUPPORTED:
25916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetObjectPropsSupported();
26016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
26116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT_HANDLES:
26216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetObjectHandles();
26316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
26416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT_PROP_VALUE:
26516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetObjectPropValue();
26616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
26716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT_INFO:
26816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetObjectInfo();
26916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
27016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT:
27116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doGetObject();
27216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
27316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_SEND_OBJECT_INFO:
27416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doSendObjectInfo();
27516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
27616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_SEND_OBJECT:
27716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doSendObject();
27816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
27916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_DELETE_OBJECT:
28016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = doDeleteObject();
28116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
28216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        case MTP_OPERATION_GET_OBJECT_PROP_DESC:
28316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        default:
28416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            response = MTP_RESPONSE_OPERATION_NOT_SUPPORTED;
28516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            break;
28616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
28716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
288916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    if (response == MTP_RESPONSE_TRANSACTION_CANCELLED)
289916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        return false;
29016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mResponse.setResponseCode(response);
291916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    return true;
29216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
29316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
29416864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetDeviceInfo() {
29516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStringBuffer   string;
296c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood    char prop_value[PROPERTY_VALUE_MAX];
29716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
29816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // fill in device info
29916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(MTP_STANDARD_VERSION);
30016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt32(6); // MTP Vendor Extension ID
30116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(MTP_STANDARD_VERSION);
30216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    string.set("microsoft.com: 1.0;");
30316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string); // MTP Extensions
30416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(0); //Functional Mode
30516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putAUInt16(kSupportedOperationCodes,
30616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            sizeof(kSupportedOperationCodes) / sizeof(uint16_t)); // Operations Supported
30716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putEmptyArray(); // Events Supported
30816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putEmptyArray(); // Device Properties Supported
30916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putEmptyArray(); // Capture Formats
31016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putAUInt16(kSupportedPlaybackFormats,
31116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            sizeof(kSupportedPlaybackFormats) / sizeof(uint16_t)); // Playback Formats
31216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // FIXME
31316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    string.set("Google, Inc.");
31416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string);   // Manufacturer
315c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood
316c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood    property_get("ro.product.model", prop_value, "MTP Device");
317c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood    string.set(prop_value);
31816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string);   // Model
31916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    string.set("1.0");
32016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string);   // Device Version
321c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood
322c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood    property_get("ro.serialno", prop_value, "????????");
323c42aa12f73edf79bc9cb0dbf6b74a7f1af11c683Mike Lockwood    string.set(prop_value);
32416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string);   // Serial Number
32516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
32616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
32716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
32816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
32916864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doOpenSession() {
33016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (mSessionOpen) {
33116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mResponse.setParameter(1, mSessionID);
33216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_ALREADY_OPEN;
33316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
33416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSessionID = mRequest.getParameter(1);
33516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSessionOpen = true;
33616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
33716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
33816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
33916864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doCloseSession() {
34016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mSessionOpen)
34116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_NOT_OPEN;
34216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSessionID = 0;
34316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSessionOpen = false;
34416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
34516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
34616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
34716864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetStorageIDs() {
34816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mSessionOpen)
34916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_NOT_OPEN;
35016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
35116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int count = mStorages.size();
35216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt32(count);
35316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    for (int i = 0; i < count; i++)
35416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mData.putUInt32(mStorages[i]->getStorageID());
35516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
35616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
35716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
35816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
35916864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetStorageInfo() {
36016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStringBuffer   string;
36116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
36216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mSessionOpen)
36316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_NOT_OPEN;
36416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorageID id = mRequest.getParameter(1);
36516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorage* storage = getStorage(id);
36616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!storage)
36716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_INVALID_STORAGE_ID;
36816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
36916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(storage->getType());
37016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(storage->getFileSystemType());
37116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt16(storage->getAccessCapability());
37216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt64(storage->getMaxCapacity());
37316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt64(storage->getFreeSpace());
37416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putUInt32(1024*1024*1024); // Free Space in Objects
37516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    string.set(storage->getDescription());
37616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putString(string);
37716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putEmptyString();   // Volume Identifier
37816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
37916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
38016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
38116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
38216864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObjectPropsSupported() {
38316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mSessionOpen)
38416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_NOT_OPEN;
38516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectFormat format = mRequest.getParameter(1);
38616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putAUInt16(kSupportedObjectProperties,
38716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            sizeof(kSupportedObjectProperties) / sizeof(uint16_t));
38816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
38916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
39016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
39116864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObjectHandles() {
39216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mSessionOpen)
39316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_SESSION_NOT_OPEN;
39416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorageID storageID = mRequest.getParameter(1);      // 0xFFFFFFFF for all storage
395e13401bf532c7e4bf9ab82c7e9b13642838a927dMike Lockwood    MtpObjectFormat format = mRequest.getParameter(2);      // 0 for all formats
39616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle parent = mRequest.getParameter(3);      // 0xFFFFFFFF for objects with no parent
39716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood                                                            // 0x00000000 for all objects?
39816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
39916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandleList* handles = mDatabase->getObjectList(storageID, format, parent);
40016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.putAUInt32(handles);
40116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    delete handles;
40216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
40316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
40416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
40516864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObjectPropValue() {
40616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle handle = mRequest.getParameter(1);
40716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectProperty property = mRequest.getParameter(2);
40816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
40916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return mDatabase->getObjectProperty(handle, property, mData);
41016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
41116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
41216864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObjectInfo() {
41316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle handle = mRequest.getParameter(1);
41416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return mDatabase->getObjectInfo(handle, mData);
41516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
41616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
41716864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObject() {
41816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle handle = mRequest.getParameter(1);
419c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    MtpString pathBuf;
42016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int64_t fileLength;
421c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    if (!mDatabase->getObjectFilePath(handle, pathBuf, fileLength))
42216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_INVALID_OBJECT_HANDLE;
423c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    const char* filePath = (const char *)pathBuf;
42416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
42516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mtp_file_range  mfr;
426c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    mfr.fd = open(filePath, O_RDONLY);
427c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    if (mfr.fd < 0) {
428c6588763ddc20541688e426a24b1b070527c051fMike Lockwood        return MTP_RESPONSE_GENERAL_ERROR;
429c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    }
43016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mfr.offset = 0;
43116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mfr.length = fileLength;
43216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
43316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // send data header
43416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.setOperationCode(mRequest.getOperationCode());
43516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.setTransactionID(mRequest.getTransactionID());
43616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.writeDataHeader(mFD, fileLength);
43716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
43816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // then transfer the file
43916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int ret = ioctl(mFD, MTP_SEND_FILE, (unsigned long)&mfr);
440c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    close(mfr.fd);
441916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    if (ret < 0) {
442916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        if (errno == ECANCELED)
443916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            return MTP_RESPONSE_TRANSACTION_CANCELLED;
444916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        else
445916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            return MTP_RESPONSE_GENERAL_ERROR;
446916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    }
44716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
44816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
44916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
45016864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doSendObjectInfo() {
45116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpString path;
45216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorageID storageID = mRequest.getParameter(1);
45316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStorage* storage = getStorage(storageID);
45416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle parent = mRequest.getParameter(2);
45516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!storage)
45616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_INVALID_STORAGE_ID;
45716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
45816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // special case the root
45916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (parent == MTP_PARENT_ROOT)
46016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        path = storage->getPath();
46116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    else {
46216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        int64_t dummy;
46316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        if (!mDatabase->getObjectFilePath(parent, path, dummy))
46416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            return MTP_RESPONSE_INVALID_OBJECT_HANDLE;
46516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
46616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
46716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // read only the fields we need
46816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // storage ID
46916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectFormat format = mData.getUInt16();
47016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt16();  // protection status
47116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSendObjectFileSize = mData.getUInt32();
47216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt16();  // thumb format
47316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // thumb compressed size
47416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // thumb pix width
47516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // thumb pix height
47616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // image pix width
47716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // image pix height
47816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // image bit depth
47916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // parent
48016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    uint16_t associationType = mData.getUInt16();
48116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    uint32_t associationDesc = mData.getUInt32();   // association desc
48216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getUInt32();  // sequence number
48316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpStringBuffer name, created, modified;
48416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getString(name);    // file name
48516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getString(created);      // date created
48616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.getString(modified);     // date modified
48716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // keywords follow
48816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
489fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    time_t modifiedTime;
49016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!parseDateTime(modified, modifiedTime))
49116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        modifiedTime = 0;
492b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood    LOGV("SendObjectInfo format: %04X size: %d name: %s, created: %s, modified: %s",
493b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            format, mSendObjectFileSize, (const char*)name, (const char*)created,
494b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood            (const char*)modified);
49516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
49616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (path[path.size() - 1] != '/')
49716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        path += "/";
49816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    path += (const char *)name;
49916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
500fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    mDatabase->beginTransaction();
501fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    MtpObjectHandle handle = mDatabase->addFile((const char*)path, format, parent, storageID,
502fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood                                    mSendObjectFileSize, modifiedTime);
503fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    if (handle == kInvalidObjectHandle) {
504fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood        mDatabase->rollbackTransaction();
50516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_GENERAL_ERROR;
506fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    }
507fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    uint32_t table = MtpDatabase::getTableForFile(format);
508fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    if (table == kObjectHandleTableAudio)
509fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood        handle = mDatabase->addAudioFile(handle);
510fceef46513db3507b413f604cea89e3c7f352663Mike Lockwood    mDatabase->commitTransaction();
51116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
51216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood  if (format == MTP_FORMAT_ASSOCIATION) {
51316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mode_t mask = umask(0);
51416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        int ret = mkdir((const char *)path, S_IRWXU | S_IRWXG | S_IRWXO);
51516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        umask(mask);
51616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        if (ret && ret != -EEXIST)
51716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood            return MTP_RESPONSE_GENERAL_ERROR;
51816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    } else {
51916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSendObjectFilePath = path;
52016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        // save the handle for the SendObject call, which should follow
52116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        mSendObjectHandle = handle;
52216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
52316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
52416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mResponse.setParameter(1, storageID);
52516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mResponse.setParameter(2, parent);
52616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mResponse.setParameter(3, handle);
52716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
52816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
52916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
53016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
53116864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doSendObject() {
53216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (mSendObjectHandle == kInvalidObjectHandle) {
533b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood        LOGE("Expected SendObjectInfo before SendObject");
53416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_NO_VALID_OBJECT_INFO;
53516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    }
53616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
53716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // read the header
53816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int ret = mData.readDataHeader(mFD);
53916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // FIXME - check for errors here.
54016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
54116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // reset so we don't attempt to send this back
54216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mData.reset();
54316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
54416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mtp_file_range  mfr;
545c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    mfr.fd = open(mSendObjectFilePath, O_RDWR | O_CREAT | O_TRUNC);
546c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    if (mfr.fd < 0) {
547c6588763ddc20541688e426a24b1b070527c051fMike Lockwood        return MTP_RESPONSE_GENERAL_ERROR;
548c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    }
54916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mfr.offset = 0;
55016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mfr.length = mSendObjectFileSize;
55116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
55216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // transfer the file
55316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    ret = ioctl(mFD, MTP_RECEIVE_FILE, (unsigned long)&mfr);
554c6588763ddc20541688e426a24b1b070527c051fMike Lockwood    close(mfr.fd);
555916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    // FIXME - we need to delete mSendObjectHandle from the database if this fails.
556b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood    LOGV("MTP_RECEIVE_FILE returned %d", ret);
55716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mSendObjectHandle = kInvalidObjectHandle;
55816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
559916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    if (ret < 0) {
560916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        unlink(mSendObjectFilePath);
561916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        if (errno == ECANCELED)
562916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            return MTP_RESPONSE_TRANSACTION_CANCELLED;
563916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood        else
564916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood            return MTP_RESPONSE_GENERAL_ERROR;
565916076c6d84dac9b104fbdf94af5dcd7bce669fdMike Lockwood    }
56616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
56716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
56816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
56916864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doDeleteObject() {
57016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectHandle handle = mRequest.getParameter(1);
57116864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectFormat format = mRequest.getParameter(1);
57216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // FIXME - support deleting all objects if handle is 0xFFFFFFFF
57316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // FIXME - implement deleting objects by format
57416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // FIXME - handle non-empty directories
57516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
57616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpString filePath;
57716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    int64_t fileLength;
57816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    if (!mDatabase->getObjectFilePath(handle, filePath, fileLength))
57916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood        return MTP_RESPONSE_INVALID_OBJECT_HANDLE;
58016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
581b14e588bec4d5e39e61b020b5b575f2ce555d316Mike Lockwood    LOGV("deleting %s", (const char *)filePath);
58216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    // one of these should work
58316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    rmdir((const char *)filePath);
58416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    unlink((const char *)filePath);
58516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
58616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    mDatabase->deleteFile(handle);
58716864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
58816864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return MTP_RESPONSE_OK;
58916864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
59016864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
59116864bae0f51c32c456da2c43adf7a057c0c4882Mike LockwoodMtpResponseCode MtpServer::doGetObjectPropDesc() {
59216864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectProperty property = mRequest.getParameter(1);
59316864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    MtpObjectFormat format = mRequest.getParameter(2);
59416864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood
59516864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood    return -1;
59616864bae0f51c32c456da2c43adf7a057c0c4882Mike Lockwood}
5977850ef999740f214a1990a9c090d3f3865d435aaMike Lockwood
5987850ef999740f214a1990a9c090d3f3865d435aaMike Lockwood}  // namespace android
599