diff options
Diffstat (limited to 'unit/rilmodem-test-server.c')
-rw-r--r-- | unit/rilmodem-test-server.c | 196 |
1 files changed, 196 insertions, 0 deletions
diff --git a/unit/rilmodem-test-server.c b/unit/rilmodem-test-server.c new file mode 100644 index 00000000..52a3c745 --- /dev/null +++ b/unit/rilmodem-test-server.c @@ -0,0 +1,196 @@ +/* + * + * oFono - Open Source Telephony + * + * Copyright (C) 2015 Canonical Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + * + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#define _GNU_SOURCE +#include <netinet/in.h> +#include <sys/socket.h> +#include <sys/un.h> +#include <unistd.h> + +#include <ofono/types.h> + +#include <gril.h> + +#include "rilmodem-test-server.h" + +#define MAX_REQUEST_SIZE 4096 + +static int server_sk; +static ConnectFunc connect_func; +static GIOChannel *server_io; +static const struct rilmodem_test_data *rtd; + +/* Warning: length is stored in network order */ +struct rsp_hdr { + uint32_t length; + uint32_t unsolicited; + uint32_t serial; + uint32_t error; +}; + +static gboolean read_server(gpointer data) +{ + GIOStatus status; + gsize offset, rbytes, wbytes; + gchar *buf, *bufp; + uint32_t req_serial; + struct rsp_hdr rsp; + + buf = g_malloc0(MAX_REQUEST_SIZE); + + status = g_io_channel_read_chars(server_io, buf, MAX_REQUEST_SIZE, + &rbytes, NULL); + g_assert(status == G_IO_STATUS_NORMAL); + + g_assert(rbytes == rtd->req_size); + + /* validate len, and request_id */ + g_assert(!memcmp(buf, rtd->req_data, (sizeof(uint32_t) * 2))); + + /* + * header: size (uint32), reqid (uin32), serial (uint32) + * header size == 16 ( excludes sizeof(size) ) + */ + + /* advance past request_no */ + bufp = buf + (sizeof(uint32_t) * 2); + + req_serial = (uint32_t) *bufp; + + /* advance past serial_no */ + bufp += sizeof(uint32_t); + + /* validate the rest of the parcel... */ + offset = (sizeof(uint32_t) * 3); + g_assert(!memcmp(bufp, rtd->req_data + offset, + rtd->req_size - offset)); + + /* Length does not include the length field. Network order. */ + rsp.length = htonl(sizeof(rsp) - sizeof(rsp.length) + rtd->rsp_size); + rsp.unsolicited = 0; + rsp.serial = req_serial; + rsp.error = rtd->rsp_error; + + /* copy header */ + memcpy(buf, &rsp, sizeof(rsp)); + + if (rtd->rsp_size) { + bufp = buf + sizeof(rsp); + + memcpy(bufp, rtd->rsp_data, rtd->rsp_size); + } + + + status = g_io_channel_write_chars(server_io, + buf, + sizeof(rsp) + rtd->rsp_size, + &wbytes, NULL); + + /* FIXME: assert wbytes is correct */ + + g_assert(status == G_IO_STATUS_NORMAL); + + g_free(buf); + g_io_channel_unref(server_io); + + return FALSE; +} + +static gboolean on_socket_connected(GIOChannel *chan, GIOCondition cond, + gpointer data) +{ + struct sockaddr saddr; + unsigned int len = sizeof(saddr); + int fd; + GIOStatus status; + + g_assert(cond == G_IO_IN); + + fd = accept(server_sk, &saddr, &len); + g_assert(fd != -1); + + server_io = g_io_channel_unix_new(fd); + g_assert(server_io != NULL); + + if (connect_func) + connect_func(data); + + status = g_io_channel_set_encoding(server_io, NULL, NULL); + g_assert(status == G_IO_STATUS_NORMAL); + + g_io_channel_set_buffered(server_io, FALSE); + g_io_channel_set_close_on_unref(server_io, TRUE); + + g_idle_add(read_server, data); + + return FALSE; +} + +void rilmodem_test_server_close(void) +{ + g_assert(server_sk); + close(server_sk); + server_sk = 0; +} + +void rilmodem_test_server_create(ConnectFunc connect, + const struct rilmodem_test_data *test_data, + void *data) +{ + GIOChannel *io; + struct sockaddr_un addr; + int retval; + + g_assert(server_sk == 0); + + connect_func = connect; + rtd = test_data; + + server_sk = socket(AF_UNIX, SOCK_STREAM, 0); + g_assert(server_sk); + + memset(&addr, 0, sizeof(addr)); + addr.sun_family = AF_UNIX; + strncpy(addr.sun_path, RIL_SERVER_SOCK_PATH, sizeof(addr.sun_path) - 1); + + /* Unlink any existing socket for this session */ + unlink(addr.sun_path); + + retval = bind(server_sk, (struct sockaddr *) &addr, sizeof(addr)); + g_assert(retval >= 0); + + retval = listen(server_sk, 0); + g_assert(retval >= 0); + + io = g_io_channel_unix_new(server_sk); + g_assert(io != NULL); + + g_io_channel_set_close_on_unref(io, TRUE); + g_io_add_watch_full(io, G_PRIORITY_DEFAULT, + G_IO_IN | G_IO_HUP | G_IO_ERR | G_IO_NVAL, + on_socket_connected, data, NULL); + + g_io_channel_unref(io); +} |