104d7f3716
wireless/gs2200m/gs2200m_main.c: ioctl_request() supports SIOCGIFHWADDR Signed-off-by: Masayuki Ishikawa <Masayuki.Ishikawa@jp.sony.com> Approved-by: Gregory Nutt <gnutt@nuttx.org>
1501 lines
34 KiB
C
1501 lines
34 KiB
C
/****************************************************************************
|
|
* apps/wireless/gs2200m_main.c
|
|
*
|
|
* Copyright 2019 Sony Home Entertainment & Sound Products Inc.
|
|
* Author: Masayuki Ishikawa <Masayuki.Ishikawa@jp.sony.com>
|
|
*
|
|
* Based on usrsocktest_daemon.c
|
|
* Copyright (C) 2015, 2017 Haltian Ltd. All rights reserved.
|
|
* Author: Jussi Kivilinna <jussi.kivilinna@haltian.com>
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <debug.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <poll.h>
|
|
#include <unistd.h>
|
|
|
|
#include <sys/socket.h>
|
|
#include <sys/ioctl.h>
|
|
#include <netinet/in.h>
|
|
#include <arpa/inet.h>
|
|
#include <net/if.h>
|
|
|
|
#include <nuttx/net/usrsock.h>
|
|
#include <nuttx/wireless/gs2200m.h>
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
/* #define GS2200M_TRACE */
|
|
|
|
#ifdef GS2200M_TRACE
|
|
# define gs2200m_printf(v, ...) printf(v, ##__VA_ARGS__)
|
|
#else
|
|
# define gs2200m_printf(v, ...)
|
|
#endif
|
|
|
|
#ifndef MIN
|
|
# define MIN(a,b) (((a) < (b)) ? (a) : (b))
|
|
#endif
|
|
|
|
#define SOCKET_BASE 10000
|
|
#define SOCKET_COUNT 16
|
|
|
|
/****************************************************************************
|
|
* Private Data Types
|
|
****************************************************************************/
|
|
|
|
enum sock_state_e
|
|
{
|
|
CLOSED,
|
|
OPENED,
|
|
CONNECTED,
|
|
};
|
|
|
|
struct usock_s
|
|
{
|
|
int8_t type;
|
|
char cid;
|
|
enum sock_state_e state;
|
|
};
|
|
|
|
struct gs2200m_s
|
|
{
|
|
char *ssid;
|
|
char *key;
|
|
uint8_t mode;
|
|
uint8_t ch;
|
|
int gsfd;
|
|
struct usock_s sockets[SOCKET_COUNT];
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Function Prototypes
|
|
****************************************************************************/
|
|
|
|
static int socket_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int close_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int connect_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int sendto_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int recvfrom_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int setsockopt_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int getsockopt_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int getsockname_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int getpeername_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int ioctl_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int bind_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int listen_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
static int accept_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf);
|
|
|
|
/****************************************************************************
|
|
* Private Data
|
|
****************************************************************************/
|
|
|
|
static const struct usrsock_req_handler_s
|
|
{
|
|
uint32_t hdrlen;
|
|
int (CODE *fn)(int fd, FAR struct gs2200m_s *priv, FAR void *req);
|
|
} handlers[USRSOCK_REQUEST__MAX] =
|
|
{
|
|
[USRSOCK_REQUEST_SOCKET] =
|
|
{
|
|
sizeof(struct usrsock_request_socket_s),
|
|
socket_request,
|
|
},
|
|
[USRSOCK_REQUEST_CLOSE] =
|
|
{
|
|
sizeof(struct usrsock_request_close_s),
|
|
close_request,
|
|
},
|
|
[USRSOCK_REQUEST_CONNECT] =
|
|
{
|
|
sizeof(struct usrsock_request_connect_s),
|
|
connect_request,
|
|
},
|
|
[USRSOCK_REQUEST_SENDTO] =
|
|
{
|
|
sizeof(struct usrsock_request_sendto_s),
|
|
sendto_request,
|
|
},
|
|
[USRSOCK_REQUEST_RECVFROM] =
|
|
{
|
|
sizeof(struct usrsock_request_recvfrom_s),
|
|
recvfrom_request,
|
|
},
|
|
[USRSOCK_REQUEST_SETSOCKOPT] =
|
|
{
|
|
sizeof(struct usrsock_request_setsockopt_s),
|
|
setsockopt_request,
|
|
},
|
|
[USRSOCK_REQUEST_GETSOCKOPT] =
|
|
{
|
|
sizeof(struct usrsock_request_getsockopt_s),
|
|
getsockopt_request,
|
|
},
|
|
[USRSOCK_REQUEST_GETSOCKNAME] =
|
|
{
|
|
sizeof(struct usrsock_request_getsockname_s),
|
|
getsockname_request,
|
|
},
|
|
[USRSOCK_REQUEST_GETPEERNAME] =
|
|
{
|
|
sizeof(struct usrsock_request_getpeername_s),
|
|
getpeername_request,
|
|
},
|
|
[USRSOCK_REQUEST_BIND] =
|
|
{
|
|
sizeof(struct usrsock_request_bind_s),
|
|
bind_request,
|
|
},
|
|
[USRSOCK_REQUEST_LISTEN] =
|
|
{
|
|
sizeof(struct usrsock_request_listen_s),
|
|
listen_request,
|
|
},
|
|
[USRSOCK_REQUEST_ACCEPT] =
|
|
{
|
|
sizeof(struct usrsock_request_accept_s),
|
|
accept_request,
|
|
},
|
|
[USRSOCK_REQUEST_IOCTL] =
|
|
{
|
|
sizeof(struct usrsock_request_ioctl_s),
|
|
ioctl_request,
|
|
},
|
|
};
|
|
|
|
static struct gs2200m_s *_daemon;
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: _write_to_usock
|
|
****************************************************************************/
|
|
|
|
static int _write_to_usock(int fd, void *buf, size_t count)
|
|
{
|
|
ssize_t wlen;
|
|
|
|
wlen = write(fd, buf, count);
|
|
|
|
if (wlen < 0)
|
|
{
|
|
return -errno;
|
|
}
|
|
|
|
if (wlen != count)
|
|
{
|
|
return -ENOSPC;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: _send_ack_common
|
|
****************************************************************************/
|
|
|
|
static int _send_ack_common(int fd,
|
|
uint8_t xid,
|
|
FAR struct usrsock_message_req_ack_s *resp)
|
|
{
|
|
resp->head.msgid = USRSOCK_MESSAGE_RESPONSE_ACK;
|
|
resp->head.flags = 0;
|
|
resp->xid = xid;
|
|
|
|
/* Send ACK response. */
|
|
|
|
return _write_to_usock(fd, resp, sizeof(*resp));
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: gs2200m_socket_alloc
|
|
****************************************************************************/
|
|
|
|
static int16_t gs2200m_socket_alloc(FAR struct gs2200m_s *priv, int type)
|
|
{
|
|
FAR struct usock_s *usock;
|
|
int16_t i;
|
|
|
|
for (i = 0; i < SOCKET_COUNT; i++)
|
|
{
|
|
usock = &priv->sockets[i];
|
|
|
|
if (CLOSED == usock->state)
|
|
{
|
|
memset(usock, 0, sizeof(*usock));
|
|
usock->cid = 'z'; /* Invalidate cid */
|
|
usock->state = OPENED;
|
|
usock->type = type;
|
|
return i + SOCKET_BASE;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: gs2200m_socket_get
|
|
****************************************************************************/
|
|
|
|
static FAR struct usock_s *gs2200m_socket_get(FAR struct gs2200m_s *priv,
|
|
int sockid)
|
|
{
|
|
if (sockid < SOCKET_BASE)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
sockid -= SOCKET_BASE;
|
|
|
|
if (sockid >= SOCKET_COUNT)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
return &priv->sockets[sockid];
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: gs2200m_find_socket_by_cid
|
|
****************************************************************************/
|
|
|
|
static FAR struct usock_s *
|
|
gs2200m_find_socket_by_cid(FAR struct gs2200m_s *priv,
|
|
char cid)
|
|
{
|
|
FAR struct usock_s *ret = NULL;
|
|
int i;
|
|
|
|
for (i = 0; i < SOCKET_COUNT; i++)
|
|
{
|
|
if (priv->sockets[i].cid == cid)
|
|
{
|
|
ret = &priv->sockets[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: gs2200m_socket_free
|
|
****************************************************************************/
|
|
|
|
static int gs2200m_socket_free(FAR struct gs2200m_s *priv, int sockid)
|
|
{
|
|
FAR struct usock_s *usock = gs2200m_socket_get(priv, sockid);
|
|
|
|
if (!usock)
|
|
{
|
|
return -EBADFD;
|
|
}
|
|
|
|
if (CLOSED == usock->state)
|
|
{
|
|
return -EFAULT;
|
|
}
|
|
|
|
usock->state = CLOSED;
|
|
usock->cid = 'z'; /* invalid */
|
|
|
|
return 0;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: read_req
|
|
****************************************************************************/
|
|
|
|
static ssize_t
|
|
read_req(int fd, FAR const struct usrsock_request_common_s *com_hdr,
|
|
FAR void *req, size_t reqsize)
|
|
{
|
|
ssize_t rlen;
|
|
|
|
rlen = read(fd, (uint8_t *)req + sizeof(*com_hdr),
|
|
reqsize - sizeof(*com_hdr));
|
|
|
|
if (rlen < 0)
|
|
{
|
|
return -errno;
|
|
}
|
|
if (rlen + sizeof(*com_hdr) != reqsize)
|
|
{
|
|
return -EMSGSIZE;
|
|
}
|
|
|
|
return rlen;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: usrsock_request
|
|
****************************************************************************/
|
|
|
|
static int usrsock_request(int fd, FAR struct gs2200m_s *priv)
|
|
{
|
|
FAR struct usrsock_request_common_s *com_hdr;
|
|
uint8_t hdrbuf[16];
|
|
ssize_t rlen;
|
|
|
|
com_hdr = (FAR void *)hdrbuf;
|
|
rlen = read(fd, com_hdr, sizeof(*com_hdr));
|
|
|
|
if (rlen < 0)
|
|
{
|
|
return -errno;
|
|
}
|
|
|
|
if (rlen != sizeof(*com_hdr))
|
|
{
|
|
return -EMSGSIZE;
|
|
}
|
|
|
|
if (com_hdr->reqid >= USRSOCK_REQUEST__MAX ||
|
|
!handlers[com_hdr->reqid].fn)
|
|
{
|
|
ASSERT(false);
|
|
return -EIO;
|
|
}
|
|
|
|
assert(handlers[com_hdr->reqid].hdrlen < sizeof(hdrbuf));
|
|
|
|
rlen = read_req(fd, com_hdr, hdrbuf,
|
|
handlers[com_hdr->reqid].hdrlen);
|
|
|
|
if (rlen < 0)
|
|
{
|
|
return rlen;
|
|
}
|
|
|
|
return handlers[com_hdr->reqid].fn(fd, priv, hdrbuf);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: usock_send_event
|
|
****************************************************************************/
|
|
|
|
static int usock_send_event(int fd, FAR struct gs2200m_s *priv,
|
|
FAR struct usock_s *usock, int events)
|
|
{
|
|
FAR struct usrsock_message_socket_event_s event;
|
|
int i;
|
|
|
|
memset(&event, 0, sizeof(event));
|
|
event.head.flags = USRSOCK_MESSAGE_FLAG_EVENT;
|
|
event.head.msgid = USRSOCK_MESSAGE_SOCKET_EVENT;
|
|
|
|
for (i = 0; i < SOCKET_COUNT; i++)
|
|
{
|
|
if (usock == &priv->sockets[i])
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (i == SOCKET_COUNT)
|
|
{
|
|
return -EINVAL;
|
|
}
|
|
|
|
event.usockid = i + SOCKET_BASE;
|
|
event.events = events;
|
|
|
|
return _write_to_usock(fd, &event, sizeof(event));
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: socket_request
|
|
****************************************************************************/
|
|
|
|
static int socket_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_socket_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
FAR struct usock_s *usock;
|
|
int16_t usockid;
|
|
int ret;
|
|
|
|
gs2200m_printf("%s: start type=%d \n",
|
|
__func__, req->type);
|
|
|
|
/* Check domain requested */
|
|
|
|
if (req->domain != AF_INET)
|
|
{
|
|
usockid = -EAFNOSUPPORT;
|
|
}
|
|
else
|
|
{
|
|
/* Allocate socket. */
|
|
|
|
usockid = gs2200m_socket_alloc(priv, req->type);
|
|
ASSERT(0 < usockid);
|
|
}
|
|
|
|
/* Send ACK response */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = usockid;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
if (req->type == SOCK_DGRAM)
|
|
{
|
|
/* NOTE: If the socket type is DGRAM, it's ready to send
|
|
* a packet after creating user socket.
|
|
*/
|
|
|
|
usock = gs2200m_socket_get(priv, usockid);
|
|
(void)usock_send_event(fd, priv, usock,
|
|
USRSOCK_EVENT_SENDTO_READY);
|
|
}
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: close_request
|
|
****************************************************************************/
|
|
|
|
static int close_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_close_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
struct gs2200m_close_msg clmsg;
|
|
FAR struct usock_s *usock;
|
|
char cid;
|
|
int ret = 0;
|
|
|
|
gs2200m_printf("%s: start \n", __func__);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
cid = usock->cid;
|
|
|
|
if (SOCK_STREAM == usock->type && CONNECTED != usock->state)
|
|
{
|
|
ret = -EBADFD;
|
|
goto errout;
|
|
}
|
|
|
|
if (SOCK_DGRAM == usock->type && 'z' == cid)
|
|
{
|
|
/* the udp socket is not bound */
|
|
|
|
goto errout;
|
|
}
|
|
|
|
memset(&clmsg, 0, sizeof(clmsg));
|
|
clmsg.cid = cid;
|
|
(void)ioctl(priv->gsfd, GS2200M_IOC_CLOSE, (unsigned long)&clmsg);
|
|
|
|
errout:
|
|
|
|
/* Send ACK response */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
/* Free socket */
|
|
|
|
ret = gs2200m_socket_free(priv, req->usockid);
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: connect_request
|
|
****************************************************************************/
|
|
|
|
static int connect_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_connect_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
struct gs2200m_connect_msg cmsg;
|
|
struct sockaddr_in addr;
|
|
FAR struct usock_s *usock;
|
|
int events;
|
|
ssize_t wlen;
|
|
ssize_t rlen;
|
|
int ret = 0;
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: start \n", __func__);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if this socket is already connected. */
|
|
|
|
if (CONNECTED == usock->state)
|
|
{
|
|
ret = -EISCONN;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if address size ok. */
|
|
|
|
if (req->addrlen > sizeof(addr))
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Read address. */
|
|
|
|
rlen = read(fd, &addr, sizeof(addr));
|
|
|
|
if (rlen < 0 || rlen < req->addrlen)
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check address family. */
|
|
|
|
if (addr.sin_family != AF_INET)
|
|
{
|
|
ret = -EAFNOSUPPORT;
|
|
goto prepare;
|
|
}
|
|
|
|
snprintf(cmsg.addr, sizeof(cmsg.addr), "%s",
|
|
inet_ntoa(addr.sin_addr));
|
|
snprintf(cmsg.port, sizeof(cmsg.port), "%d",
|
|
ntohs(addr.sin_port));
|
|
|
|
cmsg.cid = 'z'; /* set to invalid */
|
|
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_CONNECT,
|
|
(unsigned long)&cmsg);
|
|
|
|
if (0 == ret)
|
|
{
|
|
usock->cid = cmsg.cid;
|
|
usock->state = CONNECTED;
|
|
}
|
|
else
|
|
{
|
|
ret = -errno;
|
|
}
|
|
|
|
prepare:
|
|
|
|
/* Send ACK response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
events = USRSOCK_EVENT_SENDTO_READY;
|
|
wlen = usock_send_event(fd, priv, usock, events);
|
|
|
|
if (wlen < 0)
|
|
{
|
|
return wlen;
|
|
}
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: sendto_request
|
|
****************************************************************************/
|
|
|
|
static int sendto_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_sendto_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
struct gs2200m_send_msg smsg;
|
|
FAR struct usock_s *usock;
|
|
uint8_t *sendbuf = NULL;
|
|
ssize_t wlen;
|
|
ssize_t rlen;
|
|
int nret;
|
|
int ret = 0;
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: start (buflen=%d) \n",
|
|
__func__, req->buflen);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if this socket is connected. */
|
|
|
|
if (SOCK_STREAM == usock->type && CONNECTED != usock->state)
|
|
{
|
|
ret = -ENOTCONN;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if the address size is non-zero.
|
|
* connection-mode socket does not accept address
|
|
*/
|
|
|
|
if (usock->type == SOCK_STREAM && req->addrlen > 0)
|
|
{
|
|
ret = -EISCONN;
|
|
goto prepare;
|
|
}
|
|
|
|
smsg.is_tcp = (usock->type == SOCK_STREAM) ? true : false;
|
|
|
|
/* For UDP, addlen must be provided */
|
|
|
|
if (usock->type == SOCK_DGRAM)
|
|
{
|
|
if (req->addrlen == 0)
|
|
{
|
|
ret = -EINVAL;
|
|
goto prepare;
|
|
}
|
|
|
|
/* In UDP case, read the address. */
|
|
|
|
rlen = read(fd, &smsg.addr, sizeof(smsg.addr));
|
|
|
|
if (rlen < 0 || rlen < req->addrlen)
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
gs2200m_printf("%s: addr: %s:%d",
|
|
__func__,
|
|
inet_ntoa(smsg.addr.sin_addr),
|
|
ntohs(smsg.addr.sin_port));
|
|
}
|
|
|
|
/* Check if the request has data. */
|
|
|
|
if (req->buflen > 0)
|
|
{
|
|
sendbuf = calloc(1, req->buflen);
|
|
ASSERT(sendbuf);
|
|
|
|
/* Read data from usrsock. */
|
|
|
|
rlen = read(fd, sendbuf, req->buflen);
|
|
|
|
if (rlen < 0 || rlen < req->buflen)
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
smsg.cid = usock->cid;
|
|
smsg.buf = sendbuf;
|
|
smsg.len = req->buflen;
|
|
|
|
nret = ioctl(priv->gsfd, GS2200M_IOC_SEND,
|
|
(unsigned long)&smsg);
|
|
|
|
if (usock->cid != smsg.cid)
|
|
{
|
|
usock->cid = smsg.cid;
|
|
}
|
|
|
|
if (0 != nret)
|
|
{
|
|
ret = -EINVAL;
|
|
goto prepare;
|
|
}
|
|
|
|
ret = req->buflen;
|
|
}
|
|
|
|
prepare:
|
|
|
|
if (sendbuf)
|
|
{
|
|
free(sendbuf);
|
|
}
|
|
|
|
/* Send ACK response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
/* Let kernel-side know that there is space for more send data. */
|
|
|
|
wlen = usock_send_event(fd, priv, usock,
|
|
USRSOCK_EVENT_SENDTO_READY);
|
|
|
|
if (wlen < 0)
|
|
{
|
|
return wlen;
|
|
}
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: recvfrom_request
|
|
****************************************************************************/
|
|
|
|
static int recvfrom_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_recvfrom_s *req = hdrbuf;
|
|
struct usrsock_message_datareq_ack_s resp;
|
|
struct gs2200m_recv_msg rmsg;
|
|
FAR struct usock_s *usock;
|
|
int ret = 0;
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: start (req->max_buflen=%d) \n",
|
|
__func__, req->max_buflen);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if this socket is connected. */
|
|
|
|
if (SOCK_STREAM == usock->type && CONNECTED != usock->state)
|
|
{
|
|
ret = -ENOTCONN;
|
|
goto prepare;
|
|
}
|
|
|
|
memset(&rmsg, 0, sizeof(rmsg));
|
|
rmsg.buf = calloc(1, 1500);
|
|
rmsg.cid = usock->cid;
|
|
rmsg.reqlen = req->max_buflen;
|
|
rmsg.is_tcp = (usock->type == SOCK_STREAM) ? true : false;
|
|
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_RECV,
|
|
(unsigned long)&rmsg);
|
|
|
|
if (0 == ret)
|
|
{
|
|
ret = rmsg.len;
|
|
}
|
|
else
|
|
{
|
|
ret = -errno;
|
|
}
|
|
|
|
if (!rmsg.is_tcp)
|
|
{
|
|
gs2200m_printf("%s: from (%s:%d) \n",
|
|
__func__,
|
|
inet_ntoa(rmsg.addr.sin_addr),
|
|
ntohs(rmsg.addr.sin_port));
|
|
}
|
|
|
|
prepare:
|
|
|
|
/* Prepare response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.reqack.result = ret;
|
|
resp.reqack.xid = req->head.xid;
|
|
resp.reqack.head.msgid = USRSOCK_MESSAGE_RESPONSE_DATA_ACK;
|
|
resp.reqack.head.flags = 0;
|
|
|
|
if (0 <= ret)
|
|
{
|
|
resp.valuelen_nontrunc = sizeof(rmsg.addr);
|
|
resp.valuelen = MIN(resp.valuelen_nontrunc,
|
|
req->max_addrlen);
|
|
|
|
if (0 == rmsg.len)
|
|
{
|
|
usock_send_event(fd, priv, usock,
|
|
USRSOCK_EVENT_REMOTE_CLOSED
|
|
);
|
|
}
|
|
}
|
|
|
|
/* Send response. */
|
|
|
|
ret = _write_to_usock(fd, &resp, sizeof(resp));
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
|
|
if (0 < resp.valuelen)
|
|
{
|
|
/* Send address (value) */
|
|
|
|
ret = _write_to_usock(fd, &rmsg.addr, resp.valuelen);
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
}
|
|
|
|
if (resp.reqack.result > 0)
|
|
{
|
|
/* Send buffer */
|
|
|
|
ret = _write_to_usock(fd, rmsg.buf, resp.reqack.result);
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
}
|
|
|
|
err_out:
|
|
|
|
gs2200m_printf("%s: *** end ret=%d \n", __func__, ret);
|
|
|
|
free(rmsg.buf);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: bind_request
|
|
****************************************************************************/
|
|
|
|
static int bind_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_bind_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
struct gs2200m_bind_msg bmsg;
|
|
FAR struct usock_s *usock;
|
|
struct sockaddr_in addr;
|
|
ssize_t rlen;
|
|
int ret = 0;
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: called **** \n", __func__);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check if address size ok. */
|
|
|
|
if (req->addrlen > sizeof(addr))
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Read address. */
|
|
|
|
rlen = read(fd, &addr, sizeof(addr));
|
|
|
|
if (rlen < 0 || rlen < req->addrlen)
|
|
{
|
|
ret = -EFAULT;
|
|
goto prepare;
|
|
}
|
|
|
|
/* Check address family. */
|
|
|
|
if (addr.sin_family != AF_INET)
|
|
{
|
|
ret = -EAFNOSUPPORT;
|
|
goto prepare;
|
|
}
|
|
|
|
snprintf(bmsg.port, sizeof(bmsg.port), "%d", ntohs(addr.sin_port));
|
|
bmsg.cid = 'z'; /* set to invalid */
|
|
bmsg.is_tcp = (usock->type == SOCK_STREAM) ? true : false;
|
|
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_BIND, (unsigned long)&bmsg);
|
|
|
|
if (0 == ret)
|
|
{
|
|
usock->cid = bmsg.cid;
|
|
}
|
|
|
|
prepare:
|
|
|
|
/* Send ACK response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
return OK;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: listen_request
|
|
****************************************************************************/
|
|
|
|
static int listen_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_listen_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
FAR struct usock_s *usock;
|
|
int ret = 0;
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: called **** \n", __func__);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
}
|
|
|
|
/* Send ACK response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: accept_request
|
|
****************************************************************************/
|
|
|
|
static int accept_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_accept_s *req = hdrbuf;
|
|
struct usrsock_message_datareq_ack_s resp;
|
|
struct gs2200m_accept_msg amsg;
|
|
FAR struct usock_s *usock;
|
|
FAR struct usock_s *new_usock = NULL;
|
|
struct sockaddr_in ep_addr;
|
|
int ret = 0;
|
|
int16_t usockid; /* usockid for new client */
|
|
|
|
DEBUGASSERT(priv);
|
|
DEBUGASSERT(req);
|
|
|
|
gs2200m_printf("%s: called **** \n", __func__);
|
|
|
|
/* Check if this socket exists. */
|
|
|
|
usock = gs2200m_socket_get(priv, req->usockid);
|
|
|
|
if (!usock)
|
|
{
|
|
ret = -EBADFD;
|
|
goto prepare;
|
|
}
|
|
|
|
/* TODO: need to check if specified socket exists */
|
|
|
|
/* Call gs2200m driver to obtain new cid for client */
|
|
|
|
amsg.cid = usock->cid; /* Set server cid */
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_ACCEPT, (unsigned long)&amsg);
|
|
|
|
if (-1 == ret)
|
|
{
|
|
goto prepare;
|
|
}
|
|
|
|
/* allocate socket. */
|
|
|
|
usockid = gs2200m_socket_alloc(priv, SOCK_STREAM);
|
|
ASSERT(0 < usockid);
|
|
new_usock = gs2200m_socket_get(priv, usockid);
|
|
|
|
/* Set cid for the new_usock to be used in gs2200m driver */
|
|
|
|
new_usock->cid = amsg.cid;
|
|
new_usock->state = CONNECTED;
|
|
|
|
prepare:
|
|
|
|
/* Prepare response. */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.reqack.xid = req->head.xid;
|
|
resp.reqack.head.msgid = USRSOCK_MESSAGE_RESPONSE_DATA_ACK;
|
|
resp.reqack.head.flags = 0;
|
|
|
|
if (0 == ret)
|
|
{
|
|
resp.reqack.result = 2; /* ep_addr + usock */
|
|
resp.valuelen_nontrunc = sizeof(ep_addr);
|
|
resp.valuelen = resp.valuelen_nontrunc;
|
|
}
|
|
else
|
|
{
|
|
resp.reqack.result = ret;
|
|
resp.valuelen = 0;
|
|
}
|
|
|
|
/* Send response. */
|
|
|
|
ret = _write_to_usock(fd, &resp, sizeof(resp));
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
|
|
if (resp.valuelen > 0)
|
|
{
|
|
/* Send address (value) */
|
|
|
|
/* TODO: ep_addr should be set */
|
|
|
|
memset(&ep_addr, 0, sizeof(ep_addr));
|
|
|
|
ret = _write_to_usock(fd, &ep_addr, resp.valuelen);
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
|
|
/* Send new usockid info */
|
|
|
|
ret = _write_to_usock(fd, &usockid, sizeof(usockid));
|
|
|
|
if (0 > ret)
|
|
{
|
|
goto err_out;
|
|
}
|
|
|
|
/* Set events ofr new_usock */
|
|
|
|
usock_send_event(fd, priv, new_usock,
|
|
USRSOCK_EVENT_SENDTO_READY
|
|
);
|
|
}
|
|
|
|
err_out:
|
|
gs2200m_printf("%s: end \n", __func__);
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: setsockopt_request
|
|
****************************************************************************/
|
|
|
|
static int setsockopt_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
ASSERT(false);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: getsockopt_request
|
|
****************************************************************************/
|
|
|
|
static int getsockopt_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
ASSERT(false);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: getsockname_request
|
|
****************************************************************************/
|
|
|
|
static int getsockname_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
ASSERT(false);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: getsockname_request
|
|
****************************************************************************/
|
|
|
|
static int getpeername_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
ASSERT(false);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: ioctl_request
|
|
****************************************************************************/
|
|
|
|
static int ioctl_request(int fd, FAR struct gs2200m_s *priv,
|
|
FAR void *hdrbuf)
|
|
{
|
|
FAR struct usrsock_request_ioctl_s *req = hdrbuf;
|
|
struct usrsock_message_req_ack_s resp;
|
|
struct usrsock_message_datareq_ack_s resp2;
|
|
struct gs2200m_ifreq_msg imsg;
|
|
bool getreq = false;
|
|
int ret = -EINVAL;
|
|
|
|
memset(&imsg.ifr, 0, sizeof(imsg.ifr));
|
|
|
|
switch (req->cmd)
|
|
{
|
|
case SIOCGIFHWADDR:
|
|
getreq = true;
|
|
break;
|
|
|
|
case SIOCSIFADDR:
|
|
case SIOCSIFDSTADDR:
|
|
case SIOCSIFNETMASK:
|
|
|
|
(void)read(fd, &imsg.ifr, sizeof(imsg.ifr));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
imsg.cmd = req->cmd;
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_IFREQ, (unsigned long)&imsg);
|
|
|
|
if (!getreq)
|
|
{
|
|
/* Send ACK response */
|
|
|
|
memset(&resp, 0, sizeof(resp));
|
|
resp.result = ret;
|
|
ret = _send_ack_common(fd, req->head.xid, &resp);
|
|
|
|
if (0 > ret)
|
|
{
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
if (getreq)
|
|
{
|
|
resp2.reqack.result = ret;
|
|
resp2.reqack.xid = req->head.xid;
|
|
resp2.reqack.head.msgid = USRSOCK_MESSAGE_RESPONSE_DATA_ACK;
|
|
resp2.reqack.head.flags = 0;
|
|
resp2.valuelen_nontrunc = sizeof(imsg.ifr);
|
|
resp2.valuelen = sizeof(imsg.ifr);
|
|
|
|
(void)_write_to_usock(fd, &resp2, sizeof(resp2));
|
|
|
|
/* Return struct ifreq address */
|
|
|
|
_write_to_usock(fd, &imsg.ifr, resp2.valuelen);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: gs2200m_loop
|
|
****************************************************************************/
|
|
|
|
static int gs2200m_loop(FAR struct gs2200m_s *priv)
|
|
{
|
|
struct gs2200m_assoc_msg amsg;
|
|
FAR struct usock_s *usock;
|
|
struct pollfd fds[2];
|
|
int fd[2];
|
|
char cid;
|
|
int ret;
|
|
|
|
fd[0] = open("/dev/usrsock", O_RDWR);
|
|
ASSERT(0 <= fd[0]);
|
|
|
|
fd[1] = open("/dev/gs2200m", O_RDWR);
|
|
ASSERT(0 <= fd[1]);
|
|
priv->gsfd = fd[1];
|
|
|
|
amsg.ssid = priv->ssid;
|
|
amsg.key = priv->key;
|
|
amsg.mode = priv->mode;
|
|
amsg.ch = priv->ch;
|
|
ret = ioctl(priv->gsfd, GS2200M_IOC_ASSOC, (unsigned long)&amsg);
|
|
|
|
if (0 != ret)
|
|
{
|
|
fprintf(stderr, "association failed : invalid ssid or key \n");
|
|
goto errout;
|
|
}
|
|
|
|
while (true)
|
|
{
|
|
memset(fds, 0, sizeof(fds));
|
|
|
|
/* Check events from usrsock and gs2200m */
|
|
|
|
fds[0].fd = fd[0];
|
|
fds[0].events = POLLIN;
|
|
fds[1].fd = fd[1];
|
|
fds[1].events = POLLIN;
|
|
|
|
ret = poll(fds, 2, -1);
|
|
ASSERT(0 < ret);
|
|
|
|
if (fds[0].revents & POLLIN)
|
|
{
|
|
ret = usrsock_request(fd[0], priv);
|
|
ASSERT(0 == ret);
|
|
}
|
|
|
|
if (fds[1].revents & POLLIN)
|
|
{
|
|
gs2200m_printf("=== %s: event from /dev/gs2200m \n",
|
|
__func__);
|
|
|
|
/* retrieve cid from gs2200m driver */
|
|
|
|
cid = 'z';
|
|
ret = read(fd[1], &cid, sizeof(cid));
|
|
ASSERT(ret == sizeof(cid));
|
|
|
|
/* find usock by the cid */
|
|
|
|
usock = gs2200m_find_socket_by_cid(priv, cid);
|
|
|
|
if (NULL == usock)
|
|
{
|
|
gs2200m_printf("=== %s: cid=%c not found (ignored) \n",
|
|
__func__, cid);
|
|
}
|
|
else
|
|
{
|
|
/* send event to call xxxx_request() */
|
|
|
|
usock_send_event(fd[0], priv, usock,
|
|
USRSOCK_EVENT_RECVFROM_AVAIL);
|
|
}
|
|
}
|
|
|
|
usleep(1);
|
|
}
|
|
|
|
errout:
|
|
close(fd[1]);
|
|
close(fd[0]);
|
|
|
|
gs2200m_printf("finished: ret=%d\n", __func__, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: _show_usage
|
|
****************************************************************************/
|
|
|
|
static void _show_usage(FAR char *cmd)
|
|
{
|
|
fprintf(stderr,
|
|
"Usage: %s [-a [ch]] ssid key \n\n", cmd);
|
|
fprintf(stderr,
|
|
"AP mode : specify -a option (optionaly with channel) with ssid\n"
|
|
" and 10 hex digits for WEP key \n");
|
|
fprintf(stderr,
|
|
"STA mode: specify ssid and passphrase (key) for WPA/WPA2 PSK \n");
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
#ifdef BUILD_MODULE
|
|
int main(int argc, FAR char *argv[])
|
|
#else
|
|
int gs2200m_main(int argc, char **argv)
|
|
#endif
|
|
{
|
|
int option;
|
|
int ret;
|
|
bool ap_mode = false;
|
|
|
|
if (_daemon)
|
|
{
|
|
fprintf(stderr, "%s is already running! \n", argv[0]);
|
|
return -1;
|
|
}
|
|
|
|
_daemon = calloc(sizeof(struct gs2200m_s), 1);
|
|
ASSERT(_daemon);
|
|
|
|
_daemon->mode = 0; /* default mode = 0 (station) */
|
|
|
|
while ((option = getopt(argc, argv, "a:")) != ERROR)
|
|
{
|
|
switch (option)
|
|
{
|
|
case 'a':
|
|
_daemon->mode = 1; /* ap mode */
|
|
_daemon->ch = 1;
|
|
|
|
if (5 == argc)
|
|
{
|
|
_daemon->ch = (int)atoi(optarg);
|
|
}
|
|
|
|
ap_mode = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((ap_mode && (4 != argc) && (5 != argc))
|
|
|| (!ap_mode && 3 != argc))
|
|
{
|
|
_show_usage(argv[0]);
|
|
ret = ERROR;
|
|
goto errout;
|
|
}
|
|
|
|
_daemon->ssid = argv[argc - 2];
|
|
_daemon->key = argv[argc - 1];
|
|
|
|
ret = gs2200m_loop(_daemon);
|
|
|
|
errout:
|
|
|
|
if (_daemon)
|
|
{
|
|
free(_daemon);
|
|
_daemon = NULL;
|
|
}
|
|
|
|
return ret;
|
|
}
|