⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 su_epoll_port.c

📁 Sofia SIP is an open-source SIP User-Agent library, compliant with the IETF RFC3261 specification.
💻 C
📖 第 1 页 / 共 2 页
字号:
  ser->ser_next = indices[0], indices[0] = ser;  self->sup_n_registrations--;  self->sup_registers++;  return i;}/** Unregister a su_wait_t object. *   *  The function su_epoll_port_unregister() unregisters a su_wait_t object. The *  wait object, a callback function and a argument are removed from the *  port object. *  * @param self     - pointer to port object * @param root     - pointer to root object * @param wait     - pointer to wait object * @param callback - callback function pointer (may be NULL) * @param arg      - argument given to callback function when it is invoked  *                   (may be NULL) * * @deprecated Use su_epoll_port_deregister() instead.  * * @return Nonzero index of the wait object, or -1 upon an error. */int su_epoll_port_unregister(su_port_t *self,			     su_root_t *root, 			     su_wait_t *wait,				     su_wakeup_f callback, /* XXX - ignored */			     su_wakeup_arg_t *arg){  int i, I;  struct su_epoll_register *ser;  assert(self);  assert(su_port_own_thread(self));  I = self->sup_max_index;  for (i = 1; i <= I; i++) {    ser = self->sup_indices[i];    if (ser->ser_cb &&	arg == ser->ser_arg &&	SU_WAIT_CMP(wait[0], ser->ser_wait[0]) == 0)      return su_epoll_port_deregister0(self, ser->ser_id, 0);  }  su_seterrno(ENOENT);  return -1;}/** Deregister a su_wait_t object. *   *  Deregisters a registration by index. The wait object, a callback *  function and a argument are removed from the port object. The wait *  object is destroyed. *  * @param self     - pointer to port object * @param i        - registration index *  * @return Index of the wait object, or -1 upon an error. */int su_epoll_port_deregister(su_port_t *self, int i){  struct su_epoll_register *ser;  if (i <= 0 || i > self->sup_max_index)    return su_seterrno(EBADF);  ser = self->sup_indices[i];  if (!ser->ser_cb)    return su_seterrno(EBADF);  return su_epoll_port_deregister0(self, i, 1);}/** @internal * Unregister all su_wait_t objects of given su_root_t instance. * * The function su_epoll_port_unregister_all() unregisters all su_wait_t * objects associated with given root object. *  * @param  self     - pointer to port object * @param  root     - pointer to root object *  * @return Number of wait objects removed. */int su_epoll_port_unregister_all(su_port_t *self, su_root_t *root){  int i, I, n;  struct su_epoll_register *ser;  assert(self); assert(root);  assert(su_port_own_thread(self));  I = self->sup_max_index;  for (i = 1, n = 0; i <= I; i++) {    ser = self->sup_indices[i];    if (ser->ser_root != root)      continue;    su_epoll_port_deregister0(self, ser->ser_id, 0);    n++;  }  return n;}/**Set mask for a registered event. @internal * * The function su_epoll_port_eventmask() sets the mask describing events that can * signal the registered callback. * * @param port   pointer to port object * @param index  registration index * @param socket socket * @param events new event mask * * @retval 0 when successful, * @retval -1 upon an error. */int su_epoll_port_eventmask(su_port_t *self, int index, int socket, int events){  struct su_epoll_register *ser;  struct epoll_event ev;  if (index <= 0 || index > self->sup_max_index)    return su_seterrno(EBADF);  ser = self->sup_indices[index];  if (!ser->ser_cb)    return su_seterrno(EBADF);  ser->ser_wait->events = events;  ev.events = POLL2EPOLL(events);  ev.data.u64 = (uint64_t)0;  ev.data.u32 = (uint32_t)index;  if (epoll_ctl(self->sup_epoll, EPOLL_CTL_MOD, socket, &ev) == -1) {    SU_DEBUG_1(("su_port(%p): EPOLL_CTL_MOD(%u): %s\n", (void *)self, 		socket, su_strerror(su_errno())));    return -1;  }  return 0;}/** @internal Enable multishot mode. * * Enables, disables or queries the multishot mode for the port. The * multishot mode determines how the events are scheduled by port. If * multishot mode is enabled, port serves all the sockets that have received * network events. If it is disabled, only first socket event is served. * * @param self      pointer to port object * @param multishot multishot mode (0 => disables, 1 => enables, -1 => query) *  * @retval 0 multishot mode is disabled * @retval 1 multishot mode is enabled * @retval -1 an error occurred */staticint su_epoll_port_multishot(su_port_t *self, int multishot){  if (multishot < 0)    return self->sup_multishot;  else if (multishot == 0 || multishot == 1)    return self->sup_multishot = multishot;  else     return (errno = EINVAL), -1;}/** @internal * Wait (poll()) for wait objects in port. * * @param self     pointer to port * @param tout     timeout in milliseconds * * @return number of events handled */staticint su_epoll_port_wait_events(su_port_t *self, su_duration_t tout){  int j, n, events = 0, index;  unsigned version = self->sup_registers;  int const M = 4;  struct epoll_event ev[M];      n = epoll_wait(self->sup_epoll, ev, self->sup_multishot ? M : 1, tout);  assert(n <= M);  for (j = 0; j < n; j++) {    struct su_epoll_register *ser;    su_root_magic_t *magic;    index = (int)ev[j].data.u32;    if (!ev[j].events || index <= 0 || self->sup_max_index < index)      continue;    ser = self->sup_indices[index];    magic = ser->ser_root ? su_root_magic(ser->ser_root) : NULL;    ser->ser_wait->revents = ev[j].events;    ser->ser_cb(magic, ser->ser_wait, ser->ser_arg);    events++;    if (version != self->sup_registers)      /* Callback function used su_register()/su_deregister() */      return events;  }      return n;}/** Create a port using epoll() or poll(). */su_port_t *su_epoll_port_create(void){  su_port_t *self;  int epoll = epoll_create(su_root_size_hint);  if (epoll == -1) {    /* Fallback to poll() */    SU_DEBUG_3(("%s(): epoll_create() => %u: %s\n", 		"su_port_create", epoll, strerror(errno)));    return su_poll_port_create();  }  self = su_home_new(sizeof *self);  if (!self) {    close(epoll);    return self;  }  SU_DEBUG_9(("%s(%p): epoll_create() => %u: %s\n",	      "su_port_create", (void *)self, self->sup_epoll, "OK"));  if (su_home_destructor(su_port_home(self), su_epoll_port_deinit) < 0 ||      !(self->sup_indices = 	su_zalloc(su_port_home(self),		  (sizeof self->sup_indices[0]) * 		  (self->sup_size_indices = 64)))) {    su_home_unref(su_port_home(self));    close(epoll);    return NULL;  }  self->sup_epoll = epoll;  self->sup_multishot = SU_ENABLE_MULTISHOT_POLL;  if (su_socket_port_init(self->sup_base, su_epoll_port_vtable) < 0)    return su_home_unref(su_port_home(self)), NULL;  return self;}int su_epoll_clone_start(su_root_t *parent,			 su_clone_r return_clone,			 su_root_magic_t *magic,			 su_root_init_f init,			 su_root_deinit_f deinit){  return su_pthreaded_port_start(su_epoll_port_create, 				 parent, return_clone, magic, init, deinit);}#elsesu_port_t *su_epoll_port_create(void){  return su_default_port_create();}int su_epoll_clone_start(su_root_t *parent,			 su_clone_r return_clone,			 su_root_magic_t *magic,			 su_root_init_f init,			 su_root_deinit_f deinit){  return su_default_clone_start(parent, return_clone, magic, init, deinit);}#endif  /* HAVE_EPOLL */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -