395 lines
11 KiB
C
395 lines
11 KiB
C
|
/****************************************************************************
|
||
|
* apps/lte/alt1250/alt1250_usockif.c
|
||
|
*
|
||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||
|
* contributor license agreements. See the NOTICE file distributed with
|
||
|
* this work for additional information regarding copyright ownership. The
|
||
|
* ASF licenses this file to you 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.
|
||
|
*
|
||
|
****************************************************************************/
|
||
|
|
||
|
/****************************************************************************
|
||
|
* Included Files
|
||
|
****************************************************************************/
|
||
|
|
||
|
#include <nuttx/config.h>
|
||
|
|
||
|
#include <unistd.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <sys/ioctl.h>
|
||
|
#include <errno.h>
|
||
|
#include <netinet/in.h>
|
||
|
#include <assert.h>
|
||
|
|
||
|
#include "alt1250_dbg.h"
|
||
|
#include "alt1250_usockif.h"
|
||
|
|
||
|
/****************************************************************************
|
||
|
* Private Functions
|
||
|
****************************************************************************/
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: write_to_usock
|
||
|
****************************************************************************/
|
||
|
|
||
|
static int write_to_usock(int fd, FAR void *buf, size_t sz)
|
||
|
{
|
||
|
int ret;
|
||
|
|
||
|
ret = write(fd, buf, sz);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
ret = -errno;
|
||
|
}
|
||
|
else if (ret != sz)
|
||
|
{
|
||
|
ret = -ENOSPC;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ret = OK;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: send_dataack
|
||
|
****************************************************************************/
|
||
|
|
||
|
static int send_dataack(int fd, uint32_t ackxid, int32_t ackresult,
|
||
|
uint16_t valuelen, uint16_t valuelen_nontrunc,
|
||
|
FAR uint8_t *value_ptr, FAR uint8_t *buf_ptr)
|
||
|
{
|
||
|
int ret;
|
||
|
struct usrsock_message_datareq_ack_s dataack;
|
||
|
|
||
|
dataack.reqack.head.msgid = USRSOCK_MESSAGE_RESPONSE_DATA_ACK;
|
||
|
dataack.reqack.head.flags = 0;
|
||
|
dataack.reqack.xid = ackxid;
|
||
|
dataack.reqack.result = ackresult;
|
||
|
|
||
|
dataack.valuelen = valuelen;
|
||
|
dataack.valuelen_nontrunc = valuelen_nontrunc;
|
||
|
|
||
|
ret = write_to_usock(fd, &dataack, sizeof(dataack));
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
if ((valuelen > 0) && (value_ptr != NULL))
|
||
|
{
|
||
|
ret = write_to_usock(fd, value_ptr, valuelen);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
return ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ((ackresult > 0) && (buf_ptr != NULL))
|
||
|
{
|
||
|
ret = write_to_usock(fd, buf_ptr, ackresult);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
return ret;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return OK;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: read_wrapper
|
||
|
****************************************************************************/
|
||
|
|
||
|
static int read_wrapper(int fd, FAR void *buf, size_t sz)
|
||
|
{
|
||
|
int ret;
|
||
|
|
||
|
ret = read(fd, buf, sz);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
ret = -errno;
|
||
|
dbg_alt1250("failed to read usersock: %d\n", errno);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
if (ret != sz)
|
||
|
{
|
||
|
dbg_alt1250("unexpected read size: %d expected: %u\n", ret, sz);
|
||
|
return -EMSGSIZE;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* Public Functions
|
||
|
****************************************************************************/
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: init_usock_device
|
||
|
****************************************************************************/
|
||
|
|
||
|
int init_usock_device(void)
|
||
|
{
|
||
|
return open(DEV_USRSOCK, O_RDWR);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: finalize_usock_device
|
||
|
****************************************************************************/
|
||
|
|
||
|
void finalize_usock_device(int fd)
|
||
|
{
|
||
|
close(fd);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: reset_usock_device
|
||
|
****************************************************************************/
|
||
|
|
||
|
int reset_usock_device(int fd)
|
||
|
{
|
||
|
close(fd);
|
||
|
return open(DEV_USRSOCK, O_RDWR);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_readrequest
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_readrequest(int fd, FAR struct usrsock_request_buff_s *buf)
|
||
|
{
|
||
|
int ret;
|
||
|
int rsize = 0;
|
||
|
FAR char *pbuf = NULL;
|
||
|
|
||
|
/* Read common header */
|
||
|
|
||
|
ret = read_wrapper(fd, buf, sizeof(struct usrsock_request_common_s));
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/* Read each request */
|
||
|
|
||
|
switch (buf->request.head.reqid)
|
||
|
{
|
||
|
case USRSOCK_REQUEST_SOCKET:
|
||
|
rsize = sizeof(struct usrsock_request_socket_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_CLOSE:
|
||
|
rsize = sizeof(struct usrsock_request_close_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_CONNECT:
|
||
|
rsize = sizeof(struct usrsock_request_connect_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_SENDTO:
|
||
|
rsize = sizeof(struct usrsock_request_sendto_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_RECVFROM:
|
||
|
rsize = sizeof(struct usrsock_request_recvfrom_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_SETSOCKOPT:
|
||
|
rsize = sizeof(struct usrsock_request_setsockopt_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_GETSOCKOPT:
|
||
|
rsize = sizeof(struct usrsock_request_getsockopt_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_GETSOCKNAME:
|
||
|
rsize = sizeof(struct usrsock_request_getsockname_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_GETPEERNAME:
|
||
|
rsize = sizeof(struct usrsock_request_getpeername_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_BIND:
|
||
|
rsize = sizeof(struct usrsock_request_bind_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_LISTEN:
|
||
|
rsize = sizeof(struct usrsock_request_listen_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_ACCEPT:
|
||
|
rsize = sizeof(struct usrsock_request_accept_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_IOCTL:
|
||
|
rsize = sizeof(struct usrsock_request_ioctl_s);
|
||
|
break;
|
||
|
case USRSOCK_REQUEST_SHUTDOWN:
|
||
|
rsize = sizeof(struct usrsock_request_shutdown_s);
|
||
|
break;
|
||
|
default:
|
||
|
dbg_alt1250("unexpected request id: %d\n", buf->request.head.reqid);
|
||
|
return -EINVAL;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
pbuf = (FAR char *)buf + sizeof(struct usrsock_request_common_s);
|
||
|
rsize -= sizeof(struct usrsock_request_common_s);
|
||
|
|
||
|
ret = read_wrapper(fd, pbuf, rsize);
|
||
|
if (ret < 0)
|
||
|
{
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
return sizeof(struct usrsock_request_common_s) + ret;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_readreqioctl
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_readreqioctl(int fd, FAR struct usrsock_request_buff_s *buf)
|
||
|
{
|
||
|
FAR struct usrsock_request_ioctl_s *req = &buf->request.ioctl_req;
|
||
|
FAR void *pbuf = &buf->req_ioctl;
|
||
|
int rsize;
|
||
|
|
||
|
switch (req->cmd)
|
||
|
{
|
||
|
case FIONBIO:
|
||
|
rsize = sizeof(int);
|
||
|
break;
|
||
|
case SIOCLTECMD:
|
||
|
rsize = sizeof(struct lte_ioctl_data_s);
|
||
|
break;
|
||
|
case SIOCSIFFLAGS:
|
||
|
rsize = sizeof(struct ifreq);
|
||
|
break;
|
||
|
case SIOCDENYINETSOCK:
|
||
|
rsize = sizeof(uint8_t);
|
||
|
break;
|
||
|
case SIOCSMSENSTREP:
|
||
|
case SIOCSMSGREFID:
|
||
|
case SIOCSMSSSCA:
|
||
|
rsize = sizeof(struct lte_smsreq_s);
|
||
|
break;
|
||
|
default:
|
||
|
dbg_alt1250("Unsupported command:0x%08lx\n", req->cmd);
|
||
|
return -EINVAL;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (req->arglen != rsize)
|
||
|
{
|
||
|
dbg_alt1250("unexpected size: %d, expect: %d\n", req->arglen, rsize);
|
||
|
ASSERT(0);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
return read_wrapper(fd, pbuf, rsize);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_readreqaddr
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_readreqaddr(int fd, FAR struct sockaddr_storage *addr,
|
||
|
size_t sz)
|
||
|
{
|
||
|
if ((sz != sizeof(struct sockaddr_in)) &&
|
||
|
(sz != sizeof(struct sockaddr_in6)))
|
||
|
{
|
||
|
dbg_alt1250("Invalid addrlen: %u\n", sz);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
return read_wrapper(fd, addr, sz);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_readreqsendbuf
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_readreqsendbuf(int fd, FAR uint8_t *sendbuf, size_t sz)
|
||
|
{
|
||
|
return read_wrapper(fd, sendbuf, sz);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_readreqoption
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_readreqoption(int fd, FAR uint8_t *option, size_t sz)
|
||
|
{
|
||
|
return read_wrapper(fd, option, sz);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_discard
|
||
|
****************************************************************************/
|
||
|
|
||
|
void usockif_discard(int fd, size_t sz)
|
||
|
{
|
||
|
char dummy;
|
||
|
|
||
|
/* If the seek is called with the exact size, the seek will
|
||
|
* result in an error. In order to avoid this, the process of
|
||
|
* read is performed after seeking the specified size minus one byte.
|
||
|
*/
|
||
|
|
||
|
if (lseek(fd, sz - 1, SEEK_CUR) >= 0)
|
||
|
{
|
||
|
read(fd, &dummy, 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_sendack
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_sendack(int fd, int32_t usock_result, uint32_t usock_xid,
|
||
|
bool inprogress)
|
||
|
{
|
||
|
struct usrsock_message_req_ack_s ack;
|
||
|
|
||
|
ack.head.msgid = USRSOCK_MESSAGE_RESPONSE_ACK;
|
||
|
ack.head.flags =
|
||
|
(inprogress == true) ? USRSOCK_MESSAGE_FLAG_REQ_IN_PROGRESS: 0;
|
||
|
ack.xid = usock_xid;
|
||
|
ack.result = usock_result;
|
||
|
|
||
|
return write_to_usock(fd, &ack, sizeof(ack));
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_senddataack
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_senddataack(int fd, int32_t usock_result, uint32_t usock_xid,
|
||
|
FAR struct usock_ackinfo_s *ackinfo)
|
||
|
{
|
||
|
return send_dataack(fd, usock_xid, usock_result, ackinfo->valuelen,
|
||
|
ackinfo->valuelen_nontrunc, ackinfo->value_ptr,
|
||
|
ackinfo->buf_ptr);
|
||
|
}
|
||
|
|
||
|
/****************************************************************************
|
||
|
* name: usockif_sendevent
|
||
|
****************************************************************************/
|
||
|
|
||
|
int usockif_sendevent(int fd, int usockid, int event)
|
||
|
{
|
||
|
struct usrsock_message_socket_event_s msg;
|
||
|
|
||
|
msg.head.msgid = USRSOCK_MESSAGE_SOCKET_EVENT;
|
||
|
msg.head.flags = USRSOCK_MESSAGE_FLAG_EVENT;
|
||
|
msg.usockid = usockid;
|
||
|
msg.head.events = event;
|
||
|
|
||
|
return write_to_usock(fd, &msg, sizeof(msg));
|
||
|
}
|