mbed-os/connectivity/cellular/source/framework/AT/AT_CellularStack.cpp

423 lines
12 KiB
C++
Raw Normal View History

2018-02-09 11:24:27 +00:00
/*
* Copyright (c) 2017, Arm Limited and affiliates.
* SPDX-License-Identifier: Apache-2.0
*
* 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 "AT_CellularStack.h"
#include "CellularUtil.h"
#include "CellularLog.h"
2020-07-19 17:02:32 +00:00
#include "rtos/ThisThread.h"
#include "AT_CellularDevice.h"
2018-02-09 11:24:27 +00:00
using namespace mbed_cellular_util;
using namespace mbed;
AT_CellularStack::AT_CellularStack(ATHandler &at, int cid, nsapi_ip_stack_t stack_type, AT_CellularDevice &device) :
_socket(NULL), _cid(cid),
_stack_type(stack_type), _ip_ver_sendto(NSAPI_UNSPEC), _at(at), _device(device)
2018-02-09 11:24:27 +00:00
{
2018-07-27 12:33:20 +00:00
memset(_ip, 0, PDP_IPV6_SIZE);
MBED_ASSERT(_device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT) > 0);
_socket = new CellularSocket *[_device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT)]();
2018-02-09 11:24:27 +00:00
}
AT_CellularStack::~AT_CellularStack()
{
for (int i = 0; i < _device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT); i++) {
if (_socket[i]) {
delete _socket[i];
2018-02-09 11:24:27 +00:00
}
}
delete [] _socket;
2018-02-09 11:24:27 +00:00
}
int AT_CellularStack::find_socket_index(nsapi_socket_t handle)
{
if (!_socket) {
return -1;
}
for (int i = 0; i < _device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT); i++) {
if (_socket[i] == handle) {
return i;
}
}
return -1;
}
2018-02-09 11:24:27 +00:00
/** NetworkStack
*/
nsapi_error_t AT_CellularStack::get_ip_address(SocketAddress *address)
{
if (!address) {
return NSAPI_ERROR_PARAMETER;
}
_at.lock();
bool ipv4 = false, ipv6 = false;
_at.cmd_start_stop("+CGPADDR", "=", "%d", _cid);
_at.resp_start("+CGPADDR:");
if (_at.info_resp()) {
_at.skip_param();
if (_at.read_string(_ip, PDP_IPV6_SIZE) != -1) {
convert_ipv6(_ip);
address->set_ip_address(_ip);
ipv4 = (address->get_ip_version() == NSAPI_IPv4);
ipv6 = (address->get_ip_version() == NSAPI_IPv6);
// Try to look for second address ONLY if modem has support for dual stack(can handle both IPv4 and IPv6 simultaneously).
// Otherwise assumption is that second address is not reliable, even if network provides one.
2019-11-29 11:00:05 +00:00
if ((_device.get_property(AT_CellularDevice::PROPERTY_IPV4V6_PDP_TYPE) && (_at.read_string(_ip, PDP_IPV6_SIZE) != -1))) {
convert_ipv6(_ip);
address->set_ip_address(_ip);
ipv6 = (address->get_ip_version() == NSAPI_IPv6);
}
}
}
_at.resp_stop();
_at.unlock();
if (ipv4 && ipv6) {
_stack_type = IPV4V6_STACK;
} else if (ipv4) {
_stack_type = IPV4_STACK;
} else if (ipv6) {
_stack_type = IPV6_STACK;
}
return (ipv4 || ipv6) ? NSAPI_ERROR_OK : NSAPI_ERROR_NO_ADDRESS;
}
void AT_CellularStack::set_cid(int cid)
{
_cid = cid;
}
2018-02-09 11:24:27 +00:00
nsapi_error_t AT_CellularStack::socket_open(nsapi_socket_t *handle, nsapi_protocol_t proto)
{
if (!handle) {
2018-02-09 11:24:27 +00:00
return NSAPI_ERROR_UNSUPPORTED;
}
if (proto == NSAPI_UDP) {
if (!_device.get_property(AT_CellularDevice::PROPERTY_IP_UDP)) {
return NSAPI_ERROR_UNSUPPORTED;
}
} else if (proto == NSAPI_TCP) {
if (!_device.get_property(AT_CellularDevice::PROPERTY_IP_TCP)) {
return NSAPI_ERROR_UNSUPPORTED;
}
} else {
2018-02-09 11:24:27 +00:00
return NSAPI_ERROR_UNSUPPORTED;
}
_socket_mutex.lock();
int index = find_socket_index(0);
2018-02-09 11:24:27 +00:00
if (index == -1) {
tr_warn("No free sockets!");
_socket_mutex.unlock();
2018-02-09 11:24:27 +00:00
return NSAPI_ERROR_NO_SOCKET;
}
tr_info("Socket %d open", index);
2018-02-09 11:24:27 +00:00
// create local socket structure, socket on modem is created when app calls sendto/recvfrom
// Do not assign a socket ID yet. Socket is not created at the Modem yet.
// create_socket_impl(handle) will assign the correct socket ID.
2018-02-09 11:24:27 +00:00
_socket[index] = new CellularSocket;
2019-03-04 11:47:15 +00:00
CellularSocket *psock = _socket[index];
2018-02-09 11:24:27 +00:00
SocketAddress addr(0, get_dynamic_ip_port());
2018-02-09 11:24:27 +00:00
psock->localAddress = addr;
psock->proto = proto;
*handle = psock;
_socket_mutex.unlock();
2018-02-09 11:24:27 +00:00
return NSAPI_ERROR_OK;
}
nsapi_error_t AT_CellularStack::socket_close(nsapi_socket_t handle)
{
int err = NSAPI_ERROR_DEVICE_ERROR;
struct CellularSocket *socket = (struct CellularSocket *)handle;
2018-07-27 12:33:20 +00:00
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
2018-02-12 14:44:36 +00:00
}
2018-02-09 11:24:27 +00:00
int sock_id = socket->id;
2018-03-28 13:09:46 +00:00
int index = find_socket_index(handle);
2018-02-12 14:44:36 +00:00
if (index == -1) {
2018-03-28 13:09:46 +00:00
tr_error("No socket found to be closed");
2018-02-12 14:44:36 +00:00
return err;
2018-02-09 11:24:27 +00:00
}
2018-03-28 13:09:46 +00:00
2018-02-12 14:44:36 +00:00
err = NSAPI_ERROR_OK;
2018-02-09 11:24:27 +00:00
2018-03-28 13:09:46 +00:00
// Close the socket on the modem if it was created
2018-02-09 11:24:27 +00:00
_at.lock();
if (sock_id > -1) {
2018-03-28 13:09:46 +00:00
err = socket_close_impl(sock_id);
}
if (!err) {
tr_info("Socket %d closed", index);
} else {
tr_info("Socket %d close (id %d, started %d, error %d)", index, sock_id, socket->started, err);
}
_socket_mutex.lock();
_socket[index] = NULL;
delete socket;
_socket_mutex.unlock();
2018-02-09 11:24:27 +00:00
_at.unlock();
return err;
}
nsapi_error_t AT_CellularStack::socket_bind(nsapi_socket_t handle, const SocketAddress &addr)
{
struct CellularSocket *socket = (CellularSocket *)handle;
2018-02-12 14:44:36 +00:00
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
2018-02-12 14:44:36 +00:00
}
2018-02-09 11:24:27 +00:00
if (addr) {
return NSAPI_ERROR_UNSUPPORTED;
}
2018-02-09 11:24:27 +00:00
_at.lock();
uint16_t port = addr.get_port();
if (port != socket->localAddress.get_port()) {
if (port && (get_socket_index_by_port(port) == -1)) {
socket->localAddress.set_port(port);
} else {
_at.unlock();
return NSAPI_ERROR_PARAMETER;
}
}
if (socket->id == -1) {
2018-02-09 11:24:27 +00:00
create_socket_impl(socket);
}
return _at.unlock_return_error();
}
nsapi_error_t AT_CellularStack::socket_listen(nsapi_socket_t handle, int backlog)
{
return NSAPI_ERROR_UNSUPPORTED;
2018-02-09 11:24:27 +00:00
}
nsapi_error_t AT_CellularStack::socket_connect(nsapi_socket_t handle, const SocketAddress &addr)
{
CellularSocket *socket = (CellularSocket *)handle;
2018-02-12 14:44:36 +00:00
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
2018-02-12 14:44:36 +00:00
}
2018-02-09 11:24:27 +00:00
socket->remoteAddress = addr;
socket->connected = true;
return NSAPI_ERROR_OK;
}
nsapi_error_t AT_CellularStack::socket_accept(void *server, void **socket, SocketAddress *addr)
{
return NSAPI_ERROR_UNSUPPORTED;
2018-02-09 11:24:27 +00:00
}
nsapi_size_or_error_t AT_CellularStack::socket_send(nsapi_socket_t handle, const void *data, unsigned size)
{
CellularSocket *socket = (CellularSocket *)handle;
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
}
if (!socket->connected) {
return NSAPI_ERROR_NO_CONNECTION;
2018-02-09 11:24:27 +00:00
}
return socket_sendto(handle, socket->remoteAddress, data, size);
}
nsapi_size_or_error_t AT_CellularStack::socket_sendto(nsapi_socket_t handle, const SocketAddress &addr, const void *data, unsigned size)
{
CellularSocket *socket = (CellularSocket *)handle;
2018-02-12 14:44:36 +00:00
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
2018-02-12 14:44:36 +00:00
}
2018-02-09 11:24:27 +00:00
if (socket->closed && !socket->pending_bytes) {
tr_info("sendto socket %d closed", socket->id);
return NSAPI_ERROR_NO_CONNECTION;
}
if (size == 0) {
if (socket->proto == NSAPI_UDP) {
return NSAPI_ERROR_UNSUPPORTED;
} else if (socket->proto == NSAPI_TCP) {
return 0;
}
}
2018-02-09 11:24:27 +00:00
nsapi_size_or_error_t ret_val = NSAPI_ERROR_OK;
if (socket->id == -1) {
/* Check that stack type supports sendto address type*/
if (!is_addr_stack_compatible(addr)) {
return NSAPI_ERROR_PARAMETER;
}
_ip_ver_sendto = addr.get_ip_version();
2018-02-09 11:24:27 +00:00
_at.lock();
ret_val = create_socket_impl(socket);
_at.unlock();
if (ret_val != NSAPI_ERROR_OK) {
tr_error("Socket %d create %s error %d", find_socket_index(socket), addr.get_ip_address(), ret_val);
2018-02-09 11:24:27 +00:00
return ret_val;
}
}
/* Check parameters - sendto address is valid and stack type supports sending to that address type*/
if (!is_addr_stack_compatible(addr)) {
return NSAPI_ERROR_PARAMETER;
2018-02-09 11:24:27 +00:00
}
_at.lock();
ret_val = socket_sendto_impl(socket, addr, data, size);
2018-07-27 12:33:20 +00:00
_at.unlock();
if (ret_val >= 0) {
tr_info("Socket %d sent %d bytes to %s port %d", find_socket_index(socket), ret_val, addr.get_ip_address(), addr.get_port());
} else if (ret_val != NSAPI_ERROR_WOULD_BLOCK) {
tr_error("Socket %d sendto %s error %d", find_socket_index(socket), addr.get_ip_address(), ret_val);
}
2018-02-09 11:24:27 +00:00
return ret_val;
}
nsapi_size_or_error_t AT_CellularStack::socket_recv(nsapi_socket_t handle, void *data, unsigned size)
{
return socket_recvfrom(handle, NULL, data, size);
}
nsapi_size_or_error_t AT_CellularStack::socket_recvfrom(nsapi_socket_t handle, SocketAddress *addr, void *buffer, unsigned size)
{
CellularSocket *socket = (CellularSocket *)handle;
2018-02-12 14:44:36 +00:00
if (!socket) {
return NSAPI_ERROR_NO_SOCKET;
2018-02-12 14:44:36 +00:00
}
2018-02-09 11:24:27 +00:00
if (socket->closed) {
tr_info("recvfrom socket %d closed", socket->id);
return 0;
}
2018-02-09 11:24:27 +00:00
nsapi_size_or_error_t ret_val = NSAPI_ERROR_OK;
if (socket->id == -1) {
2018-02-09 11:24:27 +00:00
_at.lock();
ret_val = create_socket_impl(socket);
_at.unlock();
if (ret_val != NSAPI_ERROR_OK) {
tr_error("Socket %d create error %d", find_socket_index(socket), ret_val);
2018-02-09 11:24:27 +00:00
return ret_val;
}
}
_at.lock();
ret_val = socket_recvfrom_impl(socket, addr, buffer, size);
_at.unlock();
if (socket->closed) {
tr_info("recvfrom socket %d closed", socket->id);
return 0;
}
if (ret_val >= 0) {
2018-12-11 08:50:14 +00:00
if (addr) {
tr_info("Socket %d recv %d bytes from %s port %d", find_socket_index(socket), ret_val, addr->get_ip_address(), addr->get_port());
2018-12-11 08:50:14 +00:00
} else {
tr_info("Socket %d recv %d bytes", find_socket_index(socket), ret_val);
}
} else if (ret_val != NSAPI_ERROR_WOULD_BLOCK) {
tr_error("Socket %d recv error %d", find_socket_index(socket), ret_val);
}
2018-02-09 11:24:27 +00:00
return ret_val;
}
void AT_CellularStack::socket_attach(nsapi_socket_t handle, void (*callback)(void *), void *data)
{
CellularSocket *socket = (CellularSocket *)handle;
2018-02-12 14:44:36 +00:00
if (!socket) {
return;
}
2018-02-09 11:24:27 +00:00
socket->_cb = callback;
socket->_data = data;
}
int AT_CellularStack::get_socket_index_by_port(uint16_t port)
{
for (int i = 0; i < _device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT); i++) {
if (_socket[i] != 0) {
if (_socket[i]->localAddress.get_port() == port) {
return i;
}
}
}
return -1;
}
AT_CellularStack::CellularSocket *AT_CellularStack::find_socket(int sock_id)
{
CellularSocket *sock = NULL;
for (int i = 0; i < _device.get_property(AT_CellularDevice::PROPERTY_SOCKET_COUNT); i++) {
if (_socket[i] && _socket[i]->id == sock_id) {
sock = _socket[i];
break;
}
}
if (!sock) {
tr_error("Socket not found %d", sock_id);
}
return sock;
}
bool AT_CellularStack::is_addr_stack_compatible(const SocketAddress &addr)
{
if ((addr.get_ip_version() == NSAPI_UNSPEC) ||
(addr.get_ip_version() == NSAPI_IPv4 && _stack_type == IPV6_STACK) ||
(addr.get_ip_version() == NSAPI_IPv6 && _stack_type == IPV4_STACK)) {
return false;
}
return true;
}