freeswitch/libs/unimrcp/libs/apr-toolkit/src/apt_net_server_task.c

392 lines
11 KiB
C

/*
* Copyright 2008 Arsen Chaloyan
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "apt_net_server_task.h"
#include "apt_task.h"
#include "apt_pool.h"
#include "apt_pollset.h"
#include "apt_cyclic_queue.h"
#include "apt_log.h"
/** Network server task */
struct apt_net_server_task_t {
apr_pool_t *pool;
apt_task_t *base;
void *obj;
apr_size_t max_connection_count;
apr_thread_mutex_t *guard;
apt_cyclic_queue_t *msg_queue;
apt_pollset_t *pollset;
/* Listening socket descriptor */
apr_sockaddr_t *sockaddr;
apr_socket_t *listen_sock;
apr_pollfd_t listen_sock_pfd;
const apt_net_server_vtable_t *server_vtable;
};
static apt_bool_t apt_net_server_task_msg_signal(apt_task_t *task, apt_task_msg_t *msg);
static apt_bool_t apt_net_server_task_run(apt_task_t *task);
static apt_bool_t apt_net_server_task_on_destroy(apt_task_t *task);
/** Create connection task */
APT_DECLARE(apt_net_server_task_t*) apt_net_server_task_create(
const char *listen_ip,
apr_port_t listen_port,
apr_size_t max_connection_count,
void *obj,
const apt_net_server_vtable_t *server_vtable,
apt_task_msg_pool_t *msg_pool,
apr_pool_t *pool)
{
apt_task_vtable_t *vtable;
apt_net_server_task_t *task;
task = apr_palloc(pool,sizeof(apt_net_server_task_t));
task->pool = pool;
task->obj = obj;
task->sockaddr = NULL;
task->listen_sock = NULL;
task->pollset = NULL;
task->max_connection_count = max_connection_count;
apr_sockaddr_info_get(&task->sockaddr,listen_ip,APR_INET,listen_port,0,task->pool);
if(!task->sockaddr) {
return NULL;
}
if(!server_vtable || !server_vtable->on_connect ||
!server_vtable->on_disconnect || !server_vtable->on_receive) {
return NULL;
}
task->server_vtable = server_vtable;
task->base = apt_task_create(task,msg_pool,pool);
if(!task->base) {
return NULL;
}
vtable = apt_task_vtable_get(task->base);
if(vtable) {
vtable->run = apt_net_server_task_run;
vtable->destroy = apt_net_server_task_on_destroy;
vtable->signal_msg = apt_net_server_task_msg_signal;
}
task->msg_queue = apt_cyclic_queue_create(CYCLIC_QUEUE_DEFAULT_SIZE);
apr_thread_mutex_create(&task->guard,APR_THREAD_MUTEX_UNNESTED,pool);
return task;
}
/** Virtual destroy handler */
static apt_bool_t apt_net_server_task_on_destroy(apt_task_t *base)
{
apt_net_server_task_t *task = apt_task_object_get(base);
if(task->guard) {
apr_thread_mutex_destroy(task->guard);
task->guard = NULL;
}
if(task->msg_queue) {
apt_cyclic_queue_destroy(task->msg_queue);
task->msg_queue = NULL;
}
return TRUE;
}
/** Destroy connection task. */
APT_DECLARE(apt_bool_t) apt_net_server_task_destroy(apt_net_server_task_t *task)
{
return apt_task_destroy(task->base);
}
/** Start connection task. */
APT_DECLARE(apt_bool_t) apt_net_server_task_start(apt_net_server_task_t *task)
{
return apt_task_start(task->base);
}
/** Terminate connection task. */
APT_DECLARE(apt_bool_t) apt_net_server_task_terminate(apt_net_server_task_t *task)
{
return apt_task_terminate(task->base,TRUE);
}
/** Get task */
APT_DECLARE(apt_task_t*) apt_net_server_task_base_get(apt_net_server_task_t *task)
{
return task->base;
}
/** Get task vtable */
APT_DECLARE(apt_task_vtable_t*) apt_net_server_task_vtable_get(apt_net_server_task_t *task)
{
return apt_task_vtable_get(task->base);
}
/** Get external object */
APT_DECLARE(void*) apt_net_server_task_object_get(apt_net_server_task_t *task)
{
return task->obj;
}
/** Create listening socket and add to pollset */
static apt_bool_t apt_net_server_task_listen_socket_create(apt_net_server_task_t *task)
{
apr_status_t status;
if(!task->sockaddr) {
return FALSE;
}
/* create listening socket */
status = apr_socket_create(&task->listen_sock, task->sockaddr->family, SOCK_STREAM, APR_PROTO_TCP, task->pool);
if(status != APR_SUCCESS) {
return FALSE;
}
apr_socket_opt_set(task->listen_sock, APR_SO_NONBLOCK, 0);
apr_socket_timeout_set(task->listen_sock, -1);
apr_socket_opt_set(task->listen_sock, APR_SO_REUSEADDR, 1);
status = apr_socket_bind(task->listen_sock, task->sockaddr);
if(status != APR_SUCCESS) {
apr_socket_close(task->listen_sock);
task->listen_sock = NULL;
return FALSE;
}
status = apr_socket_listen(task->listen_sock, SOMAXCONN);
if(status != APR_SUCCESS) {
apr_socket_close(task->listen_sock);
task->listen_sock = NULL;
return FALSE;
}
memset(&task->listen_sock_pfd,0,sizeof(apr_pollfd_t));
task->listen_sock_pfd.desc_type = APR_POLL_SOCKET;
task->listen_sock_pfd.reqevents = APR_POLLIN;
task->listen_sock_pfd.desc.s = task->listen_sock;
task->listen_sock_pfd.client_data = task->listen_sock;
if(apt_pollset_add(task->pollset, &task->listen_sock_pfd) != TRUE) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Add Listen Socket to Pollset");
apr_socket_close(task->listen_sock);
task->listen_sock = NULL;
}
return TRUE;
}
/** Remove from pollset and destroy listening socket */
static void apt_net_server_task_listen_socket_destroy(apt_net_server_task_t *task)
{
apt_pollset_remove(task->pollset,&task->listen_sock_pfd);
if(task->listen_sock) {
apr_socket_close(task->listen_sock);
task->listen_sock = NULL;
}
}
/** Create the pollset */
static apt_bool_t apt_net_server_task_pollset_create(apt_net_server_task_t *task)
{
/* create pollset */
task->pollset = apt_pollset_create((apr_uint32_t)task->max_connection_count + 1, task->pool);
if(!task->pollset) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Create Pollset");
return FALSE;
}
/* create listening socket */
if(apt_net_server_task_listen_socket_create(task) != TRUE) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Create Listen Socket");
}
return TRUE;
}
/** Destroy the pollset */
static void apt_net_server_task_pollset_destroy(apt_net_server_task_t *task)
{
apt_net_server_task_listen_socket_destroy(task);
if(task->pollset) {
apt_pollset_destroy(task->pollset);
task->pollset = NULL;
}
}
static apt_bool_t apt_net_server_task_process(apt_net_server_task_t *task)
{
apt_bool_t status = TRUE;
apt_bool_t running = TRUE;
apt_task_msg_t *msg;
do {
apr_thread_mutex_lock(task->guard);
msg = apt_cyclic_queue_pop(task->msg_queue);
apr_thread_mutex_unlock(task->guard);
if(msg) {
status = apt_task_msg_process(task->base,msg);
}
else {
running = FALSE;
}
}
while(running == TRUE);
return status;
}
static apt_bool_t apt_net_server_task_accept(apt_net_server_task_t *task)
{
char *local_ip = NULL;
char *remote_ip = NULL;
apr_sockaddr_t *l_sockaddr = NULL;
apr_sockaddr_t *r_sockaddr = NULL;
apt_net_server_connection_t *connection;
apr_pool_t *pool = apt_pool_create();
if(!pool) {
return FALSE;
}
connection = apr_palloc(pool,sizeof(apt_net_server_connection_t));
connection->pool = pool;
connection->obj = NULL;
connection->sock = NULL;
connection->client_ip = NULL;
if(apr_socket_accept(&connection->sock,task->listen_sock,connection->pool) != APR_SUCCESS) {
apr_pool_destroy(pool);
return FALSE;
}
if(apr_socket_addr_get(&l_sockaddr,APR_LOCAL,connection->sock) != APR_SUCCESS ||
apr_socket_addr_get(&r_sockaddr,APR_REMOTE,connection->sock) != APR_SUCCESS) {
apr_pool_destroy(pool);
return FALSE;
}
apr_sockaddr_ip_get(&local_ip,l_sockaddr);
apr_sockaddr_ip_get(&remote_ip,r_sockaddr);
connection->client_ip = remote_ip;
connection->id = apr_psprintf(pool,"%s:%hu <-> %s:%hu",
local_ip,l_sockaddr->port,
remote_ip,r_sockaddr->port);
memset(&connection->sock_pfd,0,sizeof(apr_pollfd_t));
connection->sock_pfd.desc_type = APR_POLL_SOCKET;
connection->sock_pfd.reqevents = APR_POLLIN;
connection->sock_pfd.desc.s = connection->sock;
connection->sock_pfd.client_data = connection;
if(apt_pollset_add(task->pollset,&connection->sock_pfd) != TRUE) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Add to Pollset");
apr_socket_close(connection->sock);
apr_pool_destroy(pool);
return FALSE;
}
apt_log(APT_LOG_MARK,APT_PRIO_NOTICE,"Accepted TCP Connection %s",connection->id);
task->server_vtable->on_connect(task,connection);
return TRUE;
}
static apt_bool_t apt_net_server_task_run(apt_task_t *base)
{
apt_net_server_task_t *task = apt_task_object_get(base);
apt_bool_t running = TRUE;
apr_status_t status;
apr_int32_t num;
const apr_pollfd_t *ret_pfd;
int i;
if(!task) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Start Network Server Task");
return FALSE;
}
if(apt_net_server_task_pollset_create(task) == FALSE) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Create Pollset");
return FALSE;
}
while(running) {
status = apt_pollset_poll(task->pollset, -1, &num, &ret_pfd);
if(status != APR_SUCCESS) {
continue;
}
for(i = 0; i < num; i++) {
if(ret_pfd[i].desc.s == task->listen_sock) {
apt_log(APT_LOG_MARK,APT_PRIO_DEBUG,"Accept Connection");
apt_net_server_task_accept(task);
continue;
}
if(apt_pollset_is_wakeup(task->pollset,&ret_pfd[i])) {
apt_log(APT_LOG_MARK,APT_PRIO_DEBUG,"Process Control Message");
if(apt_net_server_task_process(task) == FALSE) {
running = FALSE;
break;
}
continue;
}
apt_log(APT_LOG_MARK,APT_PRIO_DEBUG,"Process Message");
task->server_vtable->on_receive(task,ret_pfd[i].client_data);
}
}
apt_net_server_task_pollset_destroy(task);
apt_task_child_terminate(task->base);
return TRUE;
}
static apt_bool_t apt_net_server_task_msg_signal(apt_task_t *base, apt_task_msg_t *msg)
{
apt_bool_t status;
apt_net_server_task_t *task = apt_task_object_get(base);
apr_thread_mutex_lock(task->guard);
status = apt_cyclic_queue_push(task->msg_queue,msg);
apr_thread_mutex_unlock(task->guard);
if(apt_pollset_wakeup(task->pollset) != TRUE) {
apt_log(APT_LOG_MARK,APT_PRIO_WARNING,"Failed to Signal Control Message");
status = FALSE;
}
return status;
}
/** Close connection */
APT_DECLARE(apt_bool_t) apt_net_server_connection_close(apt_net_server_task_t *task, apt_net_server_connection_t *connection)
{
if(connection->sock) {
apt_log(APT_LOG_MARK,APT_PRIO_INFO,"Close TCP Connection %s",connection->id);
apt_pollset_remove(task->pollset,&connection->sock_pfd);
apr_socket_close(connection->sock);
connection->sock = NULL;
task->server_vtable->on_disconnect(task,connection);
}
return TRUE;
}
/** Destroy connection */
APT_DECLARE(void) apt_net_server_connection_destroy(apt_net_server_connection_t *connection)
{
apt_log(APT_LOG_MARK,APT_PRIO_NOTICE,"Destroy TCP Connection %s",connection->id);
apr_pool_destroy(connection->pool);
}