Merge branch 'master' into windows-port
[platform/upstream/iotivity.git] / resource / csdk / connectivity / src / bt_le_adapter / android / caleutils.c
1 /******************************************************************
2  *
3  * Copyright 2014 Samsung Electronics All Rights Reserved.
4  *
5  *
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  *      http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  ******************************************************************/
20
21 #include <jni.h>
22 #include <stdio.h>
23 #include <android/log.h>
24
25 #include "caleutils.h"
26 #include "logger.h"
27 #include "cathreadpool.h"
28 #include "uarraylist.h"
29 #include "caadapterutils.h"
30
31 #define TAG PCF("OIC_CA_LE_UTILS")
32
33 jobject CALEGetUuidFromString(JNIEnv *env, const char* uuid)
34 {
35     VERIFY_NON_NULL_RET(uuid, TAG, "uuid is null", NULL);
36     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
37
38     jclass jni_cid_UUID = (*env)->FindClass(env, "java/util/UUID");
39     if (!jni_cid_UUID)
40     {
41         OIC_LOG(ERROR, TAG, "jni_cid_UUID is not available");
42         return NULL;
43     }
44
45     jmethodID jni_mid_fromString = (*env)->GetStaticMethodID(env, jni_cid_UUID, "fromString",
46                                                              "(Ljava/lang/String;)"
47                                                              "Ljava/util/UUID;");
48     if (!jni_mid_fromString)
49     {
50         OIC_LOG(ERROR, TAG, "jni_mid_fromString is not available");
51         return NULL;
52     }
53
54     jstring str_uuid = (*env)->NewStringUTF(env, uuid);
55     if (!str_uuid)
56     {
57         OIC_LOG(ERROR, TAG, "str_uuid is not available");
58         return NULL;
59     }
60
61     jobject jni_obj_uuid = (*env)->CallStaticObjectMethod(env, jni_cid_UUID, jni_mid_fromString,
62                                                           str_uuid);
63     if (!jni_obj_uuid)
64     {
65         OIC_LOG(ERROR, TAG, "Fail to get jni uuid object");
66         return NULL;
67     }
68
69     return jni_obj_uuid;
70 }
71
72 jobject CALEGetParcelUuid(JNIEnv *env, jobject uuid)
73 {
74     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
75     VERIFY_NON_NULL_RET(uuid, TAG, "uuid is null", NULL);
76
77     jclass jni_cid_ParcelUuid = (*env)->FindClass(env, "android/os/ParcelUuid");
78     if (!jni_cid_ParcelUuid)
79     {
80         OIC_LOG(ERROR, TAG, "jni_cid_ParcelUuid is not available");
81         return NULL;
82     }
83
84     jmethodID jni_mid_ParcelUuid = (*env)->GetMethodID(env, jni_cid_ParcelUuid, "<init>",
85                                                        "(Ljava/util/UUID;)V");
86     if (!jni_mid_ParcelUuid)
87     {
88         OIC_LOG(ERROR, TAG, "jni_mid_ParcelUuid is not available");
89         return NULL;
90     }
91
92     jobject jni_ParcelUuid = (*env)->NewObject(env, jni_cid_ParcelUuid, jni_mid_ParcelUuid, uuid);
93     if (!jni_ParcelUuid)
94     {
95         OIC_LOG(ERROR, TAG, "Fail to get jni ParcelUuid");
96         return NULL;
97     }
98
99     return jni_ParcelUuid;
100 }
101
102 bool CALEIsBondedDevice(JNIEnv *env, jobject bluetoothDevice)
103 {
104     VERIFY_NON_NULL_RET(env, TAG, "env is null", false);
105     VERIFY_NON_NULL_RET(bluetoothDevice, TAG, "bluetoothDevice is null", false);
106
107     jmethodID jni_mid_getBondState = CAGetJNIMethodID(env, "android/bluetooth/BluetoothDevice",
108                                                       "getBondState",
109                                                       "()I");
110     if (!jni_mid_getBondState)
111     {
112         OIC_LOG(ERROR, TAG, "jni_mid_getBondState is null");
113         return false;
114     }
115
116     jint jni_bondState = (jint)(*env)->CallIntMethod(env, bluetoothDevice, jni_mid_getBondState);
117
118     OIC_LOG_V(DEBUG, TAG, "bond state is %d", jni_bondState);
119
120     if (BOND_BONDED == jni_bondState)
121     {
122         OIC_LOG(DEBUG, TAG, "remote device is bonded");
123         return true;
124     }
125     else
126     {
127         OIC_LOG(DEBUG, TAG, "remote device is not bonded");
128         return false;
129     }
130
131     return false;
132 }
133
134 jobjectArray CALEGetBondedDevices(JNIEnv *env)
135 {
136     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
137
138     jclass jni_cid_BTAdapter = (*env)->FindClass(env, CLASSPATH_BT_ADAPTER);
139     if (!jni_cid_BTAdapter)
140     {
141         OIC_LOG(ERROR, TAG, "getBondedDevices: jni_cid_BTAdapter is null");
142         return NULL;
143     }
144
145     jmethodID jni_mid_getDefaultAdapter = (*env)->GetStaticMethodID(env, jni_cid_BTAdapter,
146                                                                     "getDefaultAdapter",
147                                                                     METHODID_OBJECTNONPARAM);
148
149     jobject jni_obj_BTAdapter = (*env)->CallStaticObjectMethod(env, jni_cid_BTAdapter,
150                                                                jni_mid_getDefaultAdapter);
151     if (!jni_obj_BTAdapter)
152     {
153         OIC_LOG(ERROR, TAG, "getBondedDevices: bluetooth adapter is null");
154         return NULL;
155     }
156
157     // Get a list of currently paired devices
158     jmethodID jni_mid_getBondedDevices = (*env)->GetMethodID(env, jni_cid_BTAdapter,
159                                                              "getBondedDevices",
160                                                              "()Ljava/util/Set;");
161     if (!jni_mid_getBondedDevices)
162     {
163         OIC_LOG(ERROR, TAG, "getBondedDevices: jni_mid_getBondedDevicesr is null");
164         return NULL;
165     }
166
167     jobject jni_obj_setPairedDevices = (*env)->CallObjectMethod(env, jni_obj_BTAdapter,
168                                                                 jni_mid_getBondedDevices);
169     if (!jni_obj_setPairedDevices)
170     {
171         OIC_LOG(ERROR, TAG, "getBondedDevices: jni_obj_setPairedDevices is null");
172         return NULL;
173     }
174
175     jmethodID jni_mid_toArray = CAGetJNIMethodID(env, "java/util/Set", "toArray",
176                                                  "()[Ljava/lang/Object;");
177     if (!jni_mid_toArray)
178     {
179         OIC_LOG(ERROR, TAG, "getBondedDevices: jni_mid_toArray is null");
180         return NULL;
181     }
182
183     jobjectArray jni_arrayPairedDevices = (jobjectArray)(
184             (*env)->CallObjectMethod(env, jni_obj_setPairedDevices, jni_mid_toArray));
185     if (!jni_arrayPairedDevices)
186     {
187         OIC_LOG(ERROR, TAG, "getBondedDevices: jni_arrayPairedDevices is null");
188         return NULL;
189     }
190
191     return jni_arrayPairedDevices;
192 }
193
194 jint CALEGetBTStateOnInfo(JNIEnv *env)
195 {
196     VERIFY_NON_NULL_RET(env, TAG, "env is null", -1);
197
198     jclass jni_cid_BTAdapter = (*env)->FindClass(env, CLASSPATH_BT_ADAPTER);
199     if (!jni_cid_BTAdapter)
200     {
201         OIC_LOG(ERROR, TAG, "getBTStateOnInfo: jni_cid_BTAdapter is null");
202         return -1;
203     }
204
205     jfieldID jni_fid_stateon = (*env)->GetStaticFieldID(env, jni_cid_BTAdapter, "STATE_ON", "I");
206     if (!jni_fid_stateon)
207     {
208         OIC_LOG(ERROR, TAG, "get_field_state is not available");
209         return -1;
210     }
211
212     jint jni_int_val = (*env)->GetStaticIntField(env, jni_cid_BTAdapter, jni_fid_stateon);
213     OIC_LOG_V(DEBUG, TAG, "bluetooth.STATE_ON state integer value : %d", jni_int_val);
214
215     return jni_int_val;
216 }
217
218 CAResult_t CALECheckPlatformVersion(JNIEnv *env, uint16_t level)
219 {
220     jint jni_int_sdk = CALEGetBuildVersion(env);
221     if (jni_int_sdk < level)
222     {
223         OIC_LOG(ERROR, TAG, "it is not supported");
224         return CA_NOT_SUPPORTED;
225     }
226
227     return CA_STATUS_OK;
228 }
229
230 jint CALEGetBuildVersion(JNIEnv *env)
231 {
232     VERIFY_NON_NULL_RET(env, TAG, "env is null", -1);
233
234     // VERSION is a nested class within android.os.Build (hence "$" rather than "/")
235     jclass jni_cls_version = (*env)->FindClass(env, "android/os/Build$VERSION");
236     if (!jni_cls_version)
237     {
238         OIC_LOG(ERROR, TAG, "jni_cls_version is null");
239         return -1;
240     }
241
242     jfieldID jni_fid_sdk = (*env)->GetStaticFieldID(env, jni_cls_version, "SDK_INT", "I");
243     if (!jni_fid_sdk)
244     {
245         OIC_LOG(ERROR, TAG, "jni_fid_sdk is null");
246         return -1;
247     }
248
249     jint jni_int_sdk = (*env)->GetStaticIntField(env, jni_cls_version, jni_fid_sdk);
250     OIC_LOG_V(DEBUG, TAG, "sdk version is %d", jni_int_sdk);
251
252     return jni_int_sdk;
253 }
254
255 jint CALEGetBuildVersionCodeForName(JNIEnv *env, const char* versionName)
256 {
257     VERIFY_NON_NULL_RET(env, TAG, "env is null", -1);
258     VERIFY_NON_NULL_RET(versionName, TAG, "versionName is null", -1);
259
260     // VERSION is a nested class within android.os.Build (hence "$" rather than "/")
261     jclass jni_cls_version = (*env)->FindClass(env, "android/os/Build$VERSION_CODES");
262     if (!jni_cls_version)
263     {
264         OIC_LOG(ERROR, TAG, "jni_cls_version is null");
265         return -1;
266     }
267
268     jfieldID jni_fid_version = (*env)->GetStaticFieldID(env, jni_cls_version, versionName, "I");
269     if (!jni_fid_version)
270     {
271         OIC_LOG(ERROR, TAG, "jni_fid_version is null");
272         return -1;
273     }
274
275     jint jni_int_version = (*env)->GetStaticIntField(env, jni_cls_version, jni_fid_version);
276     OIC_LOG_V(DEBUG, TAG, "version [%s] is %d",versionName, jni_int_version);
277
278     return jni_int_version;
279 }
280
281 jboolean CALEIsEnableBTAdapter(JNIEnv *env)
282 {
283     VERIFY_NON_NULL_RET(env, TAG, "env is null", JNI_FALSE);
284
285     jclass jni_cid_BTAdapter = (*env)->FindClass(env, CLASSPATH_BT_ADAPTER);
286     if (!jni_cid_BTAdapter)
287     {
288         OIC_LOG(ERROR, TAG, "jni_cid_BTAdapter: jni_cid_BTAdapter is null");
289         return JNI_FALSE;
290     }
291
292     jmethodID jni_mid_getDefaultAdapter = (*env)->GetStaticMethodID(env, jni_cid_BTAdapter,
293                                                                     "getDefaultAdapter",
294                                                                     METHODID_OBJECTNONPARAM);
295     if (!jni_mid_getDefaultAdapter)
296     {
297         OIC_LOG(ERROR, TAG, "jni_mid_getDefaultAdapter is null");
298         return JNI_FALSE;
299     }
300
301     jobject jni_obj_BTAdapter = (*env)->CallStaticObjectMethod(env, jni_cid_BTAdapter,
302                                                                jni_mid_getDefaultAdapter);
303     if (!jni_obj_BTAdapter)
304     {
305         OIC_LOG(ERROR, TAG, "jni_obj_BTAdapter is null");
306         return JNI_FALSE;
307     }
308
309     // isEnable()
310     jmethodID jni_mid_isEnable = (*env)->GetMethodID(env, jni_cid_BTAdapter, "isEnabled", "()Z");
311     if (!jni_mid_isEnable)
312     {
313         OIC_LOG(ERROR, TAG, "jni_mid_isEnable is null");
314         return JNI_FALSE;
315     }
316
317     jboolean jni_isEnable = (*env)->CallBooleanMethod(env, jni_obj_BTAdapter, jni_mid_isEnable);
318     OIC_LOG_V(DEBUG, TAG, "adapter state is %d", jni_isEnable);
319
320     return jni_isEnable;
321 }
322
323 jstring CALEGetAddressFromBTDevice(JNIEnv *env, jobject bluetoothDevice)
324 {
325     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
326     VERIFY_NON_NULL_RET(bluetoothDevice, TAG, "bluetoothDevice is null", NULL);
327
328     jmethodID jni_mid_getAddress = CAGetJNIMethodID(env, "android/bluetooth/BluetoothDevice",
329                                                     "getAddress",
330                                                     "()Ljava/lang/String;");
331     if (!jni_mid_getAddress)
332     {
333         OIC_LOG(ERROR, TAG, "jni_mid_getAddress is null");
334         return NULL;
335     }
336
337     jstring jni_address = (jstring)(*env)->CallObjectMethod(env, bluetoothDevice,
338                                                             jni_mid_getAddress);
339     if (!jni_address)
340     {
341         OIC_LOG(ERROR, TAG, "jni_address is null");
342         return NULL;
343     }
344
345     return jni_address;
346 }
347
348 jint CALEGetConstantsValue(JNIEnv *env, const char* classType, const char* name)
349 {
350     VERIFY_NON_NULL_RET(env, TAG, "env", -1);
351     VERIFY_NON_NULL_RET(classType, TAG, "classType", -1);
352     VERIFY_NON_NULL_RET(name, TAG, "name", -1);
353
354     jclass jni_cid = (*env)->FindClass(env, classType);
355     if (!jni_cid)
356     {
357         OIC_LOG(ERROR, TAG, "jni_cid is null");
358         return -1;
359     }
360
361     jfieldID jni_fieldID = (*env)->GetStaticFieldID(env, jni_cid,
362                                                     name, "I");
363     if (!jni_fieldID)
364     {
365         OIC_LOG(ERROR, TAG, "jni_fieldID is null");
366         return -1;
367     }
368
369     return (*env)->GetStaticIntField(env, jni_cid, jni_fieldID);
370 }
371
372 jobject CALEGetRemoteDevice(JNIEnv *env, jstring address)
373 {
374     OIC_LOG(DEBUG, TAG, "IN - CALEGetRemoteDevice");
375
376     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
377     VERIFY_NON_NULL_RET(address, TAG, "address is null", NULL);
378
379     jclass jni_cid_BTAdapter = (*env)->FindClass(env, CLASSPATH_BT_ADAPTER);
380     if (!jni_cid_BTAdapter)
381     {
382         OIC_LOG(ERROR, TAG, "jni_cid_BTAdapter is null");
383         return NULL;
384     }
385
386     // get remote bt adapter method
387     jmethodID jni_mid_getDefaultAdapter = (*env)->GetStaticMethodID(env, jni_cid_BTAdapter,
388                                                                     "getDefaultAdapter",
389                                                                     METHODID_OBJECTNONPARAM);
390     if (!jni_mid_getDefaultAdapter)
391     {
392         OIC_LOG(ERROR, TAG, "jni_mid_getDefaultAdapter is null");
393         return NULL;
394     }
395
396     // gat bt adapter object
397     jobject jni_obj_BTAdapter = (*env)->CallStaticObjectMethod(env, jni_cid_BTAdapter,
398                                                                jni_mid_getDefaultAdapter);
399     if (!jni_obj_BTAdapter)
400     {
401         OIC_LOG(ERROR, TAG, "jni_obj_BTAdapter is null");
402         return NULL;
403     }
404
405     jmethodID jni_mid_getRemoteDevice = (*env)->GetMethodID(env, jni_cid_BTAdapter,
406                                                             "getRemoteDevice",
407                                                             METHODID_BT_REMOTE_DEVICE);
408     if (!jni_mid_getRemoteDevice)
409     {
410         OIC_LOG(ERROR, TAG, "jni_mid_getRemoteDevice is null");
411         return NULL;
412     }
413
414     jobject jni_obj_device = (*env)->CallObjectMethod(env, jni_obj_BTAdapter,
415                                                       jni_mid_getRemoteDevice,
416                                                       address);
417     if (!jni_obj_device)
418     {
419         OIC_LOG(ERROR, TAG, "jni_obj_device is null");
420         return NULL;
421     }
422
423     OIC_LOG(DEBUG, TAG, "OUT - CALEGetRemoteDevice");
424     return jni_obj_device;
425 }
426
427 jstring CALEGetAddressFromGatt(JNIEnv *env, jobject gatt)
428 {
429     OIC_LOG(DEBUG, TAG, "IN - CALEGetAddressFromGatt");
430
431     VERIFY_NON_NULL_RET(env, TAG, "env is null", NULL);
432     VERIFY_NON_NULL_RET(gatt, TAG, "gatt is null", NULL);
433
434     jmethodID jni_mid_getDevice = CAGetJNIMethodID(env, CLASSPATH_BT_GATT, "getDevice",
435                                                    METHODID_BT_DEVICE);
436     if (!jni_mid_getDevice)
437     {
438         OIC_LOG(ERROR, TAG, "jni_mid_getDevice is null");
439         return NULL;
440     }
441
442     jobject jni_obj_device = (*env)->CallObjectMethod(env, gatt, jni_mid_getDevice);
443     if (!jni_obj_device)
444     {
445         OIC_LOG(ERROR, TAG, "jni_obj_device is null");
446         return NULL;
447     }
448
449     jstring jni_address = CALEGetAddressFromBTDevice(env, jni_obj_device);
450     if (!jni_address)
451     {
452         OIC_LOG(ERROR, TAG, "jni_address is null");
453         return NULL;
454     }
455
456     OIC_LOG(DEBUG, TAG, "OUT - CALEGetAddressFromGatt");
457     return jni_address;
458 }