Fix coverity issue
[platform/core/api/media-content.git] / src / media_content.c
1 /*
2 * Copyright (c) 2011 Samsung Electronics Co., Ltd All Rights Reserved
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
18 #include <media_info_private.h>
19 #include <media_util_private.h>
20
21 static sqlite3 *db_handle = NULL;
22 static int ref_count = 0;
23 static GMutex db_mutex;
24 static uid_t content_g_uid = 0;
25
26 sqlite3 * _content_get_db_handle(void)
27 {
28         return db_handle;
29 }
30
31 uid_t _content_get_uid(void)
32 {
33         if (content_g_uid == 0)
34                 return tzplatform_getuid(TZ_USER_NAME);
35         else
36                 return content_g_uid;
37 }
38
39 int _content_query_prepare(char *select_query, char *condition_query, char *option_query, sqlite3_stmt **stmt)
40 {
41         int ret = MEDIA_CONTENT_ERROR_NONE;
42         char *query = NULL;
43
44         media_content_retvm_if(!STRING_VALID(select_query), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "invalid select_query");
45
46         if (!STRING_VALID(condition_query))
47                 condition_query = (char *)" ";
48
49         if (!STRING_VALID(option_query))
50                 option_query = (char *)" ";
51
52         query = sqlite3_mprintf("%s %s %s", select_query, condition_query, option_query);
53         ret = _content_get_result(query, stmt);
54         SQLITE3_SAFE_FREE(query);
55
56         return ret;
57 }
58
59 int _content_get_result(char *query, sqlite3_stmt **stmt)
60 {
61         int err = MEDIA_CONTENT_ERROR_NONE;
62
63         media_content_retvm_if(_content_get_db_handle() == NULL, MEDIA_CONTENT_ERROR_DB_FAILED, "database is not connected");
64         media_content_retvm_if(!STRING_VALID(query), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "invalid query");
65
66         media_content_sec_debug("Query[%s]", query);
67
68         err = sqlite3_prepare_v2(_content_get_db_handle(), query, strlen(query), stmt, NULL);
69         if (err != SQLITE_OK) {
70                 media_content_error("Failed to sqlite3_prepare_v2[%s]", sqlite3_errmsg(_content_get_db_handle()));
71                 if (err == SQLITE_BUSY)
72                         return MEDIA_CONTENT_ERROR_DB_BUSY;
73                 else if (err == SQLITE_PERM)
74                         return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
75                 else
76                         return MEDIA_CONTENT_ERROR_DB_FAILED;
77         }
78
79         return MEDIA_CONTENT_ERROR_NONE;
80 }
81
82 #ifdef _USE_SENIOR_MODE
83 int _content_query_prepare_by_union_select(sqlite3_stmt **stmt, char *select_query1, char *condition_query1, char *option_query1, char *select_query2, char *condition_query2, char *option_query2)
84 {
85         int len = 0;
86         int err = MEDIA_CONTENT_ERROR_NONE;
87         char query[MAX_QUERY_SIZE] = {0, };
88         memset(query, '\0', sizeof(query));
89
90         media_content_retvm_if(_content_get_db_handle() == NULL, MEDIA_CONTENT_ERROR_DB_FAILED, "database is not connected");
91         media_content_retvm_if(!STRING_VALID(select_query1), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "invalid select_query1");
92         media_content_retvm_if(!STRING_VALID(select_query2), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "invalid select_query2");
93
94         if (!STRING_VALID(condition_query1))
95                 condition_query1 = (char *)" ";
96
97         if (!STRING_VALID(option_query1))
98                 option_query1 = (char *)" ";
99
100         if (!STRING_VALID(condition_query2))
101                 condition_query2 = (char *)" ";
102
103         if (!STRING_VALID(option_query2))
104                 option_query2 = (char *)" ";
105
106         len = snprintf(query, sizeof(query), "SELECT * FROM (%s %s %s) as table1 UNION ALL SELECT * FROM (%s %s %s) as table2",
107                         select_query1, condition_query1, option_query1, select_query2, condition_query2, option_query2);
108         if (len > 0 && len < sizeof(query)) {
109                 query[len] = '\0';
110         } else if (len >= sizeof(query)) {
111                 query[MAX_QUERY_SIZE -1] = '\0';
112         } else {
113                 media_content_error("snprintf failed");
114                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
115         }
116
117         media_content_sec_debug("Query : [%s]", query);
118
119         err = sqlite3_prepare_v2(_content_get_db_handle(), query, strlen(query), stmt, NULL);
120         if (err != SQLITE_OK) {
121                 media_content_error("DB_FAILED(0x%08x) fail to sqlite3_prepare(), %s", MEDIA_CONTENT_ERROR_DB_FAILED, sqlite3_errmsg(_content_get_db_handle()));
122
123                 if (err == SQLITE_BUSY) {
124                         media_content_error(" BUSY ERROR");
125                         return MEDIA_CONTENT_ERROR_DB_BUSY;
126                 } else if (err == SQLITE_PERM) {
127                         media_content_error("PERMISSION EROR");
128                         return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
129                 } else {
130                         media_content_error("OTHER ERROR");
131                         return MEDIA_CONTENT_ERROR_DB_FAILED;
132                 }
133         }
134
135         return MEDIA_CONTENT_ERROR_NONE;
136 }
137 #endif
138
139 int _content_error_capi(int content_error)
140 {
141         if (content_error != MEDIA_CONTENT_ERROR_NONE)
142                 media_content_error("MS Error[%d]", content_error);
143
144         /*Error None*/
145         if (content_error == MS_MEDIA_ERR_NONE)
146                 return MEDIA_CONTENT_ERROR_NONE;
147
148         /* Internal operation error*/
149         else if (content_error == MS_MEDIA_ERR_INVALID_PARAMETER)
150                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
151
152         else if (content_error == MS_MEDIA_ERR_OUT_OF_MEMORY)
153                 return MEDIA_CONTENT_ERROR_OUT_OF_MEMORY;
154
155         /* DB operation error*/
156         else if (content_error == MS_MEDIA_ERR_DB_BUSY_FAIL)
157                 return MEDIA_CONTENT_ERROR_DB_BUSY;
158
159         else if ((content_error <= MS_MEDIA_ERR_DB_CONSTRAINT_FAIL) && (content_error >= MS_MEDIA_ERR_DB_INTERNAL))
160                 return MEDIA_CONTENT_ERROR_DB_FAILED;
161
162         /* IPC operation error*/
163         else if (content_error == MS_MEDIA_ERR_IPC)
164                 return MEDIA_CONTENT_ERROR_NETWORK;
165
166         /* MEDIA SERVER error*/
167         else if (content_error == MS_MEDIA_ERR_PERMISSION_DENIED)
168                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
169
170         /* Thumbnail error*/
171         else if ((content_error == MS_MEDIA_ERR_THUMB_TOO_BIG) || (content_error == MS_MEDIA_ERR_THUMB_UNSUPPORTED))
172                         return MEDIA_CONTENT_ERROR_UNSUPPORTED_CONTENT;
173
174         /*ETC*/
175         return MEDIA_CONTENT_ERROR_INVALID_OPERATION;
176 }
177
178 int _content_query_sql(char *query_str)
179 {
180         int ret = MEDIA_CONTENT_ERROR_NONE;
181
182         /*DB will be updated by Media Server.*/
183         ret = media_db_request_update_db(query_str, _content_get_uid());
184
185         return _content_error_capi(ret);
186 }
187
188 int media_content_connect(void)
189 {
190         int ret = MEDIA_CONTENT_ERROR_NONE;
191
192         g_mutex_lock(&db_mutex);
193         media_content_info("ref count : %d", ref_count);
194
195         if (ref_count == 0) {
196                 if (db_handle == NULL) {
197                         ret = media_db_connect(&db_handle, _content_get_uid(), false);
198                         ret = _content_error_capi(ret);
199                         if (ret == MEDIA_CONTENT_ERROR_NONE)
200                                 ref_count++;
201                 } else {
202                         media_content_error("Wrong DB Connection status");
203                         ret = MEDIA_CONTENT_ERROR_DB_FAILED;
204                 }
205         } else {
206                 if (db_handle != NULL) {
207                         ref_count++;
208                 } else {
209                         media_content_error("Wrong DB Handle status");
210                         ret = MEDIA_CONTENT_ERROR_DB_FAILED;
211                 }
212         }
213
214         media_content_info("ref count changed to: %d", ref_count);
215         g_mutex_unlock(&db_mutex);
216
217         return ret;
218 }
219
220 int media_content_connect_with_uid(uid_t uid)
221 {
222         media_content_sec_debug("media_content_connect_with_uid [%d]", uid);
223         content_g_uid = uid;
224
225         return media_content_connect();
226 }
227
228 int media_content_disconnect(void)
229 {
230         g_mutex_lock(&db_mutex);
231         media_content_debug("ref count : %d", ref_count);
232
233         if (db_handle && ref_count > 0) {
234                 if (--ref_count == 0) {
235                         media_db_disconnect(db_handle);
236                         db_handle = NULL;
237                 }
238         } else {
239                 media_content_error("Database is not connected");
240                 g_mutex_unlock(&db_mutex);
241                 return MEDIA_CONTENT_ERROR_DB_FAILED;
242         }
243
244         g_mutex_unlock(&db_mutex);
245
246         media_content_info("ref count changed to: %d", ref_count);
247
248         return MEDIA_CONTENT_ERROR_NONE;
249 }
250
251 int media_content_scan_file(const char *path)
252 {
253         int ret = MEDIA_CONTENT_ERROR_NONE;
254         bool ignore_file = FALSE;
255         bool ignore_dir = FALSE;
256         char *folder_path = NULL;
257         int check_file = MEDIA_CONTENT_ERROR_NONE;
258         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0,};
259         char repl_path[MAX_PATH_LEN] = {0,};
260
261         media_content_retvm_if(!STRING_VALID(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "invalid path");
262
263         media_content_sec_debug("Path : %s", path);
264
265         memset(repl_path, 0, sizeof(repl_path));
266         ret = _media_content_replace_path(path, repl_path);
267         media_content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
268
269         ret = _media_util_check_ignore_file(repl_path, &ignore_file);
270         media_content_retvm_if(ignore_file == TRUE, MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
271
272         memset(storage_id, 0x00, sizeof(storage_id));
273         ret = media_svc_get_storage_id(_content_get_db_handle(), repl_path, storage_id, _content_get_uid());
274         if (ret != MS_MEDIA_ERR_NONE) {
275                 media_content_error("media_svc_get_storage_id failed : %d", ret);
276                 return _content_error_capi(ret);
277         }
278
279         check_file = _media_util_check_file_exist(repl_path);
280         if (check_file == MEDIA_CONTENT_ERROR_NONE) {
281                 /* This means this path has to be inserted or refreshed */
282                 folder_path = g_path_get_dirname(repl_path);
283                 ret = _media_util_check_ignore_dir(folder_path, &ignore_dir);
284                 SAFE_FREE(folder_path);
285
286                 media_content_retvm_if(ignore_dir == TRUE, MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
287                 /* check feature */
288                 media_content_retvm_if(!_media_util_check_support_media_type(repl_path), MEDIA_CONTENT_ERROR_NOT_SUPPORTED, "Unsupported media type");
289
290                 ms_user_storage_type_e storage_type;
291
292                 ret = ms_user_get_storage_type(_content_get_uid(), repl_path, &storage_type);
293                 if (ret != MS_MEDIA_ERR_NONE) {
294                         media_content_sec_error("ms_user_get_storage_type failed : %d (%s)", ret, repl_path);
295                         return _content_error_capi(ret);
296                 }
297                 ret = media_svc_check_item_exist_by_path(_content_get_db_handle(), storage_id, repl_path);
298                 if (ret == MS_MEDIA_ERR_NONE) {
299                         /* Refresh */
300                         ret = media_svc_refresh_item(_content_get_db_handle(), false, storage_id, storage_type, repl_path, _content_get_uid());
301                         if (ret != MS_MEDIA_ERR_NONE) {
302                                 media_content_error("media_svc_refresh_item failed : %d", ret);
303                                 return _content_error_capi(ret);
304                         }
305
306                 } else if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
307                         /* Insert */
308                         ret = media_svc_insert_item_immediately(_content_get_db_handle(), storage_id, storage_type, repl_path, _content_get_uid());
309                         if (ret != MS_MEDIA_ERR_NONE) {
310                                 if (ret == MS_MEDIA_ERR_DB_CONSTRAINT_FAIL) {
311                                         media_content_sec_error("This item is already inserted. This may be normal operation because other process already did this (%s)", repl_path);
312                                         ret = MEDIA_CONTENT_ERROR_NONE;
313                                 } else {
314                                         media_content_sec_error("media_svc_insert_item_immediately failed : %d (%s)", ret, repl_path);
315                                 }
316
317                                 return _content_error_capi(ret);
318                         }
319                 } else {
320                         media_content_error("media_svc_check_item_exist_by_path failed : %d", ret);
321                         return _content_error_capi(ret);
322                 }
323         } else if (check_file == MEDIA_CONTENT_ERROR_PERMISSION_DENIED) {
324                 media_content_error("You have no permission for this file %d", ret);
325                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
326         } else {
327                 /* This means this path has to be deleted */
328                 media_content_debug("This path doesn't exists in file system... So now start to delete it from DB");
329                 ret = media_svc_delete_item_by_path(_content_get_db_handle(), storage_id, repl_path, _content_get_uid());
330                 if (ret != MS_MEDIA_ERR_NONE) {
331                         if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
332                                 media_content_error("Does not exist in media DB also... So, this is an invalid parameter");
333                                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
334                         }
335
336                         media_content_error("media_svc_delete_item_by_path failed : %d", ret);
337                         return _content_error_capi(ret);
338                 }
339         }
340
341         return _content_error_capi(ret);
342 }
343
344 void _media_content_scan_cb(media_request_result_s* result, void *user_data)
345 {
346         int err = -1;
347         media_content_scan_cb_data *cb_data = user_data;
348
349         err = _content_error_capi(result->result);
350 #ifdef _USE_TVPD_MODE
351         if (result->request_type != MEDIA_REQUEST_SCAN_COMPLETE &&
352                 result->request_type != MEDIA_REQUEST_SCAN_PARTIAL &&
353                 result->request_type != MEDIA_RECURSIVE_START) {
354                 if (cb_data && cb_data->callback) {
355                         media_content_debug("begin:User callback is being called now, result=%d", err);
356                         cb_data->callback(err, cb_data->user_data);
357                         media_content_debug("end:User callback is being called now, result=%d", err);
358                 }
359
360                 SAFE_FREE(cb_data);
361         }
362 #else
363         if (cb_data && cb_data->callback) {
364                 media_content_debug("User callback is being called now");
365                 cb_data->callback(err, cb_data->user_data);
366         }
367
368         SAFE_FREE(cb_data);
369 #endif
370
371         return;
372 }
373
374 #ifdef _USE_TVPD_MODE
375 void _media_content_scan_cb_v2(media_request_result_s* result, void *user_data)
376 {
377         int err = -1;
378         media_content_scan_cb_data_v2 *cb_data = user_data;
379         media_content_complete_phase_e complete_phase = -1;
380         if (!cb_data)
381                 media_content_debug("cb_data is NULL");
382         err = _content_error_capi(result->result);
383         media_content_debug("result is %d", err);
384
385         if (result->request_type == MEDIA_REQUEST_SCAN_PARTIAL)
386                 complete_phase = MEDIA_CONTENT_SCAN_PARTIAL_COMPLETE;
387         else if (result->request_type == MEDIA_REQUEST_SCAN_COMPLETE)
388                 complete_phase = MEDIA_CONTENT_SCAN_COMPLETE;
389         else if (result->request_type == MEDIA_REQUEST_EXTRACT_COMPLETE)
390                 complete_phase = MEDIA_CONTENT_EXTRACT_COMPLETE;
391         else if (result->request_type == MEDIA_RECURSIVE_START)
392                 complete_phase = MEDIA_CONTENT_RECURSIVE_START;
393
394         if (cb_data && cb_data->callback)
395                 cb_data->callback(err, complete_phase, cb_data->user_data);
396         else
397                 media_content_debug("run error");
398
399         if ((result->request_type != MEDIA_REQUEST_SCAN_COMPLETE) &&
400         (result->request_type != MEDIA_REQUEST_SCAN_PARTIAL) &&
401         (result->request_type != MEDIA_RECURSIVE_START))
402                 SAFE_FREE(cb_data);
403
404         return;
405 }
406 #endif
407
408 int media_content_scan_folder(const char *path, bool is_recursive, media_scan_completed_cb callback, void *user_data)
409 {
410         int ret = MEDIA_CONTENT_ERROR_NONE;
411         bool ignore_dir = FALSE;
412         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
413         char repl_path[MAX_PATH_LEN] = {0, };
414         ms_user_storage_type_e storage_type = MS_USER_STORAGE_INTERNAL;
415
416         media_content_retvm_if(!STRING_VALID(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
417         memset(repl_path, 0, sizeof(repl_path));
418         ret = _media_content_replace_path(path, repl_path);
419         media_content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
420
421         memset(storage_id, 0x00, sizeof(storage_id));
422
423         ret = _media_content_check_dir(repl_path);
424         media_content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
425
426         if (ret == MEDIA_CONTENT_ERROR_NONE) {
427                 /* If directory exist check that's ignore directory or not*/
428                 ret = _media_util_check_ignore_dir(repl_path, &ignore_dir);
429                 media_content_retvm_if((ignore_dir == TRUE || ret != MEDIA_CONTENT_ERROR_NONE), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
430         } else {
431                 /* This means this folder has to be deleted */
432                 /* Or, it is real invalid path.. check storage type */
433                 ret = ms_user_get_storage_type(_content_get_uid(), repl_path, &storage_type);
434                 if (ret != MS_MEDIA_ERR_NONE) {
435                         media_content_sec_error("ms_user_get_storage_type failed : %d (%s)", ret, repl_path);
436                         return _content_error_capi(ret);
437                 }
438
439                 media_content_debug("This path doesn't exists in file system... So will be deleted it from DB");
440         }
441
442         ret = media_svc_get_storage_id(_content_get_db_handle(), repl_path, storage_id, _content_get_uid());
443         if (ret != MS_MEDIA_ERR_NONE) {
444                 media_content_error("media_svc_get_storage_id failed : %d", ret);
445                 return _content_error_capi(ret);
446         }
447
448         media_content_scan_cb_data *cb_data = NULL;
449         cb_data = (media_content_scan_cb_data *)malloc(sizeof(media_content_scan_cb_data));
450         media_content_retvm_if(cb_data == NULL, MEDIA_CONTENT_ERROR_OUT_OF_MEMORY, "OUT_OF_MEMORY");
451
452         cb_data->callback = callback;
453         cb_data->user_data = user_data;
454
455         ret = media_directory_scanning_async(repl_path, storage_id, is_recursive, _media_content_scan_cb, cb_data, _content_get_uid());
456         if (ret != MS_MEDIA_ERR_NONE) {
457                 media_content_error("media_directory_scanning_async failed : %d", ret);
458                 SAFE_FREE(cb_data);
459         }
460
461         return _content_error_capi(ret);
462 }
463
464 #ifdef _USE_TVPD_MODE
465 int media_content_scan_folder_v2(const char *path, bool is_recursive, media_scan_completed_cb_v2 callback, void *user_data)
466 {
467         int ret = MEDIA_CONTENT_ERROR_NONE;
468         bool ignore_dir = FALSE;
469         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
470
471         media_content_retvm_if(!STRING_VALID(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
472         memset(storage_id, 0x00, sizeof(storage_id));
473
474         ret = _media_util_check_ignore_dir(path, &ignore_dir);
475         media_content_retvm_if(ignore_dir, MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
476
477         ret = _media_content_check_dir(path);
478         media_content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
479         media_content_retvm_if(ret == MEDIA_CONTENT_ERROR_INVALID_PARAMETER, ret, "invalid path[%s]", path);
480
481         media_content_scan_cb_data_v2* cb_data = NULL;
482         cb_data = (media_content_scan_cb_data_v2*)malloc(sizeof(media_content_scan_cb_data_v2));
483         media_content_retvm_if(cb_data == NULL, MEDIA_CONTENT_ERROR_OUT_OF_MEMORY, "OUT_OF_MEMORY");
484
485         cb_data->callback = callback;
486         cb_data->user_data = user_data;
487
488         ret = media_svc_get_storage_id(_content_get_db_handle(), path, storage_id, _content_get_uid());
489         /*FIX ME. need to check ret value?*/
490
491         ret = media_directory_scanning_async(path, storage_id, is_recursive, _media_content_scan_cb_v2, cb_data, _content_get_uid());
492         if (ret != MS_MEDIA_ERR_NONE)
493                 media_content_error("media_directory_scanning_async failed : %d", ret);
494
495         return _content_error_capi(ret);
496 }
497 #endif
498
499 int media_content_cancel_scan_folder(const char *path)
500 {
501         int ret = MEDIA_CONTENT_ERROR_NONE;
502         char repl_path[MAX_PATH_LEN] = {0, };
503
504         media_content_retvm_if(!STRING_VALID(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
505
506         memset(repl_path, 0, sizeof(repl_path));
507         ret = _media_content_replace_path(path, repl_path);
508         media_content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
509
510         ret = media_directory_scanning_cancel(repl_path, _content_get_uid());
511         if (ret != MS_MEDIA_ERR_NONE)
512                 media_content_error("media_directory_scanning_async failed : %d", ret);
513
514         return _content_error_capi(ret);
515 }
516
517 void _media_content_db_update_noti_cb(
518                                                         int pid,
519                                                         media_item_type_e item,
520                                                         media_item_update_type_e update_type,
521                                                         char* path,
522                                                         char* uuid,
523                                                         media_type_e content_type,
524                                                         char *mime_type,
525                                                         void *user_data)
526 {
527         media_noti_cb_s *_noti_info = (media_noti_cb_s *)user_data;
528
529         if (_noti_info != NULL && _noti_info->update_noti_cb)
530                 _noti_info->update_noti_cb(
531                                         MEDIA_CONTENT_ERROR_NONE,
532                                         pid,
533                                         item,
534                                         update_type,
535                                         content_type,
536                                         uuid,
537                                         path,
538                                         mime_type,
539                                         _noti_info->user_data);
540 }
541
542 int media_content_add_db_updated_cb(media_content_db_update_cb callback, void *user_data, media_content_noti_h *noti_handle)
543 {
544         int ret = MEDIA_CONTENT_ERROR_NONE;
545         media_noti_cb_s *noti_info = NULL;
546
547         media_content_retvm_if(noti_handle == NULL, MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "handle is NULL");
548         media_content_retvm_if(callback == NULL, MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "callback is NULL");
549
550         noti_info = (media_noti_cb_s *)calloc(1, sizeof(media_noti_cb_s));
551         media_content_retvm_if(noti_info == NULL, MEDIA_CONTENT_ERROR_OUT_OF_MEMORY, "Failed to create noti info");
552
553         noti_info->update_noti_cb = callback;
554         noti_info->user_data = user_data;
555
556         ret = media_db_update_subscribe_internal((MediaNotiHandle*)noti_handle, _media_content_db_update_noti_cb, (void *)noti_info);
557         if (ret != MS_MEDIA_ERR_NONE)
558                 SAFE_FREE(noti_info);
559
560         return _content_error_capi(ret);
561 }
562
563 #ifdef TIZEN_FEATURE_COMPATIBILITY
564 int media_content_set_db_updated_cb(media_content_db_update_cb callback, void *user_data)
565 {
566         media_content_warn("DEPRECATION WARNING: media_content_set_db_updated_cb() is removed from 5.0.");
567
568         return MEDIA_CONTENT_ERROR_NONE;
569 }
570 #endif
571
572 void __media_content_clear_user_data(void *user_data)
573 {
574         media_noti_cb_s *noti_info = user_data;
575
576         SAFE_FREE(noti_info);
577
578         return;
579 }
580
581 int media_content_remove_db_updated_cb(media_content_noti_h noti_handle)
582 {
583         int ret = MEDIA_CONTENT_ERROR_NONE;
584
585         ret = media_db_update_unsubscribe_internal((MediaNotiHandle)noti_handle, __media_content_clear_user_data);
586
587         return _content_error_capi(ret);
588 }
589 #ifdef _USE_TVPD_MODE
590 GMutex* _content_get_db_mutex(void)
591 {
592         return &db_mutex;
593 }
594 #endif