#define EFL_NET_SERVER_UNIX_PROTECTED 1 #define EFL_NET_SERVER_FD_PROTECTED 1 #define EFL_NET_SERVER_PROTECTED 1 #define EFL_LOOP_FD_PROTECTED 1 #ifdef HAVE_CONFIG_H # include #endif #include "Ecore.h" #include "Ecore_Con.h" #include "ecore_con_private.h" #ifdef HAVE_SYS_SOCKET_H # include #endif #ifdef HAVE_SYS_UN_H #include #endif /* no include EVIL as it's not supposed to be compiled on Windows */ #define MY_CLASS EFL_NET_SERVER_UNIX_CLASS typedef struct _Efl_Net_Server_Unix_Data { Efl_Future *bind_job; Eina_Bool unlink_before_bind; } Efl_Net_Server_Unix_Data; EOLIAN static void _efl_net_server_unix_efl_object_destructor(Eo *o, Efl_Net_Server_Unix_Data *pd EINA_UNUSED) { SOCKET fd = efl_loop_fd_get(o); if (fd != INVALID_SOCKET) { const char *address = efl_net_server_address_get(o); if ((address) && (strncmp(address, "abstract:", strlen("abstract:")) != 0)) unlink(address); } efl_destructor(efl_super(o, MY_CLASS)); } static void _efl_net_server_unix_bind_job(void *data, const Efl_Event *event EINA_UNUSED) { Eo *o = data; Efl_Net_Server_Unix_Data *pd = efl_data_scope_get(o, MY_CLASS); const char *address = efl_net_server_address_get(o); struct sockaddr_un addr = { .sun_family = AF_UNIX }; socklen_t addrlen; SOCKET fd; Eina_Error err = 0; int r; pd->bind_job = NULL; efl_ref(o); /* we're emitting callbacks then continuing the workflow */ efl_net_server_fd_family_set(o, AF_UNIX); do { fd = efl_net_socket4(AF_UNIX, SOCK_STREAM, 0, efl_net_server_fd_close_on_exec_get(o)); if (fd == INVALID_SOCKET) { err = efl_net_socket_error_get(); ERR("socket(AF_UNIX, SOCK_STREAM, 0): %s", eina_error_msg_get(err)); goto error; } if (strncmp(address, "abstract:", strlen("abstract:")) == 0) { const char *path = address + strlen("abstract:"); if (strlen(path) + 2 > sizeof(addr.sun_path)) { ERR("abstract path is too long: %s", path); err = EFL_NET_SERVER_ERROR_COULDNT_RESOLVE_HOST; } addr.sun_path[0] = '\0'; memcpy(addr.sun_path + 1, path, strlen(path) + 1); addrlen = strlen(path) + 2 + offsetof(struct sockaddr_un, sun_path); } else { const char *path = address; if (strlen(path) + 1 > sizeof(addr.sun_path)) { ERR("path is too long: %s", path); err = EFL_NET_SERVER_ERROR_COULDNT_RESOLVE_HOST; } memcpy(addr.sun_path, path, strlen(path) + 1); addrlen = strlen(path) + 1 + offsetof(struct sockaddr_un, sun_path); } if ((pd->unlink_before_bind) && (addr.sun_path[0] != '\0')) { DBG("unlinking AF_UNIX path '%s'", addr.sun_path); unlink(addr.sun_path); } r = bind(fd, (struct sockaddr *)&addr, addrlen); if (r != 0) { err = efl_net_socket_error_get(); if ((err == EADDRINUSE) && (pd->unlink_before_bind) && (addr.sun_path[0] != '\0')) { closesocket(fd); err = 0; continue; } DBG("bind(%d, %s): %s", fd, address, eina_error_msg_get(err)); goto error; } break; } while (pd->unlink_before_bind); efl_loop_fd_set(o, fd); r = listen(fd, 0); if (r != 0) { err = efl_net_socket_error_get(); DBG("listen(%d): %s", fd, eina_error_msg_get(err)); goto error; } addrlen = sizeof(addr); if (getsockname(fd, (struct sockaddr *)&addr, &addrlen) != 0) { err = efl_net_socket_error_get(); ERR("getsockname(%d): %s", fd, eina_error_msg_get(err)); goto error; } else { char str[sizeof(addr) + sizeof("abstract:")]; if (!efl_net_unix_fmt(str, sizeof(str), fd, &addr, addrlen)) ERR("could not format unix address"); else { efl_net_server_address_set(o, str); address = efl_net_server_address_get(o); } } DBG("fd=%d serving at %s", fd, address); efl_net_server_serving_set(o, EINA_TRUE); error: if (err) { efl_event_callback_call(o, EFL_NET_SERVER_EVENT_ERROR, &err); if (fd) closesocket(fd); efl_loop_fd_set(o, INVALID_SOCKET); } efl_unref(o); } EOLIAN static Eina_Error _efl_net_server_unix_efl_net_server_fd_socket_activate(Eo *o, Efl_Net_Server_Unix_Data *pd EINA_UNUSED, const char *address) { EINA_SAFETY_ON_TRUE_RETURN_VAL(efl_loop_fd_get(o) != INVALID_SOCKET, EALREADY); EINA_SAFETY_ON_NULL_RETURN_VAL(address, EINVAL); #ifndef HAVE_SYSTEMD return efl_net_server_fd_socket_activate(efl_super(o, MY_CLASS), address); #else { char buf[INET6_ADDRSTRLEN + sizeof("[]:65536")]; Eina_Bool listening; Eina_Error err; struct sockaddr_storage *addr; socklen_t addrlen; int fd; err = efl_net_ip_socket_activate_check(address, AF_UNIX, SOCK_STREAM, &listening); if (err) return err; err = efl_net_server_fd_socket_activate(efl_super(o, MY_CLASS), address); if (err) return err; fd = efl_loop_fd_get(o); if (!listening) { if (listen(fd, 0) != 0) { err = efl_net_socket_error_get(); DBG("listen(%d): %s", fd, eina_error_msg_get(err)); goto error; } } addrlen = sizeof(addr); if (getsockname(fd, (struct sockaddr *)&addr, &addrlen) != 0) { err = efl_net_socket_error_get(); ERR("getsockname(%d): %s", fd, eina_error_msg_get(err)); goto error; } else if (efl_net_ip_port_fmt(buf, sizeof(buf), (struct sockaddr *)&addr)) efl_net_server_address_set(o, buf); DBG("fd=%d serving at %s", fd, address); efl_net_server_serving_set(o, EINA_TRUE); return 0; error: efl_net_server_fd_family_set(o, AF_UNSPEC); efl_loop_fd_set(o, INVALID_SOCKET); closesocket(fd); return err; } #endif } EOLIAN static Eina_Error _efl_net_server_unix_efl_net_server_serve(Eo *o, Efl_Net_Server_Unix_Data *pd, const char *address) { EINA_SAFETY_ON_NULL_RETURN_VAL(address, EINVAL); EINA_SAFETY_ON_TRUE_RETURN_VAL(address[0] == '\0', EINVAL); efl_net_server_address_set(o, address); if (pd->bind_job) efl_future_cancel(pd->bind_job); efl_future_use(&pd->bind_job, efl_loop_job(efl_loop_get(o), o)); efl_future_then(pd->bind_job, _efl_net_server_unix_bind_job, NULL, NULL, o); efl_future_link(o, pd->bind_job); return 0; } static Efl_Callback_Array_Item *_efl_net_server_unix_client_cbs(void); static void _efl_net_server_unix_client_event_closed(void *data, const Efl_Event *event) { Eo *server = data; Eo *client = event->object; efl_event_callback_array_del(client, _efl_net_server_unix_client_cbs(), server); if (efl_parent_get(client) == server) efl_parent_set(client, NULL); efl_net_server_clients_count_set(server, efl_net_server_clients_count_get(server) - 1); } EFL_CALLBACKS_ARRAY_DEFINE(_efl_net_server_unix_client_cbs, { EFL_IO_CLOSER_EVENT_CLOSED, _efl_net_server_unix_client_event_closed }); static void _efl_net_server_unix_efl_net_server_fd_client_add(Eo *o, Efl_Net_Server_Unix_Data *pd EINA_UNUSED, int client_fd) { Eo *client = efl_add(EFL_NET_SOCKET_UNIX_CLASS, o, efl_event_callback_array_add(efl_added, _efl_net_server_unix_client_cbs(), o), efl_io_closer_close_on_exec_set(efl_added, efl_net_server_fd_close_on_exec_get(o)), efl_io_closer_close_on_destructor_set(efl_added, EINA_TRUE), efl_loop_fd_set(efl_added, client_fd)); if (!client) { ERR("could not create client object fd=%d", client_fd); closesocket(client_fd); return; } efl_net_server_clients_count_set(o, efl_net_server_clients_count_get(o) + 1); efl_event_callback_call(o, EFL_NET_SERVER_EVENT_CLIENT_ADD, client); if (efl_ref_get(client) == 1) /* users must take a reference themselves */ { DBG("client %s was not handled, closing it...", efl_net_socket_address_remote_get(client)); efl_del(client); } } static void _efl_net_server_unix_efl_net_server_fd_client_reject(Eo *o, Efl_Net_Server_Unix_Data *pd EINA_UNUSED, int client_fd) { struct sockaddr_un addr; socklen_t addrlen; char str[sizeof(addr) + sizeof("abstract:")] = ""; addrlen = sizeof(addr); if (getpeername(client_fd, (struct sockaddr *)&addr, &addrlen) != 0) ERR("getpeername(%d): %s", client_fd, eina_error_msg_get(efl_net_socket_error_get())); else { if (!efl_net_unix_fmt(str, sizeof(str), client_fd, &addr, addrlen)) ERR("could not format rejected client unix address fd=%d", client_fd); } closesocket(client_fd); efl_event_callback_call(o, EFL_NET_SERVER_EVENT_CLIENT_REJECTED, str); } static void _efl_net_server_unix_unlink_before_bind_set(Eo *o EINA_UNUSED, Efl_Net_Server_Unix_Data *pd, Eina_Bool unlink_before_bind) { pd->unlink_before_bind = unlink_before_bind; } static Eina_Bool _efl_net_server_unix_unlink_before_bind_get(Eo *o EINA_UNUSED, Efl_Net_Server_Unix_Data *pd) { return pd->unlink_before_bind; } #include "efl_net_server_unix.eo.c"