blob: 0f3c063bedae006b455de451891b2647de38feba [file] [log] [blame]
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "MtpDatabaseJNI"
18#include "utils/Log.h"
19
20#include <stdio.h>
21#include <assert.h>
22#include <limits.h>
23#include <unistd.h>
24#include <fcntl.h>
25
26#include "jni.h"
27#include "JNIHelp.h"
28#include "android_runtime/AndroidRuntime.h"
29
30#include "MtpDatabase.h"
31#include "MtpDataPacket.h"
Mike Lockwood9df53fae2011-04-21 17:05:55 -070032#include "MtpObjectInfo.h"
Mike Lockwood828d19d2010-08-10 15:20:35 -040033#include "MtpProperty.h"
Mike Lockwood59e3f0d2010-09-02 14:57:30 -040034#include "MtpStringBuffer.h"
Mike Lockwoodd21eac92010-07-03 00:44:05 -040035#include "MtpUtils.h"
36#include "mtp.h"
37
Mike Lockwoodc89f2222011-04-24 18:40:17 -070038extern "C" {
39#include "jhead.h"
40}
41
Mike Lockwoodd21eac92010-07-03 00:44:05 -040042using namespace android;
43
44// ----------------------------------------------------------------------------
45
Mike Lockwoodd815f792010-07-12 08:49:01 -040046static jmethodID method_beginSendObject;
47static jmethodID method_endSendObject;
Mike Lockwoodd21eac92010-07-03 00:44:05 -040048static jmethodID method_getObjectList;
Mike Lockwood7a047c82010-08-02 10:52:20 -040049static jmethodID method_getNumObjects;
Mike Lockwood4b322ce2010-08-10 07:37:50 -040050static jmethodID method_getSupportedPlaybackFormats;
51static jmethodID method_getSupportedCaptureFormats;
52static jmethodID method_getSupportedObjectProperties;
53static jmethodID method_getSupportedDeviceProperties;
Mike Lockwood828d19d2010-08-10 15:20:35 -040054static jmethodID method_setObjectProperty;
Mike Lockwood59e3f0d2010-09-02 14:57:30 -040055static jmethodID method_getDeviceProperty;
56static jmethodID method_setDeviceProperty;
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -040057static jmethodID method_getObjectPropertyList;
Mike Lockwoodd21eac92010-07-03 00:44:05 -040058static jmethodID method_getObjectInfo;
59static jmethodID method_getObjectFilePath;
60static jmethodID method_deleteFile;
Mike Lockwood9a2046f2010-08-03 15:30:09 -040061static jmethodID method_getObjectReferences;
62static jmethodID method_setObjectReferences;
Mike Lockwood2837eef2010-08-31 16:25:12 -040063static jmethodID method_sessionStarted;
64static jmethodID method_sessionEnded;
65
Mike Lockwoodd21eac92010-07-03 00:44:05 -040066static jfieldID field_context;
67
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -040068// MtpPropertyList fields
69static jfieldID field_mCount;
70static jfieldID field_mResult;
71static jfieldID field_mObjectHandles;
72static jfieldID field_mPropertyCodes;
73static jfieldID field_mDataTypes;
74static jfieldID field_mLongValues;
75static jfieldID field_mStringValues;
76
77
Mike Lockwoodd21eac92010-07-03 00:44:05 -040078MtpDatabase* getMtpDatabase(JNIEnv *env, jobject database) {
79 return (MtpDatabase *)env->GetIntField(database, field_context);
80}
81
Mike Lockwoodff164a72010-07-15 15:01:17 -040082#ifdef HAVE_ANDROID_OS
Mike Lockwoodd21eac92010-07-03 00:44:05 -040083// ----------------------------------------------------------------------------
84
85class MyMtpDatabase : public MtpDatabase {
86private:
87 jobject mDatabase;
88 jintArray mIntBuffer;
89 jlongArray mLongBuffer;
90 jcharArray mStringBuffer;
91
92public:
93 MyMtpDatabase(JNIEnv *env, jobject client);
94 virtual ~MyMtpDatabase();
95 void cleanup(JNIEnv *env);
96
Mike Lockwoodd815f792010-07-12 08:49:01 -040097 virtual MtpObjectHandle beginSendObject(const char* path,
Mike Lockwoodd21eac92010-07-03 00:44:05 -040098 MtpObjectFormat format,
99 MtpObjectHandle parent,
100 MtpStorageID storage,
101 uint64_t size,
102 time_t modified);
103
Mike Lockwoodd815f792010-07-12 08:49:01 -0400104 virtual void endSendObject(const char* path,
105 MtpObjectHandle handle,
106 MtpObjectFormat format,
107 bool succeeded);
108
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400109 virtual MtpObjectHandleList* getObjectList(MtpStorageID storageID,
110 MtpObjectFormat format,
111 MtpObjectHandle parent);
112
Mike Lockwood7a047c82010-08-02 10:52:20 -0400113 virtual int getNumObjects(MtpStorageID storageID,
114 MtpObjectFormat format,
115 MtpObjectHandle parent);
116
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400117 // callee should delete[] the results from these
118 // results can be NULL
119 virtual MtpObjectFormatList* getSupportedPlaybackFormats();
120 virtual MtpObjectFormatList* getSupportedCaptureFormats();
121 virtual MtpObjectPropertyList* getSupportedObjectProperties(MtpObjectFormat format);
122 virtual MtpDevicePropertyList* getSupportedDeviceProperties();
123
Mike Lockwood828d19d2010-08-10 15:20:35 -0400124 virtual MtpResponseCode getObjectPropertyValue(MtpObjectHandle handle,
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400125 MtpObjectProperty property,
126 MtpDataPacket& packet);
127
Mike Lockwood828d19d2010-08-10 15:20:35 -0400128 virtual MtpResponseCode setObjectPropertyValue(MtpObjectHandle handle,
129 MtpObjectProperty property,
130 MtpDataPacket& packet);
131
132 virtual MtpResponseCode getDevicePropertyValue(MtpDeviceProperty property,
133 MtpDataPacket& packet);
134
135 virtual MtpResponseCode setDevicePropertyValue(MtpDeviceProperty property,
136 MtpDataPacket& packet);
137
138 virtual MtpResponseCode resetDeviceProperty(MtpDeviceProperty property);
139
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400140 virtual MtpResponseCode getObjectPropertyList(MtpObjectHandle handle,
Mike Lockwood7d7fb632010-12-01 18:46:23 -0500141 uint32_t format, uint32_t property,
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400142 int groupCode, int depth,
143 MtpDataPacket& packet);
144
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400145 virtual MtpResponseCode getObjectInfo(MtpObjectHandle handle,
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700146 MtpObjectInfo& info);
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400147
Mike Lockwoodc89f2222011-04-24 18:40:17 -0700148 virtual void* getThumbnail(MtpObjectHandle handle, size_t& outThumbSize);
149
Mike Lockwood59c777a2010-08-02 10:37:41 -0400150 virtual MtpResponseCode getObjectFilePath(MtpObjectHandle handle,
Mike Lockwood365e03e2010-12-08 16:08:01 -0800151 MtpString& outFilePath,
152 int64_t& outFileLength,
153 MtpObjectFormat& outFormat);
Mike Lockwood59c777a2010-08-02 10:37:41 -0400154 virtual MtpResponseCode deleteFile(MtpObjectHandle handle);
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400155
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400156 bool getObjectPropertyInfo(MtpObjectProperty property, int& type);
157 bool getDevicePropertyInfo(MtpDeviceProperty property, int& type);
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400158
159 virtual MtpObjectHandleList* getObjectReferences(MtpObjectHandle handle);
160
161 virtual MtpResponseCode setObjectReferences(MtpObjectHandle handle,
162 MtpObjectHandleList* references);
Mike Lockwood828d19d2010-08-10 15:20:35 -0400163
164 virtual MtpProperty* getObjectPropertyDesc(MtpObjectProperty property,
165 MtpObjectFormat format);
166
167 virtual MtpProperty* getDevicePropertyDesc(MtpDeviceProperty property);
Mike Lockwood2837eef2010-08-31 16:25:12 -0400168
169 virtual void sessionStarted();
170
171 virtual void sessionEnded();
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400172};
173
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400174// ----------------------------------------------------------------------------
175
176static void checkAndClearExceptionFromCallback(JNIEnv* env, const char* methodName) {
177 if (env->ExceptionCheck()) {
178 LOGE("An exception was thrown by callback '%s'.", methodName);
179 LOGE_EX(env);
180 env->ExceptionClear();
181 }
182}
183
184// ----------------------------------------------------------------------------
185
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400186MyMtpDatabase::MyMtpDatabase(JNIEnv *env, jobject client)
187 : mDatabase(env->NewGlobalRef(client)),
188 mIntBuffer(NULL),
189 mLongBuffer(NULL),
190 mStringBuffer(NULL)
191{
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400192 // create buffers for out arguments
193 // we don't need to be thread-safe so this is OK
Elliott Hughes15dd15f2011-04-08 17:42:34 -0700194 jintArray intArray = env->NewIntArray(3);
195 if (!intArray) {
196 return; // Already threw.
197 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400198 mIntBuffer = (jintArray)env->NewGlobalRef(intArray);
Elliott Hughes15dd15f2011-04-08 17:42:34 -0700199 jlongArray longArray = env->NewLongArray(2);
200 if (!longArray) {
201 return; // Already threw.
202 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400203 mLongBuffer = (jlongArray)env->NewGlobalRef(longArray);
Elliott Hughes15dd15f2011-04-08 17:42:34 -0700204 jcharArray charArray = env->NewCharArray(256);
205 if (!charArray) {
206 return; // Already threw.
207 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400208 mStringBuffer = (jcharArray)env->NewGlobalRef(charArray);
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400209}
210
211void MyMtpDatabase::cleanup(JNIEnv *env) {
212 env->DeleteGlobalRef(mDatabase);
213 env->DeleteGlobalRef(mIntBuffer);
214 env->DeleteGlobalRef(mLongBuffer);
215 env->DeleteGlobalRef(mStringBuffer);
216}
217
218MyMtpDatabase::~MyMtpDatabase() {
219}
220
Mike Lockwoodd815f792010-07-12 08:49:01 -0400221MtpObjectHandle MyMtpDatabase::beginSendObject(const char* path,
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400222 MtpObjectFormat format,
223 MtpObjectHandle parent,
224 MtpStorageID storage,
225 uint64_t size,
226 time_t modified) {
227 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood88394712010-09-27 10:01:00 -0400228 jstring pathStr = env->NewStringUTF(path);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400229 MtpObjectHandle result = env->CallIntMethod(mDatabase, method_beginSendObject,
Mike Lockwood88394712010-09-27 10:01:00 -0400230 pathStr, (jint)format, (jint)parent, (jint)storage,
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400231 (jlong)size, (jlong)modified);
232
Mike Lockwood88394712010-09-27 10:01:00 -0400233 if (pathStr)
234 env->DeleteLocalRef(pathStr);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400235 checkAndClearExceptionFromCallback(env, __FUNCTION__);
236 return result;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400237}
238
Mike Lockwoodd815f792010-07-12 08:49:01 -0400239void MyMtpDatabase::endSendObject(const char* path, MtpObjectHandle handle,
Mike Lockwood7a0bd172011-01-18 11:06:19 -0800240 MtpObjectFormat format, bool succeeded) {
Mike Lockwoodd815f792010-07-12 08:49:01 -0400241 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood88394712010-09-27 10:01:00 -0400242 jstring pathStr = env->NewStringUTF(path);
243 env->CallVoidMethod(mDatabase, method_endSendObject, pathStr,
Mike Lockwood7a0bd172011-01-18 11:06:19 -0800244 (jint)handle, (jint)format, (jboolean)succeeded);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400245
Mike Lockwood88394712010-09-27 10:01:00 -0400246 if (pathStr)
247 env->DeleteLocalRef(pathStr);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400248 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwoodd815f792010-07-12 08:49:01 -0400249}
250
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400251MtpObjectHandleList* MyMtpDatabase::getObjectList(MtpStorageID storageID,
252 MtpObjectFormat format,
253 MtpObjectHandle parent) {
254 JNIEnv* env = AndroidRuntime::getJNIEnv();
255 jintArray array = (jintArray)env->CallObjectMethod(mDatabase, method_getObjectList,
256 (jint)storageID, (jint)format, (jint)parent);
257 if (!array)
258 return NULL;
259 MtpObjectHandleList* list = new MtpObjectHandleList();
260 jint* handles = env->GetIntArrayElements(array, 0);
261 jsize length = env->GetArrayLength(array);
Mike Lockwood7a047c82010-08-02 10:52:20 -0400262 for (int i = 0; i < length; i++)
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400263 list->push(handles[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400264 env->ReleaseIntArrayElements(array, handles, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400265 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400266
267 checkAndClearExceptionFromCallback(env, __FUNCTION__);
268 return list;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400269}
270
Mike Lockwood7a047c82010-08-02 10:52:20 -0400271int MyMtpDatabase::getNumObjects(MtpStorageID storageID,
272 MtpObjectFormat format,
273 MtpObjectHandle parent) {
274 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400275 int result = env->CallIntMethod(mDatabase, method_getNumObjects,
Mike Lockwood7a047c82010-08-02 10:52:20 -0400276 (jint)storageID, (jint)format, (jint)parent);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400277
278 checkAndClearExceptionFromCallback(env, __FUNCTION__);
279 return result;
Mike Lockwood7a047c82010-08-02 10:52:20 -0400280}
281
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400282MtpObjectFormatList* MyMtpDatabase::getSupportedPlaybackFormats() {
283 JNIEnv* env = AndroidRuntime::getJNIEnv();
284 jintArray array = (jintArray)env->CallObjectMethod(mDatabase,
285 method_getSupportedPlaybackFormats);
286 if (!array)
287 return NULL;
288 MtpObjectFormatList* list = new MtpObjectFormatList();
289 jint* formats = env->GetIntArrayElements(array, 0);
290 jsize length = env->GetArrayLength(array);
291 for (int i = 0; i < length; i++)
292 list->push(formats[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400293 env->ReleaseIntArrayElements(array, formats, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400294 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400295
296 checkAndClearExceptionFromCallback(env, __FUNCTION__);
297 return list;
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400298}
299
300MtpObjectFormatList* MyMtpDatabase::getSupportedCaptureFormats() {
301 JNIEnv* env = AndroidRuntime::getJNIEnv();
302 jintArray array = (jintArray)env->CallObjectMethod(mDatabase,
303 method_getSupportedCaptureFormats);
304 if (!array)
305 return NULL;
306 MtpObjectFormatList* list = new MtpObjectFormatList();
307 jint* formats = env->GetIntArrayElements(array, 0);
308 jsize length = env->GetArrayLength(array);
309 for (int i = 0; i < length; i++)
310 list->push(formats[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400311 env->ReleaseIntArrayElements(array, formats, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400312 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400313
314 checkAndClearExceptionFromCallback(env, __FUNCTION__);
315 return list;
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400316}
317
318MtpObjectPropertyList* MyMtpDatabase::getSupportedObjectProperties(MtpObjectFormat format) {
319 JNIEnv* env = AndroidRuntime::getJNIEnv();
320 jintArray array = (jintArray)env->CallObjectMethod(mDatabase,
321 method_getSupportedObjectProperties, (jint)format);
322 if (!array)
323 return NULL;
324 MtpObjectPropertyList* list = new MtpObjectPropertyList();
325 jint* properties = env->GetIntArrayElements(array, 0);
326 jsize length = env->GetArrayLength(array);
327 for (int i = 0; i < length; i++)
328 list->push(properties[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400329 env->ReleaseIntArrayElements(array, properties, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400330 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400331
332 checkAndClearExceptionFromCallback(env, __FUNCTION__);
333 return list;
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400334}
335
336MtpDevicePropertyList* MyMtpDatabase::getSupportedDeviceProperties() {
337 JNIEnv* env = AndroidRuntime::getJNIEnv();
338 jintArray array = (jintArray)env->CallObjectMethod(mDatabase,
339 method_getSupportedDeviceProperties);
340 if (!array)
341 return NULL;
342 MtpDevicePropertyList* list = new MtpDevicePropertyList();
343 jint* properties = env->GetIntArrayElements(array, 0);
344 jsize length = env->GetArrayLength(array);
345 for (int i = 0; i < length; i++)
346 list->push(properties[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400347 env->ReleaseIntArrayElements(array, properties, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400348 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400349
350 checkAndClearExceptionFromCallback(env, __FUNCTION__);
351 return list;
Mike Lockwood4b322ce2010-08-10 07:37:50 -0400352}
353
Mike Lockwood828d19d2010-08-10 15:20:35 -0400354MtpResponseCode MyMtpDatabase::getObjectPropertyValue(MtpObjectHandle handle,
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400355 MtpObjectProperty property,
356 MtpDataPacket& packet) {
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400357 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400358 jobject list = env->CallObjectMethod(mDatabase, method_getObjectPropertyList,
Mike Lockwood7d7fb632010-12-01 18:46:23 -0500359 (jlong)handle, 0, (jlong)property, 0, 0);
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400360 MtpResponseCode result = env->GetIntField(list, field_mResult);
361 int count = env->GetIntField(list, field_mCount);
362 if (result == MTP_RESPONSE_OK && count != 1)
363 result = MTP_RESPONSE_GENERAL_ERROR;
364
365 if (result == MTP_RESPONSE_OK) {
366 jintArray objectHandlesArray = (jintArray)env->GetObjectField(list, field_mObjectHandles);
367 jintArray propertyCodesArray = (jintArray)env->GetObjectField(list, field_mPropertyCodes);
368 jintArray dataTypesArray = (jintArray)env->GetObjectField(list, field_mDataTypes);
369 jlongArray longValuesArray = (jlongArray)env->GetObjectField(list, field_mLongValues);
370 jobjectArray stringValuesArray = (jobjectArray)env->GetObjectField(list, field_mStringValues);
371
372 jint* objectHandles = env->GetIntArrayElements(objectHandlesArray, 0);
373 jint* propertyCodes = env->GetIntArrayElements(propertyCodesArray, 0);
374 jint* dataTypes = env->GetIntArrayElements(dataTypesArray, 0);
375 jlong* longValues = (longValuesArray ? env->GetLongArrayElements(longValuesArray, 0) : NULL);
376
377 int type = dataTypes[0];
378 jlong longValue = (longValues ? longValues[0] : 0);
379
380 // special case date properties, which are strings to MTP
381 // but stored internally as a uint64
382 if (property == MTP_PROPERTY_DATE_MODIFIED || property == MTP_PROPERTY_DATE_ADDED) {
383 char date[20];
384 formatDateTime(longValue, date, sizeof(date));
385 packet.putString(date);
386 goto out;
387 }
388 // release date is stored internally as just the year
389 if (property == MTP_PROPERTY_ORIGINAL_RELEASE_DATE) {
390 char date[20];
391 snprintf(date, sizeof(date), "%04lld0101T000000", longValue);
392 packet.putString(date);
393 goto out;
394 }
395
396 switch (type) {
397 case MTP_TYPE_INT8:
398 packet.putInt8(longValue);
399 break;
400 case MTP_TYPE_UINT8:
401 packet.putUInt8(longValue);
402 break;
403 case MTP_TYPE_INT16:
404 packet.putInt16(longValue);
405 break;
406 case MTP_TYPE_UINT16:
407 packet.putUInt16(longValue);
408 break;
409 case MTP_TYPE_INT32:
410 packet.putInt32(longValue);
411 break;
412 case MTP_TYPE_UINT32:
413 packet.putUInt32(longValue);
414 break;
415 case MTP_TYPE_INT64:
416 packet.putInt64(longValue);
417 break;
418 case MTP_TYPE_UINT64:
419 packet.putUInt64(longValue);
420 break;
421 case MTP_TYPE_INT128:
422 packet.putInt128(longValue);
423 break;
424 case MTP_TYPE_UINT128:
425 packet.putInt128(longValue);
426 break;
427 case MTP_TYPE_STR:
428 {
429 jstring stringValue = (jstring)env->GetObjectArrayElement(stringValuesArray, 0);
430 if (stringValue) {
431 const char* str = env->GetStringUTFChars(stringValue, NULL);
James Dong39774722011-04-06 11:57:48 -0700432 if (str == NULL) {
433 return MTP_RESPONSE_GENERAL_ERROR;
434 }
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400435 packet.putString(str);
436 env->ReleaseStringUTFChars(stringValue, str);
437 } else {
438 packet.putEmptyString();
439 }
440 break;
441 }
442 default:
443 LOGE("unsupported type in getObjectPropertyValue\n");
444 result = MTP_RESPONSE_INVALID_OBJECT_PROP_FORMAT;
445 }
446out:
447 env->ReleaseIntArrayElements(objectHandlesArray, objectHandles, 0);
448 env->ReleaseIntArrayElements(propertyCodesArray, propertyCodes, 0);
449 env->ReleaseIntArrayElements(dataTypesArray, dataTypes, 0);
450 if (longValues)
451 env->ReleaseLongArrayElements(longValuesArray, longValues, 0);
452
453 env->DeleteLocalRef(objectHandlesArray);
454 env->DeleteLocalRef(propertyCodesArray);
455 env->DeleteLocalRef(dataTypesArray);
456 if (longValuesArray)
457 env->DeleteLocalRef(longValuesArray);
458 if (stringValuesArray)
459 env->DeleteLocalRef(stringValuesArray);
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400460 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400461
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400462 env->DeleteLocalRef(list);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400463 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400464 return result;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400465}
466
Mike Lockwood828d19d2010-08-10 15:20:35 -0400467MtpResponseCode MyMtpDatabase::setObjectPropertyValue(MtpObjectHandle handle,
468 MtpObjectProperty property,
469 MtpDataPacket& packet) {
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400470 int type;
471
472 if (!getObjectPropertyInfo(property, type))
473 return MTP_RESPONSE_OBJECT_PROP_NOT_SUPPORTED;
474
475 JNIEnv* env = AndroidRuntime::getJNIEnv();
476 jlong longValue = 0;
477 jstring stringValue = NULL;
478
479 switch (type) {
480 case MTP_TYPE_INT8:
481 longValue = packet.getInt8();
482 break;
483 case MTP_TYPE_UINT8:
484 longValue = packet.getUInt8();
485 break;
486 case MTP_TYPE_INT16:
487 longValue = packet.getInt16();
488 break;
489 case MTP_TYPE_UINT16:
490 longValue = packet.getUInt16();
491 break;
492 case MTP_TYPE_INT32:
493 longValue = packet.getInt32();
494 break;
495 case MTP_TYPE_UINT32:
496 longValue = packet.getUInt32();
497 break;
498 case MTP_TYPE_INT64:
499 longValue = packet.getInt64();
500 break;
501 case MTP_TYPE_UINT64:
502 longValue = packet.getUInt64();
503 break;
504 case MTP_TYPE_STR:
505 {
506 MtpStringBuffer buffer;
507 packet.getString(buffer);
508 stringValue = env->NewStringUTF((const char *)buffer);
509 break;
510 }
511 default:
512 LOGE("unsupported type in getObjectPropertyValue\n");
513 return MTP_RESPONSE_INVALID_OBJECT_PROP_FORMAT;
514 }
515
516 jint result = env->CallIntMethod(mDatabase, method_setObjectProperty,
517 (jint)handle, (jint)property, longValue, stringValue);
Mike Lockwood88394712010-09-27 10:01:00 -0400518 if (stringValue)
519 env->DeleteLocalRef(stringValue);
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400520
521 checkAndClearExceptionFromCallback(env, __FUNCTION__);
522 return result;
Mike Lockwood828d19d2010-08-10 15:20:35 -0400523}
524
525MtpResponseCode MyMtpDatabase::getDevicePropertyValue(MtpDeviceProperty property,
526 MtpDataPacket& packet) {
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400527 int type;
528
529 if (!getDevicePropertyInfo(property, type))
530 return MTP_RESPONSE_DEVICE_PROP_NOT_SUPPORTED;
531
532 JNIEnv* env = AndroidRuntime::getJNIEnv();
533 jint result = env->CallIntMethod(mDatabase, method_getDeviceProperty,
534 (jint)property, mLongBuffer, mStringBuffer);
535 if (result != MTP_RESPONSE_OK) {
536 checkAndClearExceptionFromCallback(env, __FUNCTION__);
537 return result;
538 }
539
540 jlong* longValues = env->GetLongArrayElements(mLongBuffer, 0);
541 jlong longValue = longValues[0];
542 env->ReleaseLongArrayElements(mLongBuffer, longValues, 0);
543
544 switch (type) {
545 case MTP_TYPE_INT8:
546 packet.putInt8(longValue);
547 break;
548 case MTP_TYPE_UINT8:
549 packet.putUInt8(longValue);
550 break;
551 case MTP_TYPE_INT16:
552 packet.putInt16(longValue);
553 break;
554 case MTP_TYPE_UINT16:
555 packet.putUInt16(longValue);
556 break;
557 case MTP_TYPE_INT32:
558 packet.putInt32(longValue);
559 break;
560 case MTP_TYPE_UINT32:
561 packet.putUInt32(longValue);
562 break;
563 case MTP_TYPE_INT64:
564 packet.putInt64(longValue);
565 break;
566 case MTP_TYPE_UINT64:
567 packet.putUInt64(longValue);
568 break;
569 case MTP_TYPE_INT128:
570 packet.putInt128(longValue);
571 break;
572 case MTP_TYPE_UINT128:
573 packet.putInt128(longValue);
574 break;
575 case MTP_TYPE_STR:
576 {
577 jchar* str = env->GetCharArrayElements(mStringBuffer, 0);
578 packet.putString(str);
579 env->ReleaseCharArrayElements(mStringBuffer, str, 0);
580 break;
581 }
582 default:
583 LOGE("unsupported type in getDevicePropertyValue\n");
584 return MTP_RESPONSE_INVALID_DEVICE_PROP_FORMAT;
585 }
586
587 checkAndClearExceptionFromCallback(env, __FUNCTION__);
588 return MTP_RESPONSE_OK;
Mike Lockwood828d19d2010-08-10 15:20:35 -0400589}
590
591MtpResponseCode MyMtpDatabase::setDevicePropertyValue(MtpDeviceProperty property,
592 MtpDataPacket& packet) {
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400593 int type;
594
595 if (!getDevicePropertyInfo(property, type))
596 return MTP_RESPONSE_DEVICE_PROP_NOT_SUPPORTED;
597
598 JNIEnv* env = AndroidRuntime::getJNIEnv();
599 jlong longValue = 0;
600 jstring stringValue = NULL;
601
602 switch (type) {
603 case MTP_TYPE_INT8:
604 longValue = packet.getInt8();
605 break;
606 case MTP_TYPE_UINT8:
607 longValue = packet.getUInt8();
608 break;
609 case MTP_TYPE_INT16:
610 longValue = packet.getInt16();
611 break;
612 case MTP_TYPE_UINT16:
613 longValue = packet.getUInt16();
614 break;
615 case MTP_TYPE_INT32:
616 longValue = packet.getInt32();
617 break;
618 case MTP_TYPE_UINT32:
619 longValue = packet.getUInt32();
620 break;
621 case MTP_TYPE_INT64:
622 longValue = packet.getInt64();
623 break;
624 case MTP_TYPE_UINT64:
625 longValue = packet.getUInt64();
626 break;
627 case MTP_TYPE_STR:
628 {
629 MtpStringBuffer buffer;
630 packet.getString(buffer);
631 stringValue = env->NewStringUTF((const char *)buffer);
632 break;
633 }
634 default:
635 LOGE("unsupported type in setDevicePropertyValue\n");
636 return MTP_RESPONSE_INVALID_OBJECT_PROP_FORMAT;
637 }
638
639 jint result = env->CallIntMethod(mDatabase, method_setDeviceProperty,
640 (jint)property, longValue, stringValue);
Mike Lockwood88394712010-09-27 10:01:00 -0400641 if (stringValue)
642 env->DeleteLocalRef(stringValue);
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400643
644 checkAndClearExceptionFromCallback(env, __FUNCTION__);
645 return result;
Mike Lockwood828d19d2010-08-10 15:20:35 -0400646}
647
648MtpResponseCode MyMtpDatabase::resetDeviceProperty(MtpDeviceProperty property) {
649 return -1;
650}
651
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400652MtpResponseCode MyMtpDatabase::getObjectPropertyList(MtpObjectHandle handle,
Mike Lockwood7d7fb632010-12-01 18:46:23 -0500653 uint32_t format, uint32_t property,
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400654 int groupCode, int depth,
655 MtpDataPacket& packet) {
656 JNIEnv* env = AndroidRuntime::getJNIEnv();
657 jobject list = env->CallObjectMethod(mDatabase, method_getObjectPropertyList,
Mike Lockwood7d7fb632010-12-01 18:46:23 -0500658 (jlong)handle, (jint)format, (jlong)property, (jint)groupCode, (jint)depth);
659 checkAndClearExceptionFromCallback(env, __FUNCTION__);
660 if (!list)
661 return MTP_RESPONSE_GENERAL_ERROR;
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400662 int count = env->GetIntField(list, field_mCount);
663 MtpResponseCode result = env->GetIntField(list, field_mResult);
664
665 packet.putUInt32(count);
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400666 if (count > 0) {
667 jintArray objectHandlesArray = (jintArray)env->GetObjectField(list, field_mObjectHandles);
668 jintArray propertyCodesArray = (jintArray)env->GetObjectField(list, field_mPropertyCodes);
669 jintArray dataTypesArray = (jintArray)env->GetObjectField(list, field_mDataTypes);
670 jlongArray longValuesArray = (jlongArray)env->GetObjectField(list, field_mLongValues);
671 jobjectArray stringValuesArray = (jobjectArray)env->GetObjectField(list, field_mStringValues);
672
673 jint* objectHandles = env->GetIntArrayElements(objectHandlesArray, 0);
674 jint* propertyCodes = env->GetIntArrayElements(propertyCodesArray, 0);
675 jint* dataTypes = env->GetIntArrayElements(dataTypesArray, 0);
676 jlong* longValues = (longValuesArray ? env->GetLongArrayElements(longValuesArray, 0) : NULL);
677
678 for (int i = 0; i < count; i++) {
679 packet.putUInt32(objectHandles[i]);
680 packet.putUInt16(propertyCodes[i]);
681 int type = dataTypes[i];
682 packet.putUInt16(type);
683
684 switch (type) {
685 case MTP_TYPE_INT8:
686 packet.putInt8(longValues[i]);
687 break;
688 case MTP_TYPE_UINT8:
689 packet.putUInt8(longValues[i]);
690 break;
691 case MTP_TYPE_INT16:
692 packet.putInt16(longValues[i]);
693 break;
694 case MTP_TYPE_UINT16:
695 packet.putUInt16(longValues[i]);
696 break;
697 case MTP_TYPE_INT32:
698 packet.putInt32(longValues[i]);
699 break;
700 case MTP_TYPE_UINT32:
701 packet.putUInt32(longValues[i]);
702 break;
703 case MTP_TYPE_INT64:
704 packet.putInt64(longValues[i]);
705 break;
706 case MTP_TYPE_UINT64:
707 packet.putUInt64(longValues[i]);
708 break;
709 case MTP_TYPE_INT128:
710 packet.putInt128(longValues[i]);
711 break;
712 case MTP_TYPE_UINT128:
713 packet.putUInt128(longValues[i]);
714 break;
715 case MTP_TYPE_STR: {
716 jstring value = (jstring)env->GetObjectArrayElement(stringValuesArray, i);
Mike Lockwood2711e492010-12-11 11:24:37 -0800717 const char *valueStr = (value ? env->GetStringUTFChars(value, NULL) : NULL);
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -0400718 if (valueStr) {
719 packet.putString(valueStr);
720 env->ReleaseStringUTFChars(value, valueStr);
721 } else {
722 packet.putEmptyString();
723 }
724 env->DeleteLocalRef(value);
725 break;
726 }
727 default:
728 LOGE("bad or unsupported data type in MyMtpDatabase::getObjectPropertyList");
729 break;
730 }
731 }
732
733 env->ReleaseIntArrayElements(objectHandlesArray, objectHandles, 0);
734 env->ReleaseIntArrayElements(propertyCodesArray, propertyCodes, 0);
735 env->ReleaseIntArrayElements(dataTypesArray, dataTypes, 0);
736 if (longValues)
737 env->ReleaseLongArrayElements(longValuesArray, longValues, 0);
738
739 env->DeleteLocalRef(objectHandlesArray);
740 env->DeleteLocalRef(propertyCodesArray);
741 env->DeleteLocalRef(dataTypesArray);
742 if (longValuesArray)
743 env->DeleteLocalRef(longValuesArray);
744 if (stringValuesArray)
745 env->DeleteLocalRef(stringValuesArray);
746 }
747
748 env->DeleteLocalRef(list);
749 checkAndClearExceptionFromCallback(env, __FUNCTION__);
750 return result;
751}
752
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400753MtpResponseCode MyMtpDatabase::getObjectInfo(MtpObjectHandle handle,
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700754 MtpObjectInfo& info) {
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400755 char date[20];
756
757 JNIEnv* env = AndroidRuntime::getJNIEnv();
758 jboolean result = env->CallBooleanMethod(mDatabase, method_getObjectInfo,
759 (jint)handle, mIntBuffer, mStringBuffer, mLongBuffer);
760 if (!result)
761 return MTP_RESPONSE_INVALID_OBJECT_HANDLE;
762
763 jint* intValues = env->GetIntArrayElements(mIntBuffer, 0);
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700764 info.mStorageID = intValues[0];
765 info.mFormat = intValues[1];
766 info.mParent = intValues[2];
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400767 env->ReleaseIntArrayElements(mIntBuffer, intValues, 0);
768
769 jlong* longValues = env->GetLongArrayElements(mLongBuffer, 0);
770 uint64_t size = longValues[0];
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700771 info.mCompressedSize = (size > 0xFFFFFFFFLL ? 0xFFFFFFFF : size);
772 info.mDateModified = longValues[1];
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400773 env->ReleaseLongArrayElements(mLongBuffer, longValues, 0);
774
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700775// info.mAssociationType = (format == MTP_FORMAT_ASSOCIATION ?
Mike Lockwood828d19d2010-08-10 15:20:35 -0400776// MTP_ASSOCIATION_TYPE_GENERIC_FOLDER :
777// MTP_ASSOCIATION_TYPE_UNDEFINED);
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700778 info.mAssociationType = MTP_ASSOCIATION_TYPE_UNDEFINED;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400779
780 jchar* str = env->GetCharArrayElements(mStringBuffer, 0);
Mike Lockwood9df53fae2011-04-21 17:05:55 -0700781 MtpString temp(str);
782 info.mName = strdup((const char *)temp);
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400783 env->ReleaseCharArrayElements(mStringBuffer, str, 0);
784
Mike Lockwoodc89f2222011-04-24 18:40:17 -0700785 // read EXIF data for thumbnail information
786 if (info.mFormat == MTP_FORMAT_EXIF_JPEG || info.mFormat == MTP_FORMAT_JFIF) {
787 MtpString path;
788 int64_t length;
789 MtpObjectFormat format;
790 if (getObjectFilePath(handle, path, length, format) == MTP_RESPONSE_OK) {
791 ResetJpgfile();
792 // Start with an empty image information structure.
793 memset(&ImageInfo, 0, sizeof(ImageInfo));
794 ImageInfo.FlashUsed = -1;
795 ImageInfo.MeteringMode = -1;
796 ImageInfo.Whitebalance = -1;
797 strncpy(ImageInfo.FileName, (const char *)path, PATH_MAX);
798 if (ReadJpegFile((const char*)path, READ_METADATA)) {
799 Section_t* section = FindSection(M_EXIF);
800 if (section) {
801 info.mThumbCompressedSize = ImageInfo.ThumbnailSize;
802 info.mThumbFormat = MTP_FORMAT_EXIF_JPEG;
803 info.mImagePixWidth = ImageInfo.Width;
804 info.mImagePixHeight = ImageInfo.Height;
805 }
806 }
807 DiscardData();
808 }
809 }
810
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400811 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400812 return MTP_RESPONSE_OK;
813}
814
Mike Lockwoodc89f2222011-04-24 18:40:17 -0700815void* MyMtpDatabase::getThumbnail(MtpObjectHandle handle, size_t& outThumbSize) {
816 MtpString path;
817 int64_t length;
818 MtpObjectFormat format;
819 void* result = NULL;
820 outThumbSize = 0;
821
822 if (getObjectFilePath(handle, path, length, format) == MTP_RESPONSE_OK
823 && (format == MTP_FORMAT_EXIF_JPEG || format == MTP_FORMAT_JFIF)) {
824 ResetJpgfile();
825 // Start with an empty image information structure.
826 memset(&ImageInfo, 0, sizeof(ImageInfo));
827 ImageInfo.FlashUsed = -1;
828 ImageInfo.MeteringMode = -1;
829 ImageInfo.Whitebalance = -1;
830 strncpy(ImageInfo.FileName, (const char *)path, PATH_MAX);
831 if (ReadJpegFile((const char*)path, READ_METADATA)) {
832 Section_t* section = FindSection(M_EXIF);
833 if (section) {
834 outThumbSize = ImageInfo.ThumbnailSize;
835 result = malloc(outThumbSize);
836 if (result)
837 memcpy(result, section->Data + ImageInfo.ThumbnailOffset + 8, outThumbSize);
838 }
839 DiscardData();
840 }
841 }
842
843 return result;
844}
845
Mike Lockwood59c777a2010-08-02 10:37:41 -0400846MtpResponseCode MyMtpDatabase::getObjectFilePath(MtpObjectHandle handle,
Mike Lockwood365e03e2010-12-08 16:08:01 -0800847 MtpString& outFilePath,
848 int64_t& outFileLength,
849 MtpObjectFormat& outFormat) {
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400850 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood59c777a2010-08-02 10:37:41 -0400851 jint result = env->CallIntMethod(mDatabase, method_getObjectFilePath,
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400852 (jint)handle, mStringBuffer, mLongBuffer);
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400853 if (result != MTP_RESPONSE_OK) {
854 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwood59c777a2010-08-02 10:37:41 -0400855 return result;
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400856 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400857
858 jchar* str = env->GetCharArrayElements(mStringBuffer, 0);
Mike Lockwood365e03e2010-12-08 16:08:01 -0800859 outFilePath.setTo(str, strlen16(str));
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400860 env->ReleaseCharArrayElements(mStringBuffer, str, 0);
861
862 jlong* longValues = env->GetLongArrayElements(mLongBuffer, 0);
Mike Lockwood365e03e2010-12-08 16:08:01 -0800863 outFileLength = longValues[0];
864 outFormat = longValues[1];
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400865 env->ReleaseLongArrayElements(mLongBuffer, longValues, 0);
Elliott Hughes15dd15f2011-04-08 17:42:34 -0700866
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400867 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwood59c777a2010-08-02 10:37:41 -0400868 return result;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400869}
870
Mike Lockwood59c777a2010-08-02 10:37:41 -0400871MtpResponseCode MyMtpDatabase::deleteFile(MtpObjectHandle handle) {
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400872 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400873 MtpResponseCode result = env->CallIntMethod(mDatabase, method_deleteFile, (jint)handle);
874
875 checkAndClearExceptionFromCallback(env, __FUNCTION__);
876 return result;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400877}
878
879struct PropertyTableEntry {
880 MtpObjectProperty property;
881 int type;
882};
883
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400884static const PropertyTableEntry kObjectPropertyTable[] = {
Mike Lockwoodd3bfecb2010-09-23 23:04:28 -0400885 { MTP_PROPERTY_STORAGE_ID, MTP_TYPE_UINT32 },
886 { MTP_PROPERTY_OBJECT_FORMAT, MTP_TYPE_UINT16 },
887 { MTP_PROPERTY_PROTECTION_STATUS, MTP_TYPE_UINT16 },
888 { MTP_PROPERTY_OBJECT_SIZE, MTP_TYPE_UINT64 },
889 { MTP_PROPERTY_OBJECT_FILE_NAME, MTP_TYPE_STR },
890 { MTP_PROPERTY_DATE_MODIFIED, MTP_TYPE_STR },
891 { MTP_PROPERTY_PARENT_OBJECT, MTP_TYPE_UINT32 },
892 { MTP_PROPERTY_PERSISTENT_UID, MTP_TYPE_UINT128 },
893 { MTP_PROPERTY_NAME, MTP_TYPE_STR },
Mike Lockwoodae078f72010-09-26 12:35:51 -0400894 { MTP_PROPERTY_DISPLAY_NAME, MTP_TYPE_STR },
895 { MTP_PROPERTY_DATE_ADDED, MTP_TYPE_STR },
896 { MTP_PROPERTY_ARTIST, MTP_TYPE_STR },
897 { MTP_PROPERTY_ALBUM_NAME, MTP_TYPE_STR },
898 { MTP_PROPERTY_ALBUM_ARTIST, MTP_TYPE_STR },
899 { MTP_PROPERTY_TRACK, MTP_TYPE_UINT16 },
900 { MTP_PROPERTY_ORIGINAL_RELEASE_DATE, MTP_TYPE_STR },
901 { MTP_PROPERTY_GENRE, MTP_TYPE_STR },
902 { MTP_PROPERTY_COMPOSER, MTP_TYPE_STR },
903 { MTP_PROPERTY_DURATION, MTP_TYPE_UINT32 },
904 { MTP_PROPERTY_DESCRIPTION, MTP_TYPE_STR },
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400905};
906
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400907static const PropertyTableEntry kDevicePropertyTable[] = {
Mike Lockwoodea93fa12010-12-07 10:41:35 -0800908 { MTP_DEVICE_PROPERTY_SYNCHRONIZATION_PARTNER, MTP_TYPE_STR },
909 { MTP_DEVICE_PROPERTY_DEVICE_FRIENDLY_NAME, MTP_TYPE_STR },
910 { MTP_DEVICE_PROPERTY_IMAGE_SIZE, MTP_TYPE_STR },
Mike Lockwood59e3f0d2010-09-02 14:57:30 -0400911};
912
913bool MyMtpDatabase::getObjectPropertyInfo(MtpObjectProperty property, int& type) {
914 int count = sizeof(kObjectPropertyTable) / sizeof(kObjectPropertyTable[0]);
915 const PropertyTableEntry* entry = kObjectPropertyTable;
916 for (int i = 0; i < count; i++, entry++) {
917 if (entry->property == property) {
918 type = entry->type;
919 return true;
920 }
921 }
922 return false;
923}
924
925bool MyMtpDatabase::getDevicePropertyInfo(MtpDeviceProperty property, int& type) {
926 int count = sizeof(kDevicePropertyTable) / sizeof(kDevicePropertyTable[0]);
927 const PropertyTableEntry* entry = kDevicePropertyTable;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400928 for (int i = 0; i < count; i++, entry++) {
929 if (entry->property == property) {
930 type = entry->type;
931 return true;
932 }
933 }
934 return false;
935}
936
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400937MtpObjectHandleList* MyMtpDatabase::getObjectReferences(MtpObjectHandle handle) {
938 JNIEnv* env = AndroidRuntime::getJNIEnv();
939 jintArray array = (jintArray)env->CallObjectMethod(mDatabase, method_getObjectReferences,
940 (jint)handle);
941 if (!array)
942 return NULL;
943 MtpObjectHandleList* list = new MtpObjectHandleList();
944 jint* handles = env->GetIntArrayElements(array, 0);
945 jsize length = env->GetArrayLength(array);
946 for (int i = 0; i < length; i++)
947 list->push(handles[i]);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400948 env->ReleaseIntArrayElements(array, handles, 0);
Mike Lockwood88394712010-09-27 10:01:00 -0400949 env->DeleteLocalRef(array);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400950
951 checkAndClearExceptionFromCallback(env, __FUNCTION__);
952 return list;
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400953}
954
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400955MtpResponseCode MyMtpDatabase::setObjectReferences(MtpObjectHandle handle,
956 MtpObjectHandleList* references) {
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400957 JNIEnv* env = AndroidRuntime::getJNIEnv();
958 int count = references->size();
959 jintArray array = env->NewIntArray(count);
960 if (!array) {
961 LOGE("out of memory in setObjectReferences");
962 return false;
963 }
964 jint* handles = env->GetIntArrayElements(array, 0);
965 for (int i = 0; i < count; i++)
966 handles[i] = (*references)[i];
967 env->ReleaseIntArrayElements(array, handles, 0);
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400968 MtpResponseCode result = env->CallIntMethod(mDatabase, method_setObjectReferences,
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400969 (jint)handle, array);
Mike Lockwood88394712010-09-27 10:01:00 -0400970 env->DeleteLocalRef(array);
Mike Lockwood9a2046f2010-08-03 15:30:09 -0400971
Mike Lockwood0a7fa0a2010-08-24 11:25:28 -0400972 checkAndClearExceptionFromCallback(env, __FUNCTION__);
973 return result;
Mike Lockwoodd21eac92010-07-03 00:44:05 -0400974}
975
Mike Lockwood828d19d2010-08-10 15:20:35 -0400976MtpProperty* MyMtpDatabase::getObjectPropertyDesc(MtpObjectProperty property,
977 MtpObjectFormat format) {
978 MtpProperty* result = NULL;
979 switch (property) {
980 case MTP_PROPERTY_OBJECT_FORMAT:
Mike Lockwood9b5e9c42010-12-07 18:53:50 -0800981 // use format as default value
982 result = new MtpProperty(property, MTP_TYPE_UINT16, false, format);
983 break;
Mike Lockwood828d19d2010-08-10 15:20:35 -0400984 case MTP_PROPERTY_PROTECTION_STATUS:
Mike Lockwoodae078f72010-09-26 12:35:51 -0400985 case MTP_PROPERTY_TRACK:
Mike Lockwood828d19d2010-08-10 15:20:35 -0400986 result = new MtpProperty(property, MTP_TYPE_UINT16);
987 break;
988 case MTP_PROPERTY_STORAGE_ID:
989 case MTP_PROPERTY_PARENT_OBJECT:
Mike Lockwoodae078f72010-09-26 12:35:51 -0400990 case MTP_PROPERTY_DURATION:
Mike Lockwood828d19d2010-08-10 15:20:35 -0400991 result = new MtpProperty(property, MTP_TYPE_UINT32);
992 break;
993 case MTP_PROPERTY_OBJECT_SIZE:
994 result = new MtpProperty(property, MTP_TYPE_UINT64);
995 break;
996 case MTP_PROPERTY_PERSISTENT_UID:
997 result = new MtpProperty(property, MTP_TYPE_UINT128);
998 break;
Mike Lockwoodd3bfecb2010-09-23 23:04:28 -0400999 case MTP_PROPERTY_NAME:
Mike Lockwoodae078f72010-09-26 12:35:51 -04001000 case MTP_PROPERTY_DISPLAY_NAME:
Mike Lockwoodae078f72010-09-26 12:35:51 -04001001 case MTP_PROPERTY_ARTIST:
1002 case MTP_PROPERTY_ALBUM_NAME:
1003 case MTP_PROPERTY_ALBUM_ARTIST:
Mike Lockwoodae078f72010-09-26 12:35:51 -04001004 case MTP_PROPERTY_GENRE:
1005 case MTP_PROPERTY_COMPOSER:
1006 case MTP_PROPERTY_DESCRIPTION:
Mike Lockwood828d19d2010-08-10 15:20:35 -04001007 result = new MtpProperty(property, MTP_TYPE_STR);
1008 break;
Mike Lockwood5b19af02010-11-23 18:38:55 -05001009 case MTP_PROPERTY_DATE_MODIFIED:
1010 case MTP_PROPERTY_DATE_ADDED:
1011 case MTP_PROPERTY_ORIGINAL_RELEASE_DATE:
1012 result = new MtpProperty(property, MTP_TYPE_STR);
1013 result->setFormDateTime();
1014 break;
Mike Lockwood5ebac832010-10-12 11:33:47 -04001015 case MTP_PROPERTY_OBJECT_FILE_NAME:
Mike Lockwood6a6a3af2010-10-12 14:19:51 -04001016 // We allow renaming files and folders
1017 result = new MtpProperty(property, MTP_TYPE_STR, true);
Mike Lockwood5ebac832010-10-12 11:33:47 -04001018 break;
Mike Lockwood828d19d2010-08-10 15:20:35 -04001019 }
1020
1021 return result;
1022}
1023
1024MtpProperty* MyMtpDatabase::getDevicePropertyDesc(MtpDeviceProperty property) {
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001025 JNIEnv* env = AndroidRuntime::getJNIEnv();
Mike Lockwood59e3f0d2010-09-02 14:57:30 -04001026 MtpProperty* result = NULL;
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001027 bool writable = false;
1028
Mike Lockwood59e3f0d2010-09-02 14:57:30 -04001029 switch (property) {
1030 case MTP_DEVICE_PROPERTY_SYNCHRONIZATION_PARTNER:
1031 case MTP_DEVICE_PROPERTY_DEVICE_FRIENDLY_NAME:
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001032 writable = true;
1033 // fall through
1034 case MTP_DEVICE_PROPERTY_IMAGE_SIZE:
1035 result = new MtpProperty(property, MTP_TYPE_STR, writable);
Mike Lockwooda2a21282010-09-25 21:21:05 -04001036
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001037 // get current value
Mike Lockwooda2a21282010-09-25 21:21:05 -04001038 jint ret = env->CallIntMethod(mDatabase, method_getDeviceProperty,
1039 (jint)property, mLongBuffer, mStringBuffer);
1040 if (ret == MTP_RESPONSE_OK) {
1041 jchar* str = env->GetCharArrayElements(mStringBuffer, 0);
1042 result->setCurrentValue(str);
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001043 // for read-only properties it is safe to assume current value is default value
1044 if (!writable)
1045 result->setDefaultValue(str);
Mike Lockwooda2a21282010-09-25 21:21:05 -04001046 env->ReleaseCharArrayElements(mStringBuffer, str, 0);
1047 } else {
1048 LOGE("unable to read device property, response: %04X", ret);
1049 }
Mike Lockwood59e3f0d2010-09-02 14:57:30 -04001050 break;
1051 }
1052
Mike Lockwoodea93fa12010-12-07 10:41:35 -08001053 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwood59e3f0d2010-09-02 14:57:30 -04001054 return result;
Mike Lockwood828d19d2010-08-10 15:20:35 -04001055}
1056
Mike Lockwood2837eef2010-08-31 16:25:12 -04001057void MyMtpDatabase::sessionStarted() {
1058 JNIEnv* env = AndroidRuntime::getJNIEnv();
1059 env->CallVoidMethod(mDatabase, method_sessionStarted);
1060 checkAndClearExceptionFromCallback(env, __FUNCTION__);
1061}
1062
1063void MyMtpDatabase::sessionEnded() {
1064 JNIEnv* env = AndroidRuntime::getJNIEnv();
1065 env->CallVoidMethod(mDatabase, method_sessionEnded);
1066 checkAndClearExceptionFromCallback(env, __FUNCTION__);
1067}
1068
Mike Lockwoodff164a72010-07-15 15:01:17 -04001069#endif // HAVE_ANDROID_OS
1070
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001071// ----------------------------------------------------------------------------
1072
1073static void
Mike Lockwood0cd01362010-12-30 11:54:33 -05001074android_mtp_MtpDatabase_setup(JNIEnv *env, jobject thiz)
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001075{
Mike Lockwoodff164a72010-07-15 15:01:17 -04001076#ifdef HAVE_ANDROID_OS
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001077 MyMtpDatabase* database = new MyMtpDatabase(env, thiz);
1078 env->SetIntField(thiz, field_context, (int)database);
1079 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwoodff164a72010-07-15 15:01:17 -04001080#endif
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001081}
1082
1083static void
Mike Lockwood0cd01362010-12-30 11:54:33 -05001084android_mtp_MtpDatabase_finalize(JNIEnv *env, jobject thiz)
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001085{
Mike Lockwoodff164a72010-07-15 15:01:17 -04001086#ifdef HAVE_ANDROID_OS
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001087 MyMtpDatabase* database = (MyMtpDatabase *)env->GetIntField(thiz, field_context);
1088 database->cleanup(env);
1089 delete database;
1090 env->SetIntField(thiz, field_context, 0);
1091 checkAndClearExceptionFromCallback(env, __FUNCTION__);
Mike Lockwoodff164a72010-07-15 15:01:17 -04001092#endif
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001093}
1094
Mike Lockwood31599912010-11-15 13:43:30 -05001095static jstring
Mike Lockwood0cd01362010-12-30 11:54:33 -05001096android_mtp_MtpPropertyGroup_format_date_time(JNIEnv *env, jobject thiz, jlong seconds)
Mike Lockwood31599912010-11-15 13:43:30 -05001097{
1098#ifdef HAVE_ANDROID_OS
1099 char date[20];
1100 formatDateTime(seconds, date, sizeof(date));
1101 return env->NewStringUTF(date);
1102#else
1103 return NULL;
1104#endif
1105}
1106
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001107// ----------------------------------------------------------------------------
1108
Mike Lockwood7d7fb632010-12-01 18:46:23 -05001109static JNINativeMethod gMtpDatabaseMethods[] = {
Mike Lockwood0cd01362010-12-30 11:54:33 -05001110 {"native_setup", "()V", (void *)android_mtp_MtpDatabase_setup},
1111 {"native_finalize", "()V", (void *)android_mtp_MtpDatabase_finalize},
Mike Lockwood7d7fb632010-12-01 18:46:23 -05001112};
1113
1114static JNINativeMethod gMtpPropertyGroupMethods[] = {
Mike Lockwood31599912010-11-15 13:43:30 -05001115 {"format_date_time", "(J)Ljava/lang/String;",
Mike Lockwood0cd01362010-12-30 11:54:33 -05001116 (void *)android_mtp_MtpPropertyGroup_format_date_time},
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001117};
1118
Mike Lockwood0cd01362010-12-30 11:54:33 -05001119static const char* const kClassPathName = "android/mtp/MtpDatabase";
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001120
Mike Lockwood0cd01362010-12-30 11:54:33 -05001121int register_android_mtp_MtpDatabase(JNIEnv *env)
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001122{
1123 jclass clazz;
1124
Mike Lockwood0cd01362010-12-30 11:54:33 -05001125 clazz = env->FindClass("android/mtp/MtpDatabase");
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001126 if (clazz == NULL) {
Mike Lockwood0cd01362010-12-30 11:54:33 -05001127 LOGE("Can't find android/mtp/MtpDatabase");
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001128 return -1;
1129 }
Mike Lockwoodd815f792010-07-12 08:49:01 -04001130 method_beginSendObject = env->GetMethodID(clazz, "beginSendObject", "(Ljava/lang/String;IIIJJ)I");
1131 if (method_beginSendObject == NULL) {
1132 LOGE("Can't find beginSendObject");
1133 return -1;
1134 }
Mike Lockwood7a0bd172011-01-18 11:06:19 -08001135 method_endSendObject = env->GetMethodID(clazz, "endSendObject", "(Ljava/lang/String;IIZ)V");
Mike Lockwoodd815f792010-07-12 08:49:01 -04001136 if (method_endSendObject == NULL) {
1137 LOGE("Can't find endSendObject");
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001138 return -1;
1139 }
1140 method_getObjectList = env->GetMethodID(clazz, "getObjectList", "(III)[I");
1141 if (method_getObjectList == NULL) {
1142 LOGE("Can't find getObjectList");
1143 return -1;
1144 }
Mike Lockwood7a047c82010-08-02 10:52:20 -04001145 method_getNumObjects = env->GetMethodID(clazz, "getNumObjects", "(III)I");
1146 if (method_getNumObjects == NULL) {
1147 LOGE("Can't find getNumObjects");
1148 return -1;
1149 }
Mike Lockwood4b322ce2010-08-10 07:37:50 -04001150 method_getSupportedPlaybackFormats = env->GetMethodID(clazz, "getSupportedPlaybackFormats", "()[I");
1151 if (method_getSupportedPlaybackFormats == NULL) {
1152 LOGE("Can't find getSupportedPlaybackFormats");
1153 return -1;
1154 }
1155 method_getSupportedCaptureFormats = env->GetMethodID(clazz, "getSupportedCaptureFormats", "()[I");
1156 if (method_getSupportedCaptureFormats == NULL) {
1157 LOGE("Can't find getSupportedCaptureFormats");
1158 return -1;
1159 }
1160 method_getSupportedObjectProperties = env->GetMethodID(clazz, "getSupportedObjectProperties", "(I)[I");
1161 if (method_getSupportedObjectProperties == NULL) {
1162 LOGE("Can't find getSupportedObjectProperties");
1163 return -1;
1164 }
1165 method_getSupportedDeviceProperties = env->GetMethodID(clazz, "getSupportedDeviceProperties", "()[I");
1166 if (method_getSupportedDeviceProperties == NULL) {
1167 LOGE("Can't find getSupportedDeviceProperties");
1168 return -1;
1169 }
Mike Lockwood59e3f0d2010-09-02 14:57:30 -04001170 method_setObjectProperty = env->GetMethodID(clazz, "setObjectProperty", "(IIJLjava/lang/String;)I");
1171 if (method_setObjectProperty == NULL) {
1172 LOGE("Can't find setObjectProperty");
1173 return -1;
1174 }
1175 method_getDeviceProperty = env->GetMethodID(clazz, "getDeviceProperty", "(I[J[C)I");
1176 if (method_getDeviceProperty == NULL) {
1177 LOGE("Can't find getDeviceProperty");
1178 return -1;
1179 }
1180 method_setDeviceProperty = env->GetMethodID(clazz, "setDeviceProperty", "(IJLjava/lang/String;)I");
1181 if (method_setDeviceProperty == NULL) {
1182 LOGE("Can't find setDeviceProperty");
1183 return -1;
1184 }
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -04001185 method_getObjectPropertyList = env->GetMethodID(clazz, "getObjectPropertyList",
Mike Lockwood0cd01362010-12-30 11:54:33 -05001186 "(JIJII)Landroid/mtp/MtpPropertyList;");
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -04001187 if (method_getObjectPropertyList == NULL) {
1188 LOGE("Can't find getObjectPropertyList");
1189 return -1;
1190 }
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001191 method_getObjectInfo = env->GetMethodID(clazz, "getObjectInfo", "(I[I[C[J)Z");
1192 if (method_getObjectInfo == NULL) {
1193 LOGE("Can't find getObjectInfo");
1194 return -1;
1195 }
Mike Lockwood59c777a2010-08-02 10:37:41 -04001196 method_getObjectFilePath = env->GetMethodID(clazz, "getObjectFilePath", "(I[C[J)I");
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001197 if (method_getObjectFilePath == NULL) {
1198 LOGE("Can't find getObjectFilePath");
1199 return -1;
1200 }
Mike Lockwood59c777a2010-08-02 10:37:41 -04001201 method_deleteFile = env->GetMethodID(clazz, "deleteFile", "(I)I");
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001202 if (method_deleteFile == NULL) {
1203 LOGE("Can't find deleteFile");
1204 return -1;
1205 }
Mike Lockwood9a2046f2010-08-03 15:30:09 -04001206 method_getObjectReferences = env->GetMethodID(clazz, "getObjectReferences", "(I)[I");
1207 if (method_getObjectReferences == NULL) {
1208 LOGE("Can't find getObjectReferences");
1209 return -1;
1210 }
1211 method_setObjectReferences = env->GetMethodID(clazz, "setObjectReferences", "(I[I)I");
1212 if (method_setObjectReferences == NULL) {
1213 LOGE("Can't find setObjectReferences");
1214 return -1;
1215 }
Mike Lockwood2837eef2010-08-31 16:25:12 -04001216 method_sessionStarted = env->GetMethodID(clazz, "sessionStarted", "()V");
1217 if (method_sessionStarted == NULL) {
1218 LOGE("Can't find sessionStarted");
1219 return -1;
1220 }
1221 method_sessionEnded = env->GetMethodID(clazz, "sessionEnded", "()V");
1222 if (method_sessionEnded == NULL) {
1223 LOGE("Can't find sessionEnded");
1224 return -1;
1225 }
1226
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001227 field_context = env->GetFieldID(clazz, "mNativeContext", "I");
1228 if (field_context == NULL) {
1229 LOGE("Can't find MtpDatabase.mNativeContext");
1230 return -1;
1231 }
1232
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -04001233 // now set up fields for MtpPropertyList class
Mike Lockwood0cd01362010-12-30 11:54:33 -05001234 clazz = env->FindClass("android/mtp/MtpPropertyList");
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -04001235 if (clazz == NULL) {
Mike Lockwood0cd01362010-12-30 11:54:33 -05001236 LOGE("Can't find android/mtp/MtpPropertyList");
Mike Lockwoode2ad6ec2010-10-14 18:03:25 -04001237 return -1;
1238 }
1239 field_mCount = env->GetFieldID(clazz, "mCount", "I");
1240 if (field_mCount == NULL) {
1241 LOGE("Can't find MtpPropertyList.mCount");
1242 return -1;
1243 }
1244 field_mResult = env->GetFieldID(clazz, "mResult", "I");
1245 if (field_mResult == NULL) {
1246 LOGE("Can't find MtpPropertyList.mResult");
1247 return -1;
1248 }
1249 field_mObjectHandles = env->GetFieldID(clazz, "mObjectHandles", "[I");
1250 if (field_mObjectHandles == NULL) {
1251 LOGE("Can't find MtpPropertyList.mObjectHandles");
1252 return -1;
1253 }
1254 field_mPropertyCodes = env->GetFieldID(clazz, "mPropertyCodes", "[I");
1255 if (field_mPropertyCodes == NULL) {
1256 LOGE("Can't find MtpPropertyList.mPropertyCodes");
1257 return -1;
1258 }
1259 field_mDataTypes = env->GetFieldID(clazz, "mDataTypes", "[I");
1260 if (field_mDataTypes == NULL) {
1261 LOGE("Can't find MtpPropertyList.mDataTypes");
1262 return -1;
1263 }
1264 field_mLongValues = env->GetFieldID(clazz, "mLongValues", "[J");
1265 if (field_mLongValues == NULL) {
1266 LOGE("Can't find MtpPropertyList.mLongValues");
1267 return -1;
1268 }
1269 field_mStringValues = env->GetFieldID(clazz, "mStringValues", "[Ljava/lang/String;");
1270 if (field_mStringValues == NULL) {
1271 LOGE("Can't find MtpPropertyList.mStringValues");
1272 return -1;
1273 }
1274
Mike Lockwood7d7fb632010-12-01 18:46:23 -05001275 if (AndroidRuntime::registerNativeMethods(env,
Mike Lockwood0cd01362010-12-30 11:54:33 -05001276 "android/mtp/MtpDatabase", gMtpDatabaseMethods, NELEM(gMtpDatabaseMethods)))
Mike Lockwood7d7fb632010-12-01 18:46:23 -05001277 return -1;
1278
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001279 return AndroidRuntime::registerNativeMethods(env,
Mike Lockwood0cd01362010-12-30 11:54:33 -05001280 "android/mtp/MtpPropertyGroup", gMtpPropertyGroupMethods, NELEM(gMtpPropertyGroupMethods));
Mike Lockwoodd21eac92010-07-03 00:44:05 -04001281}