Only user session api calls can get internal storage root path
[platform/core/system/libstorage.git] / src / storage.c
index b98fded..d79a00d 100644 (file)
 #include <stdlib.h>
 #include <string.h>
 #include <errno.h>
-#include <system_settings.h>
+#include <vconf.h>
+#include <tzplatform_config.h>
 
 #include "common.h"
 #include "list.h"
 #include "log.h"
 #include "storage-external.h"
 
-const char *dir_path[STORAGE_DIRECTORY_MAX] = {
-       [STORAGE_DIRECTORY_IMAGES] = "Images",
-       [STORAGE_DIRECTORY_SOUNDS] = "Sounds",
-       [STORAGE_DIRECTORY_VIDEOS] = "Videos",
-       [STORAGE_DIRECTORY_CAMERA] = "Camera",
-       [STORAGE_DIRECTORY_DOWNLOADS] = "Downloads",
-       [STORAGE_DIRECTORY_MUSIC] = "Music",
-       [STORAGE_DIRECTORY_DOCUMENTS] = "Documents",
-       [STORAGE_DIRECTORY_OTHERS] = "Others",
-       [STORAGE_DIRECTORY_SYSTEM_RINGTONES] = "",
+#define BLOCK_CONF_FILE         "/etc/storaged/block.conf"
+#define USER_UID_START         5000
+
+const char *dir_path[STORAGE_DIRECTORY_MAX];
+
+const int tz_id[STORAGE_DIRECTORY_MAX] = {
+       [STORAGE_DIRECTORY_IMAGES] = TZ_USER_IMAGES,
+       [STORAGE_DIRECTORY_SOUNDS] = TZ_USER_SOUNDS,
+       [STORAGE_DIRECTORY_VIDEOS] = TZ_USER_VIDEOS,
+       [STORAGE_DIRECTORY_CAMERA] = TZ_USER_CAMERA,
+       [STORAGE_DIRECTORY_DOWNLOADS] = TZ_USER_DOWNLOADS,
+       [STORAGE_DIRECTORY_MUSIC] = TZ_USER_MUSIC,
+       [STORAGE_DIRECTORY_DOCUMENTS] = TZ_USER_DOCUMENTS,
+       [STORAGE_DIRECTORY_OTHERS] = TZ_USER_OTHERS,
 };
 
 static dd_list *st_int_head; /* Internal storage list */
 
+static dd_list *compat_cb_list;
+struct compat_cb_info {
+       storage_state_changed_cb user_cb;
+       void *user_data;
+};
+
 void add_device(const struct storage_ops *st)
 {
        DD_LIST_APPEND(st_int_head, st);
@@ -50,28 +61,54 @@ void remove_device(const struct storage_ops *st)
        DD_LIST_REMOVE(st_int_head, st);
 }
 
+int storage_ext_is_supported(void)
+{
+       static int support = -1;
+
+       if (support >= 0)
+               return support;
+
+       if (access(BLOCK_CONF_FILE, R_OK) == 0)
+               support = 1;
+       else
+               support = 0;
+
+       return support;
+}
+
 API int storage_foreach_device_supported(storage_device_supported_cb callback, void *user_data)
 {
        const struct storage_ops *st;
        dd_list *elem;
        int ret;
+       bool user = true;
 
        if (!callback) {
                _E("Invalid parameter");
                return STORAGE_ERROR_INVALID_PARAMETER;
        }
 
+       if (getuid() < USER_UID_START)
+               user = false;
+
        DD_LIST_FOREACH(st_int_head, elem, st) {
-               ret = callback(st->storage_id, st->type, st->get_state(),
-                               st->root(), user_data);
-               /* if the return value is false, will be stop to iterate */
-               if (!ret)
-                       break;
+               if (user) {
+                       ret = callback(st->storage_id, st->type, st->get_state(),
+                                       st->root(), user_data);
+                       /* if the return value is false, will be stop to iterate */
+                       if (!ret)
+                               break;
+               }
+       }
+
+       if (!storage_ext_is_supported()) {
+               _D("Block module is not enabled");
+               return STORAGE_ERROR_NONE;
        }
 
        ret = storage_ext_foreach_device_list(callback, user_data);
        if (ret < 0) {
-               _E("Failed to iterate external devices (%d)", ret);
+               _E("Failed to iterate external devices (%d)", ret); //LCOV_EXCL_LINE
                return STORAGE_ERROR_OPERATION_FAILED;
        }
 
@@ -84,6 +121,8 @@ API int storage_get_root_directory(int storage_id, char **path)
        dd_list *elem;
        char root[PATH_MAX];
        int ret;
+       bool extendedint;
+       bool user = true;
 
        if (storage_id < 0)
                return STORAGE_ERROR_NOT_SUPPORTED;
@@ -93,28 +132,39 @@ API int storage_get_root_directory(int storage_id, char **path)
                return STORAGE_ERROR_INVALID_PARAMETER;
        }
 
+       if (getuid() < USER_UID_START)
+               user = false;
+
        /* internal storage */
        DD_LIST_FOREACH(st_int_head, elem, st) {
                if (st->storage_id != storage_id)
                        continue;
+               if (!user) {
+                       _E("Only Tizen applications and user session daemons can use \
+                               storage_get_root_directory(id , ...)");
+                       return STORAGE_ERROR_INVALID_PARAMETER;
+               }
+
                *path = strdup(st->root());
                if (!*path) {
+//LCOV_EXCL_START System Error
                        _E("Failed to copy the root string : %d", errno);
                        return STORAGE_ERROR_OUT_OF_MEMORY;
+//LCOV_EXCL_STOP
                }
                return STORAGE_ERROR_NONE;
        }
 
        /* external storage */
-       ret = storage_ext_get_root(storage_id, root, sizeof(root));
+       ret = storage_ext_get_root(storage_id, root, sizeof(root), &extendedint);
        if (ret < 0) {
-               _E("Failed to get root path of external storage(%d, %d", storage_id, ret);
+               _E("Failed to get root path of external storage(%d, %d", storage_id, ret); //LCOV_EXCL_LINE
                return STORAGE_ERROR_INVALID_PARAMETER;
        }
 
        *path = strdup(root);
        if (!*path) {
-               _E("Failed to copy the root string : %d", errno);
+               _E("Failed to copy the root string : %d", errno); //LCOV_EXCL_LINE System Error
                return STORAGE_ERROR_OUT_OF_MEMORY;
        }
 
@@ -130,6 +180,8 @@ API int storage_get_directory(int storage_id, storage_directory_e type, char **p
        int ret;
        dd_list *elem;
        bool found;
+       bool extendedint;
+       bool user = true;
 
        if (storage_id < 0)
                return STORAGE_ERROR_NOT_SUPPORTED;
@@ -153,14 +205,21 @@ API int storage_get_directory(int storage_id, storage_directory_e type, char **p
                break;
        }
 
+       if (getuid() < USER_UID_START)
+               user = false;
+
        if (found && st) {
+               if (!user) {
+                       _E("Only Tizen applications and user session daemons can use \
+                               storage_get_directory(id, ...)");
+                       return STORAGE_ERROR_INVALID_PARAMETER;
+               }
+
                snprintf(root, sizeof(root), "%s", st->root());
                if (type == STORAGE_DIRECTORY_SYSTEM_RINGTONES) {
-                       ret = system_settings_get_value_string(SYSTEM_SETTINGS_KEY_INCOMING_CALL_RINGTONE, &temp2);
-                       if (ret < 0) {
-                               _E("Failed to get ringtone path : %d", ret);
+                       temp2 = vconf_get_str(VCONFKEY_SETAPPL_CALL_RINGTONE_PATH_STR);
+                       if (temp2 == NULL)
                                return STORAGE_ERROR_OPERATION_FAILED;
-                       }
                        end = strrchr(temp2, '/');
                        if (end)
                                *end = '\0';
@@ -174,22 +233,25 @@ API int storage_get_directory(int storage_id, storage_directory_e type, char **p
 
        /* external storage */
        if (type == STORAGE_DIRECTORY_SYSTEM_RINGTONES) {
-               _E("Not support directory : id(%d) type(%d)", storage_id, type);
+               _E("Not support directory : id(%d) type(%d)", storage_id, type); //LCOV_EXCL_LINE
                return STORAGE_ERROR_NOT_SUPPORTED;
        }
 
-       ret = storage_ext_get_root(storage_id, root, sizeof(root));
+       ret = storage_ext_get_root(storage_id, root, sizeof(root), &extendedint);
        if (ret < 0) {
-               _E("Failed to get root dir for external storage(id:%d, ret:%d)", storage_id, ret);
-               return STORAGE_ERROR_OPERATION_FAILED;
+               _E("Failed to get root dir for external storage(id:%d, ret:%d)", storage_id, ret); //LCOV_EXCL_LINE
+               return STORAGE_ERROR_NOT_SUPPORTED;
        }
+       /* The operation is not decided */
+       if (extendedint)
+               return STORAGE_ERROR_NOT_SUPPORTED;
 
        snprintf(temp, sizeof(temp), "%s/%s", root, dir_path[type]);
 
 out:
        *path = strdup(temp);
        if (!*path) {
-               _E("Failed to copy the directory(%d) string : %d", type, errno);
+               _E("Failed to copy the directory(%d) string : %d", type, errno); //LCOV_EXCL_LINE
                return STORAGE_ERROR_OUT_OF_MEMORY;
        }
 
@@ -200,6 +262,9 @@ API int storage_get_type(int storage_id, storage_type_e *type)
 {
        const struct storage_ops *st;
        dd_list *elem;
+       char root[PATH_MAX];
+       int ret;
+       bool extendedint;
 
        if (storage_id < 0)
                return STORAGE_ERROR_NOT_SUPPORTED;
@@ -218,7 +283,15 @@ API int storage_get_type(int storage_id, storage_type_e *type)
        }
 
        /* external storage */
-       *type = STORAGE_TYPE_EXTERNAL;
+       ret = storage_ext_get_root(storage_id, root, sizeof(root), &extendedint);
+       if (ret < 0) {
+               _E("Failed to get type of external storage");
+               return STORAGE_ERROR_NOT_SUPPORTED;
+       }
+       if (extendedint)
+               *type = STORAGE_TYPE_EXTENDED_INTERNAL;
+       else
+               *type = STORAGE_TYPE_EXTERNAL;
 
        return STORAGE_ERROR_NONE;
 }
@@ -249,7 +322,7 @@ API int storage_get_state(int storage_id, storage_state_e *state)
        /* external storage */
        ret = storage_ext_get_state(storage_id, &st);
        if (ret < 0) {
-               _E("Failed to get state (storage id(%d), ret(%d))", storage_id, ret);
+               _E("Failed to get state (storage id(%d), ret(%d))", storage_id, ret); //LCOV_EXCL_LINE
                return STORAGE_ERROR_OPERATION_FAILED;
        }
 
@@ -257,6 +330,21 @@ API int storage_get_state(int storage_id, storage_state_e *state)
        return STORAGE_ERROR_NONE;
 }
 
+//LCOV_EXCL_START Not called Callback
+static void compat_cb(int storage_id,
+               storage_dev_e dev, storage_state_e state,
+               const char *fstype, const char *fsuuid, const char *mountpath,
+               bool primary, int flags, void *user_data)
+{
+       struct compat_cb_info* ccb_info;
+       dd_list *elem;
+
+       if (storage_id == STORAGE_TYPE_EXTERNAL && dev == STORAGE_DEV_EXT_SDCARD)
+               DD_LIST_FOREACH(compat_cb_list, elem, ccb_info)
+                       ccb_info->user_cb(storage_id, state, ccb_info->user_data);
+}
+//LCOV_EXCL_STOP
+
 API int storage_set_state_changed_cb(int storage_id, storage_state_changed_cb callback, void *user_data)
 {
        const struct storage_ops *st;
@@ -264,6 +352,9 @@ API int storage_set_state_changed_cb(int storage_id, storage_state_changed_cb ca
        int ret;
        dd_list *elem;
 
+       struct compat_cb_info* ccb_info;
+       static int compat_cb_init = 0;
+
        if (storage_id < 0)
                return STORAGE_ERROR_NOT_SUPPORTED;
 
@@ -272,6 +363,26 @@ API int storage_set_state_changed_cb(int storage_id, storage_state_changed_cb ca
                return STORAGE_ERROR_INVALID_PARAMETER;
        }
 
+       /* For backward compatability */
+       if (storage_id == STORAGE_TYPE_EXTERNAL) {
+               if (!compat_cb_init) {
+                       ret = storage_set_changed_cb(STORAGE_TYPE_EXTERNAL, compat_cb, NULL);
+                       if (ret == STORAGE_ERROR_NONE)
+                               compat_cb_init = 1;
+                       else
+                               return ret;
+               }
+
+               ccb_info = malloc(sizeof(struct compat_cb_info));
+               if (ccb_info == NULL)
+                       return STORAGE_ERROR_OPERATION_FAILED;
+               ccb_info->user_cb = callback;
+               ccb_info->user_data = user_data;
+               DD_LIST_APPEND(compat_cb_list, ccb_info);
+
+               return STORAGE_ERROR_NONE;
+       }
+
        /* Internal storage does not support registering changed callback */
        DD_LIST_FOREACH(st_int_head, elem, st)
                if (st->storage_id == storage_id)
@@ -282,9 +393,9 @@ API int storage_set_state_changed_cb(int storage_id, storage_state_changed_cb ca
        info.state_cb = callback;
        info.user_data = user_data;
 
-       ret = storage_ext_register_cb(STORAGE_CALLBACK_STATE, &info);
+       ret = storage_ext_register_cb(STORAGE_CALLBACK_ID, &info);
        if (ret < 0) {
-               _E("Failed to register callback : id(%d)", storage_id);
+               _E("Failed to register callback : id(%d)", storage_id); //LCOV_EXCL_LINE
                return STORAGE_ERROR_OPERATION_FAILED;
        }
 
@@ -306,6 +417,21 @@ API int storage_unset_state_changed_cb(int storage_id, storage_state_changed_cb
                return STORAGE_ERROR_INVALID_PARAMETER;
        }
 
+       /* For backward compatability */
+       if (storage_id == STORAGE_TYPE_EXTERNAL) {
+               dd_list *elem_n;
+               struct compat_cb_info* ccb_info;
+
+               DD_LIST_FOREACH_SAFE(compat_cb_list, elem, elem_n, ccb_info) {
+                       if (ccb_info->user_cb == callback) {
+                               DD_LIST_REMOVE(compat_cb_list, ccb_info);
+                               free(ccb_info);
+                               return STORAGE_ERROR_NONE;
+                       }
+               }
+               return STORAGE_ERROR_OPERATION_FAILED;
+       }
+
        /* Internal storage does not support registering changed callback */
        DD_LIST_FOREACH(st_int_head, elem, st)
                if (st->storage_id == storage_id)
@@ -315,9 +441,9 @@ API int storage_unset_state_changed_cb(int storage_id, storage_state_changed_cb
        info.id = storage_id;
        info.state_cb = callback;
 
-       ret = storage_ext_unregister_cb(STORAGE_CALLBACK_STATE, &info);
+       ret = storage_ext_unregister_cb(STORAGE_CALLBACK_ID, &info);
        if (ret < 0) {
-               _E("Failed to unregister callback : id(%d)", storage_id);
+               _E("Failed to unregister callback : id(%d)", storage_id); //LCOV_EXCL_LINE
                return STORAGE_ERROR_OPERATION_FAILED;
        }
 
@@ -352,7 +478,7 @@ API int storage_get_total_space(int storage_id, unsigned long long *bytes)
 
 out:
        if (ret < 0) {
-               _E("Failed to get total memory : id(%d)", storage_id);
+               _E("Failed to get total memory : id(%d)", storage_id); //LCOV_EXCL_LINE
                if (ret == -ENOTSUP)
                        return STORAGE_ERROR_NOT_SUPPORTED;
                return STORAGE_ERROR_OPERATION_FAILED;
@@ -386,11 +512,11 @@ API int storage_get_available_space(int storage_id, unsigned long long *bytes)
        }
 
        /* external storage */
-       ret = storage_ext_get_space(storage_id,NULL, &avail);
+       ret = storage_ext_get_space(storage_id, NULL, &avail);
 
 out:
        if (ret < 0) {
-               _E("Failed to get available memory : id(%d)", storage_id);
+               _E("Failed to get available memory : id(%d)", storage_id); //LCOV_EXCL_LINE
                if (ret == -ENOTSUP)
                        return STORAGE_ERROR_NOT_SUPPORTED;
                return STORAGE_ERROR_OPERATION_FAILED;
@@ -399,3 +525,98 @@ out:
        *bytes = avail;
        return STORAGE_ERROR_NONE;
 }
+
+API int storage_set_changed_cb(storage_type_e type, storage_changed_cb callback, void *user_data)
+{
+       int ret;
+       struct storage_cb_info info;
+
+       if (type == STORAGE_TYPE_INTERNAL) {
+               _E("Internal storage is not supported");
+               return STORAGE_ERROR_NOT_SUPPORTED;
+       }
+
+       if (type != STORAGE_TYPE_EXTERNAL && type != STORAGE_TYPE_EXTENDED_INTERNAL) {
+               _E("Invalid type (%d)", type);
+               return STORAGE_ERROR_INVALID_PARAMETER;
+       }
+
+       if (!callback) {
+               _E("Callback is NULL");
+               return STORAGE_ERROR_INVALID_PARAMETER;
+       }
+
+       if (!storage_ext_is_supported()) {
+               _E("Block module is not enabled");
+               return STORAGE_ERROR_NOT_SUPPORTED;
+       }
+
+       /* external storage */
+       info.type = type;
+       info.type_cb = callback;
+       info.user_data = user_data;
+
+       ret = storage_ext_register_cb(STORAGE_CALLBACK_TYPE, &info);
+       if (ret < 0) {
+               _E("Failed to register storage callback(ret:%d)", ret); //LCOV_EXCL_LINE
+               return STORAGE_ERROR_OPERATION_FAILED;
+       }
+
+       return STORAGE_ERROR_NONE;
+}
+
+API int storage_unset_changed_cb(storage_type_e type, storage_changed_cb callback)
+{
+       struct storage_cb_info info;
+       int ret;
+
+       if (type == STORAGE_TYPE_INTERNAL) {
+               _E("Internal storage is not supported");
+               return STORAGE_ERROR_NOT_SUPPORTED;
+       }
+
+       if (type != STORAGE_TYPE_EXTERNAL && type != STORAGE_TYPE_EXTENDED_INTERNAL) {
+               _E("Invalid type (%d)", type);
+               return STORAGE_ERROR_INVALID_PARAMETER;
+       }
+
+       if (!callback) {
+               _E("Callback is NULL");
+               return STORAGE_ERROR_INVALID_PARAMETER;
+       }
+
+       if (!storage_ext_is_supported()) {
+               _E("Block module is not enabled");
+               return STORAGE_ERROR_NOT_SUPPORTED;
+       }
+
+       /* external storage */
+       info.type = type;
+       info.type_cb = callback;
+
+       ret = storage_ext_unregister_cb(STORAGE_CALLBACK_TYPE, &info);
+       if (ret < 0) {
+               _E("Failed to unregister storage callback(ret:%d)", ret); //LCOV_EXCL_LINE
+               return STORAGE_ERROR_OPERATION_FAILED;
+       }
+
+       return STORAGE_ERROR_NONE;
+}
+
+static void __CONSTRUCTOR__ init(void)
+{
+       const char *tmp;
+       char *token;
+       int i;
+
+       for (i = 0 ; i <= STORAGE_DIRECTORY_OTHERS ; i++) {
+               tmp = tzplatform_getenv(tz_id[i]);
+               if (tmp != NULL) {
+                       token = rindex(tmp, '/');
+                       if (token != NULL) {
+                               token++;
+                               dir_path[i] = strdup(token);
+                       }
+               }
+       }
+}