4 * Copyright (c) 2010 - 2015 Samsung Electronics Co., Ltd. All rights reserved.
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
20 #include <sys/socket.h>
25 #include "ctsvc_internal.h"
26 #include "ctsvc_socket.h"
27 #include "ctsvc_mutex.h"
28 #include "ctsvc_inotify.h"
29 #ifdef _CONTACTS_IPC_CLIENT
30 #include "ctsvc_client_utils.h"
32 static int __ctsvc_conn_refcnt = 0;
33 static int __ctsvc_sockfd = -1;
35 int ctsvc_socket_init(void)
38 struct sockaddr_un caddr;
41 WARN("ctsvc_socket_init: socket ref count : %d", __ctsvc_conn_refcnt);
42 if (0 < __ctsvc_conn_refcnt) {
43 __ctsvc_conn_refcnt++;
44 return CONTACTS_ERROR_NONE;
47 char sock_file[CTSVC_PATH_MAX_LEN] = {0};
49 #ifdef _CONTACTS_IPC_CLIENT
50 if (ctsvc_client_is_in_system_session()) {
52 if (CONTACTS_ERROR_NONE != ctsvc_client_get_active_uid(&uid))
53 return CONTACTS_ERROR_SYSTEM;
58 snprintf(sock_file, sizeof(sock_file), CTSVC_SOCK_PATH"/%s", uid, CTSVC_SOCKET_FILE);
60 bzero(&caddr, sizeof(caddr));
61 caddr.sun_family = AF_UNIX;
62 snprintf(caddr.sun_path, sizeof(caddr.sun_path), "%s", sock_file);
64 __ctsvc_sockfd = socket(PF_UNIX, SOCK_STREAM, 0);
65 RETVM_IF(-1 == __ctsvc_sockfd, CONTACTS_ERROR_IPC,
66 "socket() Fail(errno = %d)", errno);
68 ret = connect(__ctsvc_sockfd, (struct sockaddr *)&caddr, sizeof(caddr));
71 ERR("connect() Fail(errno = %d)", errno);
72 close(__ctsvc_sockfd);
74 return CONTACTS_ERROR_IPC;
78 __ctsvc_conn_refcnt++;
79 return CONTACTS_ERROR_NONE;
82 void ctsvc_socket_final(void)
84 if (1 < __ctsvc_conn_refcnt) {
85 DBG("socket ref count : %d", __ctsvc_conn_refcnt);
86 __ctsvc_conn_refcnt--;
88 } else if (__ctsvc_conn_refcnt < 1) {
89 DBG("Please call connection API. socket ref count : %d", __ctsvc_conn_refcnt);
92 __ctsvc_conn_refcnt--;
94 close(__ctsvc_sockfd);
98 static inline int __ctsvc_safe_write(int fd, const char *buf, int buf_size)
102 ret = write(fd, buf+writed, buf_size);
115 static inline int __ctsvc_safe_read(int fd, char *buf, int buf_size)
117 int ret, read_size = 0;
119 ret = read(fd, buf+read_size, buf_size);
132 static int __ctsvc_socket_handle_return(int fd, ctsvc_socket_msg_s *msg)
137 ret = __ctsvc_safe_read(fd, (char *)msg, sizeof(ctsvc_socket_msg_s));
138 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_read() Fail(errno = %d)", errno);
140 WARN_IF(CTSVC_SOCKET_MSG_TYPE_REQUEST_RETURN_VALUE != msg->type,
141 "Unknown Type(%d), ret=%d, attach_num= %d,"
142 "attach1 = %d, attach2 = %d, attach3 = %d, attach4 = %d",
143 msg->type, msg->val, msg->attach_num,
144 msg->attach_sizes[0], msg->attach_sizes[1], msg->attach_sizes[2],
145 msg->attach_sizes[3]);
147 RETVM_IF(CTSVC_SOCKET_MSG_REQUEST_MAX_ATTACH < msg->attach_num, CONTACTS_ERROR_IPC,
148 "Invalid msg(attach_num = %d)", msg->attach_num);
150 return CONTACTS_ERROR_NONE;
153 static int __ctsvc_cancel_sim_import(int fd)
156 ctsvc_socket_msg_s msg = {0};
158 RETVM_IF(-1 == fd, CONTACTS_ERROR_IPC, "socket is not connected");
160 msg.type = CTSVC_SOCKET_MSG_TYPE_CANCEL_IMPORT_SIM;
162 ret = __ctsvc_safe_write(fd, (char *)&msg, sizeof(msg));
163 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_write() Fail(errno = %d)", errno);
165 return CONTACTS_ERROR_NONE;
168 static int __ctsvc_client_socket_response_handler(int fd, ctsvc_socket_msg_s *msg, contacts_sim_import_progress_cb callback, void *user_data)
174 ret = __ctsvc_safe_read(fd, (char *)msg, sizeof(ctsvc_socket_msg_s));
175 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_read() Fail(errno = %d)", errno);
176 RETVM_IF(CTSVC_SOCKET_MSG_REQUEST_MAX_ATTACH < msg->attach_num, CONTACTS_ERROR_IPC,
177 "Invalid msg(attach_num = %d)", msg->attach_num);
180 case CTSVC_SOCKET_MSG_TYPE_REQUEST_RETURN_VALUE:
183 case CTSVC_SOCKET_MSG_TYPE_REQUEST_INVOKE_CB:
184 DBG("total[%d], index[%d]", msg->val, msg->data);
186 if (false == callback(msg->val, msg->data, user_data))
187 __ctsvc_cancel_sim_import(fd);
193 /* LCOV_EXCL_START */
195 WARN("Unknown Type(%d), ret=%d, attach_num= %d,"
196 "attach1 = %d, attach2 = %d, attach3 = %d, attach4 = %d",
197 msg->type, msg->val, msg->attach_num,
198 msg->attach_sizes[0], msg->attach_sizes[1], msg->attach_sizes[2],
199 msg->attach_sizes[3]);
203 } while (CTSVC_SOCKET_MSG_TYPE_REQUEST_RETURN_VALUE != msg->type);
205 return CONTACTS_ERROR_NONE;
208 static void __ctsvc_remove_invalid_msg(int fd, int size)
211 char dummy[CTSVC_SOCKET_MSG_SIZE] = {0};
214 if (sizeof(dummy) < size) {
215 ret = read(fd, dummy, sizeof(dummy));
224 ret = read(fd, dummy, size);
236 int ctsvc_request_sim_import(int sim_slot_no, contacts_sim_import_progress_cb callback,
240 ctsvc_socket_msg_s msg = {0};
243 RETVM_IF(-1 == __ctsvc_sockfd, CONTACTS_ERROR_IPC, "socket is not connected");
245 msg.type = CTSVC_SOCKET_MSG_TYPE_REQUEST_IMPORT_SIM;
247 snprintf(src, sizeof(src), "%d", sim_slot_no);
249 msg.attach_sizes[0] = strlen(src);
251 ret = __ctsvc_safe_write(__ctsvc_sockfd, (char *)&msg, sizeof(msg));
252 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_write() Fail(errno = %d)", errno);
254 ret = __ctsvc_safe_write(__ctsvc_sockfd, src, msg.attach_sizes[0]);
255 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_write() Fail(errno = %d)", errno);
257 ret = __ctsvc_client_socket_response_handler(__ctsvc_sockfd, &msg, callback, user_data);
258 RETVM_IF(CONTACTS_ERROR_NONE != ret, ret, "__ctsvc_socket_handle_return() Fail(%d)", ret);
259 DBG("attach_num = %d", msg.attach_num);
261 for (i = 0; i < msg.attach_num; i++)
262 __ctsvc_remove_invalid_msg(__ctsvc_sockfd, msg.attach_sizes[i]);
267 int ctsvc_request_sim_get_initialization_status(int sim_slot_no, bool *completed)
270 ctsvc_socket_msg_s msg = {0};
271 char dest[CTSVC_SOCKET_MSG_SIZE] = {0};
274 RETVM_IF(-1 == __ctsvc_sockfd, CONTACTS_ERROR_IPC, "socket is not connected");
276 msg.type = CTSVC_SOCKET_MSG_TYPE_REQUEST_SIM_INIT_COMPLETE;
278 snprintf(src, sizeof(src), "%d", sim_slot_no);
280 msg.attach_sizes[0] = strlen(src);
282 ret = __ctsvc_safe_write(__ctsvc_sockfd, (char *)&msg, sizeof(msg));
283 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_write() Fail(errno = %d)", errno);
285 ret = __ctsvc_safe_write(__ctsvc_sockfd, src, msg.attach_sizes[0]);
286 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_write() Fail(errno = %d)", errno);
288 ret = __ctsvc_socket_handle_return(__ctsvc_sockfd, &msg);
289 RETVM_IF(CONTACTS_ERROR_NONE != ret, ret, "__ctsvc_socket_handle_return() Fail(%d)", ret);
290 DBG("attach_num = %d", msg.attach_num);
292 ret = __ctsvc_safe_read(__ctsvc_sockfd, dest, msg.attach_sizes[0]);
293 RETVM_IF(-1 == ret, CONTACTS_ERROR_IPC, "__ctsvc_safe_read() Fail(errno = %d)", errno);
300 INFO("sim init complete : %d", *completed);