157 lines
3.7 KiB
C
157 lines
3.7 KiB
C
|
/*
|
||
|
* Asterisk -- An open source telephony toolkit.
|
||
|
*
|
||
|
* Copyright (C) 2021, Sangoma Technologies Corporation
|
||
|
*
|
||
|
* Kevin Harwell <kharwell@sangoma.com>
|
||
|
*
|
||
|
* See http://www.asterisk.org for more information about
|
||
|
* the Asterisk project. Please do not directly contact
|
||
|
* any of the maintainers of this project for assistance;
|
||
|
* the project provides a web site, mailing lists and IRC
|
||
|
* channels for your use.
|
||
|
*
|
||
|
* This program is free software, distributed under the terms of
|
||
|
* the GNU General Public License Version 2. See the LICENSE file
|
||
|
* at the top of the source tree.
|
||
|
*/
|
||
|
|
||
|
#include "asterisk.h"
|
||
|
|
||
|
#include "asterisk/utils.h"
|
||
|
|
||
|
#include "logger.h"
|
||
|
#include "transport.h"
|
||
|
#include "transport_websocket.h"
|
||
|
|
||
|
struct aeap_transport *aeap_transport_create(const char *type)
|
||
|
{
|
||
|
struct aeap_transport *transport = NULL;
|
||
|
|
||
|
if (!strncasecmp(type, "ws", 2)) {
|
||
|
transport = (struct aeap_transport *)aeap_transport_websocket_create();
|
||
|
}
|
||
|
|
||
|
if (!transport) {
|
||
|
ast_log(LOG_ERROR, "AEAP transport: failed to create for type '%s'\n", type);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
ast_mutex_init(&transport->read_lock);
|
||
|
ast_mutex_init(&transport->write_lock);
|
||
|
|
||
|
transport->connected = 0;
|
||
|
|
||
|
return transport;
|
||
|
}
|
||
|
|
||
|
int aeap_transport_connect(struct aeap_transport *transport, const char *url,
|
||
|
const char *protocol, int timeout)
|
||
|
{
|
||
|
int res;
|
||
|
|
||
|
SCOPED_MUTEX(rlock, &transport->read_lock);
|
||
|
SCOPED_MUTEX(wlock, &transport->write_lock);
|
||
|
|
||
|
if (aeap_transport_is_connected(transport)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
res = transport->vtable->connect(transport, url, protocol, timeout);
|
||
|
if (!res) {
|
||
|
transport->connected = 1;
|
||
|
}
|
||
|
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
struct aeap_transport *aeap_transport_create_and_connect(const char *type,
|
||
|
const char *url, const char *protocol, int timeout)
|
||
|
{
|
||
|
struct aeap_transport *transport = aeap_transport_create(type);
|
||
|
|
||
|
if (!transport) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if (aeap_transport_connect(transport, url, protocol, timeout)) {
|
||
|
aeap_transport_destroy(transport);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
return transport;
|
||
|
}
|
||
|
|
||
|
int aeap_transport_is_connected(struct aeap_transport *transport)
|
||
|
{
|
||
|
/*
|
||
|
* Avoid using a lock to 'read' the 'connected' variable in order to
|
||
|
* keep things slightly more efficient.
|
||
|
*/
|
||
|
return ast_atomic_fetch_add(&transport->connected, 0, __ATOMIC_RELAXED);
|
||
|
}
|
||
|
|
||
|
int aeap_transport_disconnect(struct aeap_transport *transport)
|
||
|
{
|
||
|
int res;
|
||
|
|
||
|
SCOPED_MUTEX(rlock, &transport->read_lock);
|
||
|
SCOPED_MUTEX(wlock, &transport->write_lock);
|
||
|
|
||
|
if (!aeap_transport_is_connected(transport)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
res = transport->vtable->disconnect(transport);
|
||
|
|
||
|
/*
|
||
|
* Even though the transport is locked here use atomics to set the value of
|
||
|
* 'connected' since it's possible the variable is being 'read' by another
|
||
|
* thread via the 'is_connected' call.
|
||
|
*/
|
||
|
ast_atomic_fetch_sub(&transport->connected, 1, __ATOMIC_RELAXED);
|
||
|
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
void aeap_transport_destroy(struct aeap_transport *transport)
|
||
|
{
|
||
|
if (!transport) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
/* Ensure an orderly disconnect occurs before final destruction */
|
||
|
aeap_transport_disconnect(transport);
|
||
|
|
||
|
transport->vtable->destroy(transport);
|
||
|
|
||
|
ast_mutex_destroy(&transport->read_lock);
|
||
|
ast_mutex_destroy(&transport->write_lock);
|
||
|
|
||
|
ast_free(transport);
|
||
|
}
|
||
|
|
||
|
intmax_t aeap_transport_read(struct aeap_transport *transport, void *buf, intmax_t size,
|
||
|
enum AST_AEAP_DATA_TYPE *rtype)
|
||
|
{
|
||
|
SCOPED_MUTEX(lock, &transport->read_lock);
|
||
|
|
||
|
if (!aeap_transport_is_connected(transport)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return transport->vtable->read(transport, buf, size, rtype);
|
||
|
}
|
||
|
|
||
|
intmax_t aeap_transport_write(struct aeap_transport *transport, const void *buf, intmax_t size,
|
||
|
enum AST_AEAP_DATA_TYPE wtype)
|
||
|
{
|
||
|
SCOPED_MUTEX(lock, &transport->write_lock);
|
||
|
|
||
|
if (!aeap_transport_is_connected(transport)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return transport->vtable->write(transport, buf, size, wtype);
|
||
|
}
|