• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 #include "init_service_socket.h"
16 
17 #include <errno.h>
18 #include <fcntl.h>
19 #include <netinet/in.h>
20 #include <string.h>
21 #include <sys/socket.h>
22 #include <sys/stat.h>
23 #include <sys/uio.h>
24 
25 #include "init_log.h"
26 #include "init_service.h"
27 #include "loop_event.h"
28 #include "securec.h"
29 #define SOCKET_BUFF_SIZE (256 * 1024)
30 
31 #define HOS_SOCKET_DIR "/dev/unix/socket"
32 #define HOS_SOCKET_ENV_PREFIX "OHOS_SOCKET_"
33 #define MAX_SOCKET_ENV_PREFIX_LEN 64
34 #define MAX_SOCKET_FD_LEN 16
35 
GetSocketAddr(struct sockaddr_un * addr,const char * name)36 static int GetSocketAddr(struct sockaddr_un *addr, const char *name)
37 {
38     (void)memset_s(addr, sizeof(struct sockaddr_un), 0x0, sizeof(struct sockaddr_un));
39     addr->sun_family = AF_UNIX;
40     size_t addrLen = sizeof(addr->sun_path);
41     int ret = snprintf_s(addr->sun_path, addrLen, addrLen - 1, HOS_SOCKET_DIR "/%s", name);
42     INIT_ERROR_CHECK(ret >= 0, return -1, "Failed to format addr %s", name);
43     return 0;
44 }
45 
SetSocketAddr(ServiceSocket * sockopt,sockaddr_union * addr)46 static int SetSocketAddr(ServiceSocket *sockopt, sockaddr_union *addr)
47 {
48     int ret = 0;
49     if (sockopt->family == AF_NETLINK) {
50 #ifndef __LITEOS__
51         if (memset_s(&(addr->addrnl), sizeof(addr->addrnl), 0, sizeof(addr->addrnl)) != EOK) {
52             INIT_LOGE("Faild to clear socket address");
53             return -1;
54         }
55         addr->addrnl.nl_family = AF_NETLINK;
56         addr->addrnl.nl_pid = getpid();
57         addr->addrnl.nl_groups = 0xffffffff;
58 #else
59         INIT_LOGE("No support in liteos kernel");
60         return -1;
61 #endif
62     } else {
63         ret = GetSocketAddr(&(addr->addrun), sockopt->name);
64         INIT_ERROR_CHECK(ret == 0, return -1, "Failed to format addr %s", sockopt->name);
65         if (access(addr->addrun.sun_path, F_OK) == 0) {
66             INIT_LOGI("%s already exist, remove it", addr->addrun.sun_path);
67             unlink(addr->addrun.sun_path);
68         }
69     }
70     return ret;
71 }
72 
SetSocketOptionAndBind(ServiceSocket * sockopt)73 static int SetSocketOptionAndBind(ServiceSocket *sockopt)
74 {
75     if (sockopt->option & SOCKET_OPTION_PASSCRED) {
76         int on = 1;
77         if (setsockopt(sockopt->sockFd, SOL_SOCKET, SO_PASSCRED, &on, sizeof(on))) {
78             INIT_LOGE("Failed to setsockopt");
79             return -1;
80         }
81     }
82     if (sockopt->option & SOCKET_OPTION_RCVBUFFORCE) {
83         int buffSize = SOCKET_BUFF_SIZE;
84         if (setsockopt(sockopt->sockFd, SOL_SOCKET, SO_RCVBUFFORCE, &buffSize, sizeof(buffSize))) {
85             INIT_LOGE("Failed to setsockopt");
86             return -1;
87         }
88     }
89     sockaddr_union addr = {};
90     if (SetSocketAddr(sockopt, &addr) != 0) {
91         INIT_LOGE("Failed to set socket addr");
92         return -1;
93     }
94     if (sockopt->family == AF_NETLINK) {
95 #ifndef __LITEOS__
96         if (bind(sockopt->sockFd, (struct sockaddr *)&(addr.addrnl), sizeof(addr.addrnl))) {
97             INIT_LOGE("Create socket for service %s failed: %d", sockopt->name, errno);
98             return -1;
99         }
100 #else
101         INIT_LOGE("No support in liteos kernel");
102         return -1;
103 #endif
104     } else {
105         if (bind(sockopt->sockFd, (struct sockaddr *)&(addr.addrun), sizeof(addr.addrun))) {
106             INIT_LOGE("Create socket for service %s failed: %d", sockopt->name, errno);
107             return -1;
108         }
109         if (lchown(addr.addrun.sun_path, sockopt->uid, sockopt->gid)) {
110             INIT_LOGE("lchown fail %d ", errno);
111             unlink(addr.addrun.sun_path);
112             return -1;
113         }
114         if (fchmodat(AT_FDCWD, addr.addrun.sun_path, sockopt->perm, AT_SYMLINK_NOFOLLOW)) {
115             INIT_LOGE("fchmodat fail %d ", errno);
116             unlink(addr.addrun.sun_path);
117             return -1;
118         }
119     }
120     return 0;
121 }
122 
CreateSocket(ServiceSocket * sockopt)123 static int CreateSocket(ServiceSocket *sockopt)
124 {
125     INIT_ERROR_CHECK(sockopt != NULL, return SERVICE_FAILURE, "Invalid socket opt");
126     INIT_LOGI("name: %s, family: %d, type: %u, protocol: %d, perm: %u, uid: %u, gid: %u, option: %u",
127         sockopt->name, sockopt->family, sockopt->type, sockopt->protocol,
128         sockopt->perm, sockopt->uid,    sockopt->gid,  sockopt->option);
129     if (sockopt->sockFd >= 0) {
130         close(sockopt->sockFd);
131         sockopt->sockFd = -1;
132     }
133     sockopt->sockFd = socket(sockopt->family, sockopt->type, sockopt->protocol);
134     INIT_ERROR_CHECK(sockopt->sockFd >= 0, return -1, "socket fail %d ", errno);
135 
136     int ret = SetSocketOptionAndBind(sockopt);
137     if (ret != 0) {
138         close(sockopt->sockFd);
139         return -1;
140     }
141     INIT_LOGI("CreateSocket %s success", sockopt->name);
142     return sockopt->sockFd;
143 }
144 
SetSocketEnv(int fd,const char * name)145 static int SetSocketEnv(int fd, const char *name)
146 {
147     INIT_ERROR_CHECK(name != NULL, return SERVICE_FAILURE, "Invalid name");
148     char pubName[MAX_SOCKET_ENV_PREFIX_LEN] = { 0 };
149     char val[MAX_SOCKET_FD_LEN] = { 0 };
150     INIT_CHECK_RETURN_VALUE(snprintf_s(pubName, sizeof(pubName), sizeof(pubName) - 1, HOS_SOCKET_ENV_PREFIX "%s",
151         name) >= 0, -1);
152     INIT_CHECK_RETURN_VALUE(snprintf_s(val, sizeof(val), sizeof(val) - 1, "%d", fd) >= 0, -1);
153     int ret = setenv(pubName, val, 1);
154     INIT_ERROR_CHECK(ret >= 0, return -1, "setenv fail %d ", errno);
155     fcntl(fd, F_SETFD, 0);
156     return 0;
157 }
158 
ProcessWatchEvent_(const WatcherHandle watcherHandle,int fd,uint32_t * events,const void * context)159 static void ProcessWatchEvent_(const WatcherHandle watcherHandle, int fd, uint32_t *events, const void *context)
160 {
161     *events = 0;
162     Service *service = (Service *)context;
163     ServiceSocket *tmpSock = service->socketCfg;
164     while (tmpSock != NULL) {
165         if (tmpSock->sockFd == fd) {
166             tmpSock->watcher = NULL;
167             break;
168         }
169         tmpSock = tmpSock->next;
170     }
171     if (tmpSock == NULL) { // not found socket
172         INIT_LOGE("Service %s not match socket fd %d!", service->name, fd);
173         close(fd);
174         return;
175     }
176     INIT_LOGI("Socket information detected, fd:%d service name:%s", fd, service->name);
177     SocketDelWatcher(watcherHandle);
178     if (ServiceStart(service) != SERVICE_SUCCESS) {
179         INIT_LOGE("Service %s start failed!", service->name);
180     }
181 }
182 
SocketAddWatcher(ServiceWatcher * watcherHandle,Service * service,int fd)183 int SocketAddWatcher(ServiceWatcher *watcherHandle, Service *service, int fd)
184 {
185     WatcherHandle handle;
186     LE_WatchInfo info = {};
187     info.fd = fd;
188     info.flags = WATCHER_ONCE;
189     info.events = Event_Read;
190     info.processEvent = ProcessWatchEvent_;
191     int ret = LE_StartWatcher(LE_GetDefaultLoop(), &handle, &info, service);
192     INIT_LOGI("Start to monitor socket, fd:%d service name:%s", fd, service->name);
193     *watcherHandle = (ServiceWatcher)handle;
194     return ret;
195 }
196 
SocketDelWatcher(ServiceWatcher watcherHandle)197 void SocketDelWatcher(ServiceWatcher watcherHandle)
198 {
199     LE_RemoveWatcher(LE_GetDefaultLoop(), (WatcherHandle)watcherHandle);
200 }
201 
CreateServiceSocket(Service * service)202 int CreateServiceSocket(Service *service)
203 {
204     INIT_CHECK(service != NULL && service->socketCfg != NULL, return 0);
205     INIT_CHECK(service->socketCfg->sockFd == -1, return 0);
206     int ret = 0;
207     ServiceSocket *tmpSock = service->socketCfg;
208     while (tmpSock != NULL) {
209         int fd = CreateSocket(tmpSock);
210         INIT_CHECK_RETURN_VALUE(fd >= 0, -1);
211         if (IsOnDemandService(service)) {
212             if (IsConnectionBasedSocket(tmpSock)) {
213                 ret = listen(tmpSock->sockFd, MAX_SOCKET_FD_LEN);
214                 INIT_CHECK_RETURN_VALUE(ret == 0, -1);
215             }
216             if (strcmp(service->name, "ueventd") != 0) {
217                 ret = SocketAddWatcher(&tmpSock->watcher, service, tmpSock->sockFd);
218                 INIT_CHECK_RETURN_VALUE(ret == 0, -1);
219             }
220         }
221         ret = SetSocketEnv(fd, tmpSock->name);
222         INIT_CHECK_RETURN_VALUE(ret >= 0, -1);
223         tmpSock = tmpSock->next;
224     }
225     return 0;
226 }
227 
CloseServiceSocket(Service * service)228 void CloseServiceSocket(Service *service)
229 {
230     INIT_CHECK(service != NULL && service->socketCfg != NULL, return);
231     struct sockaddr_un addr;
232     ServiceSocket *sockopt = service->socketCfg;
233     while (sockopt != NULL) {
234         if (sockopt->watcher != NULL) {
235             SocketDelWatcher(sockopt->watcher);
236         }
237         if (sockopt->sockFd >= 0) {
238             close(sockopt->sockFd);
239             sockopt->sockFd = -1;
240         }
241         if (GetSocketAddr(&addr, sockopt->name) == 0) {
242             unlink(addr.sun_path);
243         }
244         sockopt = sockopt->next;
245     }
246     return;
247 }
248