2016-05-18 11:50:48 +02:00
|
|
|
/** @file
|
|
|
|
* @brief Misc network utility functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Intel Corporation
|
|
|
|
*
|
2017-01-19 02:01:01 +01:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2016-05-18 11:50:48 +02:00
|
|
|
*/
|
|
|
|
|
2018-11-30 11:54:56 +01:00
|
|
|
#include <logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(net_utils, CONFIG_NET_UTILS_LOG_LEVEL);
|
2016-05-19 11:33:42 +02:00
|
|
|
|
2019-06-12 13:06:52 +02:00
|
|
|
#include <kernel.h>
|
2016-05-18 11:50:48 +02:00
|
|
|
#include <stdlib.h>
|
2019-06-12 13:06:52 +02:00
|
|
|
#include <syscall_handler.h>
|
Introduce new sized integer typedefs
This is a start to move away from the C99 {u}int{8,16,32,64}_t types to
Zephyr defined u{8,16,32,64}_t and s{8,16,32,64}_t. This allows Zephyr
to define the sized types in a consistent manor across all the
architectures we support and not conflict with what various compilers
and libc might do with regards to the C99 types.
We introduce <zephyr/types.h> as part of this and have it include
<stdint.h> for now until we transition all the code away from the C99
types.
We go with u{8,16,32,64}_t and s{8,16,32,64}_t as there are some
existing variables defined u8 & u16 as well as to be consistent with
Zephyr naming conventions.
Jira: ZEP-2051
Change-Id: I451fed0623b029d65866622e478225dfab2c0ca8
Signed-off-by: Kumar Gala <kumar.gala@linaro.org>
2017-04-19 17:32:08 +02:00
|
|
|
#include <zephyr/types.h>
|
2016-05-18 11:50:48 +02:00
|
|
|
#include <stdbool.h>
|
2016-08-24 13:57:55 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
2016-05-19 11:31:36 +02:00
|
|
|
#include <net/net_ip.h>
|
2017-04-03 17:14:35 +02:00
|
|
|
#include <net/net_pkt.h>
|
2016-05-19 11:31:36 +02:00
|
|
|
#include <net/net_core.h>
|
2019-02-06 14:16:50 +01:00
|
|
|
#include <net/socket_can.h>
|
2016-05-18 11:50:48 +02:00
|
|
|
|
2018-08-16 13:23:55 +02:00
|
|
|
char *net_sprint_addr(sa_family_t af, const void *addr)
|
|
|
|
{
|
|
|
|
#define NBUFS 3
|
|
|
|
static char buf[NBUFS][NET_IPV6_ADDR_LEN];
|
|
|
|
static int i;
|
|
|
|
char *s = buf[++i % NBUFS];
|
|
|
|
|
|
|
|
return net_addr_ntop(af, addr, s, NET_IPV6_ADDR_LEN);
|
|
|
|
}
|
|
|
|
|
2019-02-06 14:16:50 +01:00
|
|
|
const char *net_proto2str(int family, int proto)
|
2017-03-13 16:45:22 +01:00
|
|
|
{
|
2019-02-06 14:16:50 +01:00
|
|
|
if (family == AF_INET || family == AF_INET6) {
|
|
|
|
switch (proto) {
|
|
|
|
case IPPROTO_ICMP:
|
|
|
|
return "ICMPv4";
|
|
|
|
case IPPROTO_TCP:
|
|
|
|
return "TCP";
|
|
|
|
case IPPROTO_UDP:
|
|
|
|
return "UDP";
|
|
|
|
case IPPROTO_ICMPV6:
|
|
|
|
return "ICMPv6";
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (family == AF_CAN) {
|
|
|
|
switch (proto) {
|
|
|
|
case CAN_RAW:
|
|
|
|
return "CAN_RAW";
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2017-03-13 16:45:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return "UNK_PROTO";
|
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
char *net_byte_to_hex(char *ptr, u8_t byte, char base, bool pad)
|
2016-05-18 11:50:48 +02:00
|
|
|
{
|
|
|
|
int i, val;
|
|
|
|
|
|
|
|
for (i = 0, val = (byte & 0xf0) >> 4; i < 2; i++, val = byte & 0x0f) {
|
|
|
|
if (i == 0 && !pad && !val) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (val < 10) {
|
|
|
|
*ptr++ = (char) (val + '0');
|
|
|
|
} else {
|
|
|
|
*ptr++ = (char) (val - 10 + base);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*ptr = '\0';
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
char *net_sprint_ll_addr_buf(const u8_t *ll, u8_t ll_len,
|
2016-05-18 11:50:48 +02:00
|
|
|
char *buf, int buflen)
|
|
|
|
{
|
2017-04-21 16:27:50 +02:00
|
|
|
u8_t i, len, blen;
|
2016-05-18 11:50:48 +02:00
|
|
|
char *ptr = buf;
|
|
|
|
|
|
|
|
switch (ll_len) {
|
|
|
|
case 8:
|
2018-11-29 20:23:03 +01:00
|
|
|
len = 8U;
|
2016-05-18 11:50:48 +02:00
|
|
|
break;
|
|
|
|
case 6:
|
2018-11-29 20:23:03 +01:00
|
|
|
len = 6U;
|
2016-05-18 11:50:48 +02:00
|
|
|
break;
|
2019-08-06 16:12:24 +02:00
|
|
|
case 2:
|
|
|
|
len = 2U;
|
|
|
|
break;
|
2016-05-18 11:50:48 +02:00
|
|
|
default:
|
2018-11-29 20:23:03 +01:00
|
|
|
len = 6U;
|
2016-05-18 11:50:48 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U, blen = buflen; i < len && blen > 0; i++) {
|
2017-02-17 10:37:26 +01:00
|
|
|
ptr = net_byte_to_hex(ptr, (char)ll[i], 'A', true);
|
2016-05-18 11:50:48 +02:00
|
|
|
*ptr++ = ':';
|
2019-03-27 02:57:45 +01:00
|
|
|
blen -= 3U;
|
2016-05-18 11:50:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!(ptr - buf)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*(ptr - 1) = '\0';
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
static int net_value_to_udec(char *buf, u32_t value, int precision)
|
2016-05-18 11:50:48 +02:00
|
|
|
{
|
2017-04-21 16:27:50 +02:00
|
|
|
u32_t divisor;
|
2016-05-18 11:50:48 +02:00
|
|
|
int i;
|
|
|
|
int temp;
|
|
|
|
char *start = buf;
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
divisor = 1000000000U;
|
2019-06-04 16:52:23 +02:00
|
|
|
if (precision < 0) {
|
2016-05-18 11:50:48 +02:00
|
|
|
precision = 1;
|
2019-06-04 16:52:23 +02:00
|
|
|
}
|
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
for (i = 9; i >= 0; i--, divisor /= 10U) {
|
2016-05-18 11:50:48 +02:00
|
|
|
temp = value / divisor;
|
|
|
|
value = value % divisor;
|
|
|
|
if ((precision > i) || (temp != 0)) {
|
|
|
|
precision = i;
|
|
|
|
*buf++ = (char) (temp + '0');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*buf = 0;
|
|
|
|
|
|
|
|
return buf - start;
|
|
|
|
}
|
|
|
|
|
2019-06-12 13:06:52 +02:00
|
|
|
char *z_impl_net_addr_ntop(sa_family_t family, const void *src,
|
|
|
|
char *dst, size_t size)
|
2016-05-18 11:50:48 +02:00
|
|
|
{
|
2017-01-30 11:30:13 +01:00
|
|
|
struct in_addr *addr;
|
|
|
|
struct in6_addr *addr6;
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t *w;
|
2018-11-29 20:23:03 +01:00
|
|
|
u8_t i, bl, bh, longest = 1U;
|
2017-04-21 16:27:50 +02:00
|
|
|
s8_t pos = -1;
|
2016-05-18 11:50:48 +02:00
|
|
|
char delim = ':';
|
|
|
|
unsigned char zeros[8] = { 0 };
|
2017-01-30 11:30:13 +01:00
|
|
|
char *ptr = dst;
|
2016-05-18 11:50:48 +02:00
|
|
|
int len = -1;
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t value;
|
2016-05-18 11:50:48 +02:00
|
|
|
bool needcolon = false;
|
|
|
|
|
2017-01-30 11:30:13 +01:00
|
|
|
if (family == AF_INET6) {
|
|
|
|
addr6 = (struct in6_addr *)src;
|
2017-04-21 16:27:50 +02:00
|
|
|
w = (u16_t *)addr6->s6_addr16;
|
2016-05-18 11:50:48 +02:00
|
|
|
len = 8;
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U; i < 8; i++) {
|
2017-04-21 16:27:50 +02:00
|
|
|
u8_t j;
|
2016-05-18 11:50:48 +02:00
|
|
|
|
|
|
|
for (j = i; j < 8; j++) {
|
2017-04-10 16:17:36 +02:00
|
|
|
if (UNALIGNED_GET(&w[j]) != 0) {
|
2016-05-18 11:50:48 +02:00
|
|
|
break;
|
|
|
|
}
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
zeros[i]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U; i < 8; i++) {
|
2016-05-18 11:50:48 +02:00
|
|
|
if (zeros[i] > longest) {
|
|
|
|
longest = zeros[i];
|
|
|
|
pos = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
if (longest == 1U) {
|
2016-05-18 11:50:48 +02:00
|
|
|
pos = -1;
|
|
|
|
}
|
|
|
|
|
2017-01-30 11:30:13 +01:00
|
|
|
} else if (family == AF_INET) {
|
|
|
|
addr = (struct in_addr *)src;
|
2016-05-18 11:50:48 +02:00
|
|
|
len = 4;
|
|
|
|
delim = '.';
|
2017-01-30 11:30:13 +01:00
|
|
|
} else {
|
2016-05-18 11:50:48 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U; i < len; i++) {
|
2016-05-18 11:50:48 +02:00
|
|
|
/* IPv4 address a.b.c.d */
|
|
|
|
if (len == 4) {
|
2017-04-21 16:27:50 +02:00
|
|
|
u8_t l;
|
2016-05-18 11:50:48 +02:00
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
value = (u32_t)addr->s4_addr[i];
|
2016-05-18 11:50:48 +02:00
|
|
|
|
|
|
|
/* net_byte_to_udec() eats 0 */
|
2019-03-27 02:57:45 +01:00
|
|
|
if (value == 0U) {
|
2016-05-18 11:50:48 +02:00
|
|
|
*ptr++ = '0';
|
|
|
|
*ptr++ = delim;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
l = net_value_to_udec(ptr, value, 0);
|
|
|
|
|
|
|
|
ptr += l;
|
|
|
|
*ptr++ = delim;
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* IPv6 address */
|
|
|
|
if (i == pos) {
|
2019-03-27 02:57:45 +01:00
|
|
|
if (needcolon || i == 0U) {
|
2016-05-18 11:50:48 +02:00
|
|
|
*ptr++ = ':';
|
|
|
|
}
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
*ptr++ = ':';
|
|
|
|
needcolon = false;
|
2019-03-27 02:57:45 +01:00
|
|
|
i += longest - 1U;
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (needcolon) {
|
|
|
|
*ptr++ = ':';
|
|
|
|
needcolon = false;
|
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
value = (u32_t)sys_be16_to_cpu(UNALIGNED_GET(&w[i]));
|
2016-05-18 11:50:48 +02:00
|
|
|
bh = value >> 8;
|
|
|
|
bl = value & 0xff;
|
|
|
|
|
|
|
|
if (bh) {
|
|
|
|
if (bh > 0x0f) {
|
|
|
|
ptr = net_byte_to_hex(ptr, bh, 'a', false);
|
|
|
|
} else {
|
|
|
|
if (bh < 10) {
|
|
|
|
*ptr++ = (char)(bh + '0');
|
|
|
|
} else {
|
|
|
|
*ptr++ = (char) (bh - 10 + 'a');
|
|
|
|
}
|
|
|
|
}
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
ptr = net_byte_to_hex(ptr, bl, 'a', true);
|
|
|
|
} else if (bl > 0x0f) {
|
|
|
|
ptr = net_byte_to_hex(ptr, bl, 'a', false);
|
|
|
|
} else {
|
|
|
|
if (bl < 10) {
|
|
|
|
*ptr++ = (char)(bl + '0');
|
|
|
|
} else {
|
|
|
|
*ptr++ = (char) (bl - 10 + 'a');
|
|
|
|
}
|
|
|
|
}
|
2017-01-30 11:30:13 +01:00
|
|
|
|
2016-05-18 11:50:48 +02:00
|
|
|
needcolon = true;
|
|
|
|
}
|
|
|
|
|
2017-01-30 11:30:13 +01:00
|
|
|
if (!(ptr - dst)) {
|
2016-05-18 11:50:48 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-01-30 11:30:13 +01:00
|
|
|
if (family == AF_INET) {
|
2016-05-18 11:50:48 +02:00
|
|
|
*(ptr - 1) = '\0';
|
|
|
|
} else {
|
|
|
|
*ptr = '\0';
|
|
|
|
}
|
|
|
|
|
2017-01-30 11:30:13 +01:00
|
|
|
return dst;
|
2016-05-18 11:50:48 +02:00
|
|
|
}
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2019-06-12 13:06:52 +02:00
|
|
|
#if defined(CONFIG_USERSPACE)
|
userspace: Support for split 64 bit arguments
System call arguments, at the arch layer, are single words. So
passing wider values requires splitting them into two registers at
call time. This gets even more complicated for values (e.g
k_timeout_t) that may have different sizes depending on configuration.
This patch adds a feature to gen_syscalls.py to detect functions with
wide arguments and automatically generates code to split/unsplit them.
Unfortunately the current scheme of Z_SYSCALL_DECLARE_* macros won't
work with functions like this, because for N arguments (our current
maximum N is 10) there are 2^N possible configurations of argument
widths. So this generates the complete functions for each handler and
wrapper, effectively doing in python what was originally done in the
preprocessor.
Another complexity is that traditional the z_hdlr_*() function for a
system call has taken the raw list of word arguments, which does not
work when some of those arguments must be 64 bit types. So instead of
using a single Z_SYSCALL_HANDLER macro, this splits the job of
z_hdlr_*() into two steps: An automatically-generated unmarshalling
function, z_mrsh_*(), which then calls a user-supplied verification
function z_vrfy_*(). The verification function is typesafe, and is a
simple C function with exactly the same argument and return signature
as the syscall impl function. It is also not responsible for
validating the pointers to the extra parameter array or a wide return
value, that code gets automatically generated.
This commit includes new vrfy/msrh handling for all syscalls invoked
during CI runs. Future commits will port the less testable code.
Signed-off-by: Andy Ross <andrew.j.ross@intel.com>
2019-08-06 22:34:31 +02:00
|
|
|
char *z_vrfy_net_addr_ntop(sa_family_t family, const void *src,
|
|
|
|
char *dst, size_t size)
|
2019-06-12 13:06:52 +02:00
|
|
|
{
|
|
|
|
char str[INET6_ADDRSTRLEN];
|
|
|
|
struct in6_addr addr6;
|
|
|
|
struct in_addr addr4;
|
|
|
|
char *out;
|
|
|
|
const void *addr;
|
|
|
|
|
|
|
|
Z_OOPS(Z_SYSCALL_MEMORY_WRITE(dst, size));
|
|
|
|
|
|
|
|
if (family == AF_INET) {
|
|
|
|
Z_OOPS(z_user_from_copy(&addr4, (const void *)src,
|
|
|
|
sizeof(addr4)));
|
|
|
|
addr = &addr4;
|
|
|
|
} else if (family == AF_INET6) {
|
|
|
|
Z_OOPS(z_user_from_copy(&addr6, (const void *)src,
|
|
|
|
sizeof(addr6)));
|
|
|
|
addr = &addr6;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
out = z_impl_net_addr_ntop(family, addr, str, sizeof(str));
|
|
|
|
if (!out) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Z_OOPS(z_user_to_copy((void *)dst, str, MIN(size, sizeof(str))));
|
|
|
|
|
userspace: Support for split 64 bit arguments
System call arguments, at the arch layer, are single words. So
passing wider values requires splitting them into two registers at
call time. This gets even more complicated for values (e.g
k_timeout_t) that may have different sizes depending on configuration.
This patch adds a feature to gen_syscalls.py to detect functions with
wide arguments and automatically generates code to split/unsplit them.
Unfortunately the current scheme of Z_SYSCALL_DECLARE_* macros won't
work with functions like this, because for N arguments (our current
maximum N is 10) there are 2^N possible configurations of argument
widths. So this generates the complete functions for each handler and
wrapper, effectively doing in python what was originally done in the
preprocessor.
Another complexity is that traditional the z_hdlr_*() function for a
system call has taken the raw list of word arguments, which does not
work when some of those arguments must be 64 bit types. So instead of
using a single Z_SYSCALL_HANDLER macro, this splits the job of
z_hdlr_*() into two steps: An automatically-generated unmarshalling
function, z_mrsh_*(), which then calls a user-supplied verification
function z_vrfy_*(). The verification function is typesafe, and is a
simple C function with exactly the same argument and return signature
as the syscall impl function. It is also not responsible for
validating the pointers to the extra parameter array or a wide return
value, that code gets automatically generated.
This commit includes new vrfy/msrh handling for all syscalls invoked
during CI runs. Future commits will port the less testable code.
Signed-off-by: Andy Ross <andrew.j.ross@intel.com>
2019-08-06 22:34:31 +02:00
|
|
|
return dst;
|
2019-06-12 13:06:52 +02:00
|
|
|
}
|
userspace: Support for split 64 bit arguments
System call arguments, at the arch layer, are single words. So
passing wider values requires splitting them into two registers at
call time. This gets even more complicated for values (e.g
k_timeout_t) that may have different sizes depending on configuration.
This patch adds a feature to gen_syscalls.py to detect functions with
wide arguments and automatically generates code to split/unsplit them.
Unfortunately the current scheme of Z_SYSCALL_DECLARE_* macros won't
work with functions like this, because for N arguments (our current
maximum N is 10) there are 2^N possible configurations of argument
widths. So this generates the complete functions for each handler and
wrapper, effectively doing in python what was originally done in the
preprocessor.
Another complexity is that traditional the z_hdlr_*() function for a
system call has taken the raw list of word arguments, which does not
work when some of those arguments must be 64 bit types. So instead of
using a single Z_SYSCALL_HANDLER macro, this splits the job of
z_hdlr_*() into two steps: An automatically-generated unmarshalling
function, z_mrsh_*(), which then calls a user-supplied verification
function z_vrfy_*(). The verification function is typesafe, and is a
simple C function with exactly the same argument and return signature
as the syscall impl function. It is also not responsible for
validating the pointers to the extra parameter array or a wide return
value, that code gets automatically generated.
This commit includes new vrfy/msrh handling for all syscalls invoked
during CI runs. Future commits will port the less testable code.
Signed-off-by: Andy Ross <andrew.j.ross@intel.com>
2019-08-06 22:34:31 +02:00
|
|
|
#include <syscalls/net_addr_ntop_mrsh.c>
|
2019-06-12 13:06:52 +02:00
|
|
|
#endif /* CONFIG_USERSPACE */
|
|
|
|
|
|
|
|
int z_impl_net_addr_pton(sa_family_t family, const char *src,
|
|
|
|
void *dst)
|
2016-08-24 13:57:55 +02:00
|
|
|
{
|
|
|
|
if (family == AF_INET) {
|
|
|
|
struct in_addr *addr = (struct in_addr *)dst;
|
2016-12-20 20:18:16 +01:00
|
|
|
size_t i, len;
|
2016-08-24 13:57:55 +02:00
|
|
|
|
|
|
|
len = strlen(src);
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!(src[i] >= '0' && src[i] <= '9') &&
|
|
|
|
src[i] != '.') {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-12 04:09:03 +02:00
|
|
|
(void)memset(addr, 0, sizeof(struct in_addr));
|
2016-08-24 13:57:55 +02:00
|
|
|
|
2016-11-12 20:16:21 +01:00
|
|
|
for (i = 0; i < sizeof(struct in_addr); i++) {
|
|
|
|
char *endptr;
|
|
|
|
|
|
|
|
addr->s4_addr[i] = strtol(src, &endptr, 10);
|
|
|
|
|
|
|
|
src = ++endptr;
|
2016-08-24 13:57:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} else if (family == AF_INET6) {
|
|
|
|
/* If the string contains a '.', it means it's of the form
|
|
|
|
* X:X:X:X:X:X:x.x.x.x, and contains only 6 16-bit pieces
|
|
|
|
*/
|
|
|
|
int expected_groups = strchr(src, '.') ? 6 : 8;
|
|
|
|
struct in6_addr *addr = (struct in6_addr *)dst;
|
|
|
|
int i, len;
|
|
|
|
|
|
|
|
if (*src == ':') {
|
|
|
|
/* Ignore a leading colon, makes parsing neater */
|
|
|
|
src++;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = strlen(src);
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!(src[i] >= '0' && src[i] <= '9') &&
|
|
|
|
!(src[i] >= 'A' && src[i] <= 'F') &&
|
|
|
|
!(src[i] >= 'a' && src[i] <= 'f') &&
|
2019-06-04 16:52:23 +02:00
|
|
|
src[i] != '.' && src[i] != ':') {
|
2016-08-24 13:57:55 +02:00
|
|
|
return -EINVAL;
|
2019-06-04 16:52:23 +02:00
|
|
|
}
|
2016-08-24 13:57:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < expected_groups; i++) {
|
|
|
|
char *tmp;
|
|
|
|
|
|
|
|
if (!src || *src == '\0') {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*src != ':') {
|
|
|
|
/* Normal IPv6 16-bit piece */
|
2017-04-10 16:13:35 +02:00
|
|
|
UNALIGNED_PUT(htons(strtol(src, NULL, 16)),
|
|
|
|
&addr->s6_addr16[i]);
|
2016-08-24 13:57:55 +02:00
|
|
|
src = strchr(src, ':');
|
2018-07-18 15:54:10 +02:00
|
|
|
if (src) {
|
|
|
|
src++;
|
|
|
|
} else {
|
|
|
|
if (i < expected_groups - 1) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2016-08-24 13:57:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Two colons in a row */
|
|
|
|
|
|
|
|
for (; i < expected_groups; i++) {
|
2017-04-10 16:13:35 +02:00
|
|
|
UNALIGNED_PUT(0, &addr->s6_addr16[i]);
|
2016-08-24 13:57:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
tmp = strrchr(src, ':');
|
|
|
|
if (src == tmp && (expected_groups == 6 || !src[1])) {
|
|
|
|
src++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expected_groups == 6) {
|
|
|
|
/* we need to drop the trailing
|
|
|
|
* colon since it's between the
|
|
|
|
* ipv6 and ipv4 addresses, rather than being
|
|
|
|
* a part of the ipv6 address
|
|
|
|
*/
|
|
|
|
tmp--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate the amount of skipped zeros */
|
|
|
|
i = expected_groups - 1;
|
|
|
|
do {
|
|
|
|
if (*tmp == ':') {
|
|
|
|
i--;
|
|
|
|
}
|
2018-07-03 04:27:29 +02:00
|
|
|
|
|
|
|
if (i < 0) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2016-08-24 13:57:55 +02:00
|
|
|
} while (tmp-- != src);
|
|
|
|
|
|
|
|
src++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expected_groups == 6) {
|
|
|
|
/* Parse the IPv4 part */
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
if (!src || !*src) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
addr->s6_addr[12 + i] = strtol(src, NULL, 10);
|
|
|
|
|
|
|
|
src = strchr(src, '.');
|
2018-07-18 15:54:10 +02:00
|
|
|
if (src) {
|
|
|
|
src++;
|
|
|
|
} else {
|
|
|
|
if (i < 3) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2016-08-24 13:57:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-12 13:06:52 +02:00
|
|
|
#if defined(CONFIG_USERSPACE)
|
userspace: Support for split 64 bit arguments
System call arguments, at the arch layer, are single words. So
passing wider values requires splitting them into two registers at
call time. This gets even more complicated for values (e.g
k_timeout_t) that may have different sizes depending on configuration.
This patch adds a feature to gen_syscalls.py to detect functions with
wide arguments and automatically generates code to split/unsplit them.
Unfortunately the current scheme of Z_SYSCALL_DECLARE_* macros won't
work with functions like this, because for N arguments (our current
maximum N is 10) there are 2^N possible configurations of argument
widths. So this generates the complete functions for each handler and
wrapper, effectively doing in python what was originally done in the
preprocessor.
Another complexity is that traditional the z_hdlr_*() function for a
system call has taken the raw list of word arguments, which does not
work when some of those arguments must be 64 bit types. So instead of
using a single Z_SYSCALL_HANDLER macro, this splits the job of
z_hdlr_*() into two steps: An automatically-generated unmarshalling
function, z_mrsh_*(), which then calls a user-supplied verification
function z_vrfy_*(). The verification function is typesafe, and is a
simple C function with exactly the same argument and return signature
as the syscall impl function. It is also not responsible for
validating the pointers to the extra parameter array or a wide return
value, that code gets automatically generated.
This commit includes new vrfy/msrh handling for all syscalls invoked
during CI runs. Future commits will port the less testable code.
Signed-off-by: Andy Ross <andrew.j.ross@intel.com>
2019-08-06 22:34:31 +02:00
|
|
|
int z_vrfy_net_addr_pton(sa_family_t family, const char *src,
|
|
|
|
void *dst)
|
2019-06-12 13:06:52 +02:00
|
|
|
{
|
|
|
|
char str[INET6_ADDRSTRLEN];
|
|
|
|
struct in6_addr addr6;
|
|
|
|
struct in_addr addr4;
|
|
|
|
void *addr;
|
|
|
|
size_t size;
|
|
|
|
size_t nlen;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
if (family == AF_INET) {
|
|
|
|
size = sizeof(struct in_addr);
|
|
|
|
addr = &addr4;
|
|
|
|
} else if (family == AF_INET6) {
|
|
|
|
size = sizeof(struct in6_addr);
|
|
|
|
addr = &addr6;
|
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(str, 0, sizeof(str));
|
|
|
|
|
|
|
|
nlen = z_user_string_nlen((const char *)src, sizeof(str), &err);
|
|
|
|
if (err) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
Z_OOPS(Z_SYSCALL_MEMORY_WRITE(dst, size));
|
|
|
|
Z_OOPS(Z_SYSCALL_MEMORY_READ(src, nlen));
|
|
|
|
Z_OOPS(z_user_from_copy(str, (const void *)src, nlen));
|
|
|
|
|
|
|
|
err = z_impl_net_addr_pton(family, str, addr);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
Z_OOPS(z_user_to_copy((void *)src, addr, size));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
userspace: Support for split 64 bit arguments
System call arguments, at the arch layer, are single words. So
passing wider values requires splitting them into two registers at
call time. This gets even more complicated for values (e.g
k_timeout_t) that may have different sizes depending on configuration.
This patch adds a feature to gen_syscalls.py to detect functions with
wide arguments and automatically generates code to split/unsplit them.
Unfortunately the current scheme of Z_SYSCALL_DECLARE_* macros won't
work with functions like this, because for N arguments (our current
maximum N is 10) there are 2^N possible configurations of argument
widths. So this generates the complete functions for each handler and
wrapper, effectively doing in python what was originally done in the
preprocessor.
Another complexity is that traditional the z_hdlr_*() function for a
system call has taken the raw list of word arguments, which does not
work when some of those arguments must be 64 bit types. So instead of
using a single Z_SYSCALL_HANDLER macro, this splits the job of
z_hdlr_*() into two steps: An automatically-generated unmarshalling
function, z_mrsh_*(), which then calls a user-supplied verification
function z_vrfy_*(). The verification function is typesafe, and is a
simple C function with exactly the same argument and return signature
as the syscall impl function. It is also not responsible for
validating the pointers to the extra parameter array or a wide return
value, that code gets automatically generated.
This commit includes new vrfy/msrh handling for all syscalls invoked
during CI runs. Future commits will port the less testable code.
Signed-off-by: Andy Ross <andrew.j.ross@intel.com>
2019-08-06 22:34:31 +02:00
|
|
|
#include <syscalls/net_addr_pton_mrsh.c>
|
2019-06-12 13:06:52 +02:00
|
|
|
#endif /* CONFIG_USERSPACE */
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
static u16_t calc_chksum(u16_t sum, const u8_t *data, size_t len)
|
2016-05-19 11:31:36 +02:00
|
|
|
{
|
2017-04-21 16:27:50 +02:00
|
|
|
const u8_t *end;
|
2018-12-03 20:35:16 +01:00
|
|
|
u16_t tmp;
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
end = data + len - 1;
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
while (data < end) {
|
|
|
|
tmp = (data[0] << 8) + data[1];
|
2016-05-19 11:31:36 +02:00
|
|
|
sum += tmp;
|
|
|
|
if (sum < tmp) {
|
|
|
|
sum++;
|
|
|
|
}
|
2018-12-03 20:35:16 +01:00
|
|
|
|
|
|
|
data += 2;
|
2016-05-19 11:31:36 +02:00
|
|
|
}
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
if (data == end) {
|
|
|
|
tmp = data[0] << 8;
|
2016-05-19 11:31:36 +02:00
|
|
|
sum += tmp;
|
|
|
|
if (sum < tmp) {
|
|
|
|
sum++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return sum;
|
|
|
|
}
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
static inline u16_t pkt_calc_chksum(struct net_pkt *pkt, u16_t sum)
|
2016-05-19 11:31:36 +02:00
|
|
|
{
|
2018-12-03 20:35:16 +01:00
|
|
|
struct net_pkt_cursor *cur = &pkt->cursor;
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
if (!cur->buf || !cur->pos) {
|
|
|
|
return sum;
|
2016-05-19 11:31:36 +02:00
|
|
|
}
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
len = cur->buf->len - (cur->pos - cur->buf->data);
|
2017-06-28 12:01:12 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
while (cur->buf) {
|
|
|
|
sum = calc_chksum(sum, cur->pos, len);
|
2017-06-28 12:01:12 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
cur->buf = cur->buf->frags;
|
|
|
|
if (!cur->buf || !cur->buf->len) {
|
2016-05-19 11:31:36 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
cur->pos = cur->buf->data;
|
2016-05-19 11:31:36 +02:00
|
|
|
|
|
|
|
if (len % 2) {
|
2018-12-03 20:35:16 +01:00
|
|
|
sum += *cur->pos;
|
|
|
|
if (sum < *cur->pos) {
|
2016-05-19 11:31:36 +02:00
|
|
|
sum++;
|
|
|
|
}
|
2018-12-03 20:35:16 +01:00
|
|
|
|
|
|
|
cur->pos++;
|
|
|
|
len = cur->buf->len - 1;
|
2016-05-19 11:31:36 +02:00
|
|
|
} else {
|
2018-12-03 20:35:16 +01:00
|
|
|
len = cur->buf->len;
|
2016-05-19 11:31:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return sum;
|
|
|
|
}
|
|
|
|
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t net_calc_chksum(struct net_pkt *pkt, u8_t proto)
|
2016-05-19 11:31:36 +02:00
|
|
|
{
|
2018-12-03 20:35:16 +01:00
|
|
|
size_t len = 0U;
|
2018-11-29 20:23:03 +01:00
|
|
|
u16_t sum = 0U;
|
2018-12-03 20:35:16 +01:00
|
|
|
struct net_pkt_cursor backup;
|
2018-12-05 13:58:58 +01:00
|
|
|
bool ow;
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
if (IS_ENABLED(CONFIG_NET_IPV4) &&
|
|
|
|
net_pkt_family(pkt) == AF_INET) {
|
2017-06-28 23:14:45 +02:00
|
|
|
if (proto != IPPROTO_ICMP) {
|
2018-12-03 20:35:16 +01:00
|
|
|
len = 2 * sizeof(struct in_addr);
|
|
|
|
sum = net_pkt_get_len(pkt) -
|
|
|
|
net_pkt_ip_hdr_len(pkt) + proto;
|
2016-05-24 16:19:00 +02:00
|
|
|
}
|
2018-12-03 20:35:16 +01:00
|
|
|
} else if (IS_ENABLED(CONFIG_NET_IPV6) &&
|
|
|
|
net_pkt_family(pkt) == AF_INET6) {
|
|
|
|
len = 2 * sizeof(struct in6_addr);
|
|
|
|
sum = net_pkt_get_len(pkt) -
|
|
|
|
net_pkt_ip_hdr_len(pkt) -
|
|
|
|
net_pkt_ipv6_ext_len(pkt) + proto;
|
|
|
|
} else {
|
2017-04-05 08:37:44 +02:00
|
|
|
NET_DBG("Unknown protocol family %d", net_pkt_family(pkt));
|
2016-05-19 11:31:36 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-02-01 21:19:14 +01:00
|
|
|
net_pkt_cursor_backup(pkt, &backup);
|
|
|
|
net_pkt_cursor_init(pkt);
|
|
|
|
|
2018-12-05 13:58:58 +01:00
|
|
|
ow = net_pkt_is_being_overwritten(pkt);
|
|
|
|
net_pkt_set_overwrite(pkt, true);
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
net_pkt_skip(pkt, net_pkt_ip_hdr_len(pkt) - len);
|
|
|
|
|
|
|
|
sum = calc_chksum(sum, pkt->cursor.pos, len);
|
|
|
|
|
|
|
|
net_pkt_skip(pkt, len + net_pkt_ipv6_ext_len(pkt));
|
|
|
|
|
|
|
|
sum = pkt_calc_chksum(pkt, sum);
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
sum = (sum == 0U) ? 0xffff : htons(sum);
|
2016-05-19 11:31:36 +02:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
net_pkt_cursor_restore(pkt, &backup);
|
|
|
|
|
2018-12-05 13:58:58 +01:00
|
|
|
net_pkt_set_overwrite(pkt, ow);
|
|
|
|
|
2018-12-04 09:55:31 +01:00
|
|
|
return ~sum;
|
2016-05-19 11:31:36 +02:00
|
|
|
}
|
2016-09-09 15:06:28 +02:00
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV4)
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t net_calc_chksum_ipv4(struct net_pkt *pkt)
|
2016-09-09 15:06:28 +02:00
|
|
|
{
|
2017-04-21 16:27:50 +02:00
|
|
|
u16_t sum;
|
2016-09-09 15:06:28 +02:00
|
|
|
|
2019-03-08 16:47:59 +01:00
|
|
|
sum = calc_chksum(0, pkt->buffer->data, net_pkt_ip_hdr_len(pkt));
|
2016-09-09 15:06:28 +02:00
|
|
|
|
2019-03-27 02:57:45 +01:00
|
|
|
sum = (sum == 0U) ? 0xffff : htons(sum);
|
2016-09-09 15:06:28 +02:00
|
|
|
|
2018-12-04 09:55:31 +01:00
|
|
|
return ~sum;
|
2016-09-09 15:06:28 +02:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV4 */
|
2017-07-04 11:50:08 +02:00
|
|
|
|
2017-08-20 20:55:23 +02:00
|
|
|
#if defined(CONFIG_NET_IPV6) || defined(CONFIG_NET_IPV4)
|
|
|
|
static bool convert_port(const char *buf, u16_t *port)
|
|
|
|
{
|
|
|
|
unsigned long tmp;
|
|
|
|
char *endptr;
|
|
|
|
|
|
|
|
tmp = strtoul(buf, &endptr, 10);
|
|
|
|
if ((endptr == buf && tmp == 0) ||
|
|
|
|
!(*buf != '\0' && *endptr == '\0') ||
|
|
|
|
((unsigned long)(unsigned short)tmp != tmp)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*port = tmp;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV6 || CONFIG_NET_IPV4 */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6)
|
|
|
|
static bool parse_ipv6(const char *str, size_t str_len,
|
|
|
|
struct sockaddr *addr, bool has_port)
|
|
|
|
{
|
|
|
|
char *ptr = NULL;
|
|
|
|
struct in6_addr *addr6;
|
|
|
|
char ipaddr[INET6_ADDRSTRLEN + 1];
|
|
|
|
int end, len, ret, i;
|
|
|
|
u16_t port;
|
|
|
|
|
2019-02-11 18:14:19 +01:00
|
|
|
len = MIN(INET6_ADDRSTRLEN, str_len);
|
2017-08-20 20:55:23 +02:00
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!str[i]) {
|
|
|
|
len = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_port) {
|
|
|
|
/* IPv6 address with port number */
|
|
|
|
ptr = memchr(str, ']', len);
|
|
|
|
if (!ptr) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-11 18:14:19 +01:00
|
|
|
end = MIN(len, ptr - (str + 1));
|
2017-08-20 20:55:23 +02:00
|
|
|
memcpy(ipaddr, str + 1, end);
|
|
|
|
} else {
|
|
|
|
end = len;
|
|
|
|
memcpy(ipaddr, str, end);
|
|
|
|
}
|
|
|
|
|
|
|
|
ipaddr[end] = '\0';
|
|
|
|
|
|
|
|
addr6 = &net_sin6(addr)->sin6_addr;
|
|
|
|
|
|
|
|
ret = net_addr_pton(AF_INET6, ipaddr, addr6);
|
|
|
|
if (ret < 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_sin6(addr)->sin6_family = AF_INET6;
|
|
|
|
|
|
|
|
if (!has_port) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((ptr + 1) < (str + str_len) && *(ptr + 1) == ':') {
|
2019-08-25 14:54:09 +02:00
|
|
|
/* -1 as end does not contain first [
|
|
|
|
* -2 as pointer is advanced by 2, skipping ]:
|
|
|
|
*/
|
|
|
|
len = str_len - end - 1 - 2;
|
|
|
|
|
|
|
|
ptr += 2;
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!ptr[i]) {
|
|
|
|
len = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-08-20 20:55:23 +02:00
|
|
|
|
|
|
|
/* Re-use the ipaddr buf for port conversion */
|
2019-08-25 14:54:09 +02:00
|
|
|
memcpy(ipaddr, ptr, len);
|
2017-08-20 20:55:23 +02:00
|
|
|
ipaddr[len] = '\0';
|
|
|
|
|
|
|
|
ret = convert_port(ipaddr, &port);
|
|
|
|
if (!ret) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_sin6(addr)->sin6_port = htons(port);
|
|
|
|
|
|
|
|
NET_DBG("IPv6 host %s port %d",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_addr_ntop(AF_INET6, addr6,
|
|
|
|
ipaddr, sizeof(ipaddr) - 1)),
|
2017-08-20 20:55:23 +02:00
|
|
|
port);
|
|
|
|
} else {
|
|
|
|
NET_DBG("IPv6 host %s",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_addr_ntop(AF_INET6, addr6,
|
|
|
|
ipaddr, sizeof(ipaddr) - 1)));
|
2017-08-20 20:55:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2019-02-07 14:00:44 +01:00
|
|
|
#else
|
|
|
|
static inline bool parse_ipv6(const char *str, size_t str_len,
|
|
|
|
struct sockaddr *addr, bool has_port)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2017-08-20 20:55:23 +02:00
|
|
|
#endif /* CONFIG_NET_IPV6 */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV4)
|
|
|
|
static bool parse_ipv4(const char *str, size_t str_len,
|
|
|
|
struct sockaddr *addr, bool has_port)
|
|
|
|
{
|
|
|
|
char *ptr = NULL;
|
|
|
|
char ipaddr[NET_IPV4_ADDR_LEN + 1];
|
|
|
|
struct in_addr *addr4;
|
|
|
|
int end, len, ret, i;
|
|
|
|
u16_t port;
|
|
|
|
|
2019-02-11 18:14:19 +01:00
|
|
|
len = MIN(NET_IPV4_ADDR_LEN, str_len);
|
2017-08-20 20:55:23 +02:00
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
if (!str[i]) {
|
|
|
|
len = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_port) {
|
|
|
|
/* IPv4 address with port number */
|
|
|
|
ptr = memchr(str, ':', len);
|
|
|
|
if (!ptr) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-11 18:14:19 +01:00
|
|
|
end = MIN(len, ptr - str);
|
2017-08-20 20:55:23 +02:00
|
|
|
} else {
|
|
|
|
end = len;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(ipaddr, str, end);
|
|
|
|
ipaddr[end] = '\0';
|
|
|
|
|
|
|
|
addr4 = &net_sin(addr)->sin_addr;
|
|
|
|
|
|
|
|
ret = net_addr_pton(AF_INET, ipaddr, addr4);
|
|
|
|
if (ret < 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_sin(addr)->sin_family = AF_INET;
|
|
|
|
|
|
|
|
if (!has_port) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(ipaddr, ptr + 1, str_len - end);
|
|
|
|
ipaddr[str_len - end] = '\0';
|
|
|
|
|
|
|
|
ret = convert_port(ipaddr, &port);
|
|
|
|
if (!ret) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_sin(addr)->sin_port = htons(port);
|
|
|
|
|
|
|
|
NET_DBG("IPv4 host %s port %d",
|
2018-10-02 13:57:55 +02:00
|
|
|
log_strdup(net_addr_ntop(AF_INET, addr4,
|
|
|
|
ipaddr, sizeof(ipaddr) - 1)),
|
2017-08-20 20:55:23 +02:00
|
|
|
port);
|
|
|
|
return true;
|
|
|
|
}
|
2019-02-07 14:00:44 +01:00
|
|
|
#else
|
|
|
|
static inline bool parse_ipv4(const char *str, size_t str_len,
|
|
|
|
struct sockaddr *addr, bool has_port)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2017-08-20 20:55:23 +02:00
|
|
|
#endif /* CONFIG_NET_IPV4 */
|
|
|
|
|
|
|
|
bool net_ipaddr_parse(const char *str, size_t str_len, struct sockaddr *addr)
|
|
|
|
{
|
|
|
|
int i, count;
|
|
|
|
|
2017-09-14 09:40:06 +02:00
|
|
|
if (!str || str_len == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We cannot accept empty string here */
|
|
|
|
if (*str == '\0') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-20 20:55:23 +02:00
|
|
|
if (*str == '[') {
|
|
|
|
return parse_ipv6(str, str_len, addr, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (count = i = 0; str[i] && i < str_len; i++) {
|
|
|
|
if (str[i] == ':') {
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count == 1) {
|
|
|
|
return parse_ipv4(str, str_len, addr, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV4) && defined(CONFIG_NET_IPV6)
|
|
|
|
if (!parse_ipv4(str, str_len, addr, false)) {
|
|
|
|
return parse_ipv6(str, str_len, addr, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV4) && !defined(CONFIG_NET_IPV6)
|
|
|
|
return parse_ipv4(str, str_len, addr, false);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6) && !defined(CONFIG_NET_IPV4)
|
|
|
|
return parse_ipv6(str, str_len, addr, false);
|
|
|
|
#endif
|
2019-02-07 14:00:44 +01:00
|
|
|
return false;
|
2017-08-20 20:55:23 +02:00
|
|
|
}
|
2018-02-23 11:22:00 +01:00
|
|
|
|
|
|
|
int net_bytes_from_str(u8_t *buf, int buf_len, const char *src)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
char *endptr;
|
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U; i < strlen(src); i++) {
|
2018-02-23 11:22:00 +01:00
|
|
|
if (!(src[i] >= '0' && src[i] <= '9') &&
|
|
|
|
!(src[i] >= 'A' && src[i] <= 'F') &&
|
|
|
|
!(src[i] >= 'a' && src[i] <= 'f') &&
|
|
|
|
src[i] != ':') {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-12 04:09:03 +02:00
|
|
|
(void)memset(buf, 0, buf_len);
|
2018-02-23 11:22:00 +01:00
|
|
|
|
2018-11-29 20:23:03 +01:00
|
|
|
for (i = 0U; i < buf_len; i++) {
|
2018-02-23 11:22:00 +01:00
|
|
|
buf[i] = strtol(src, &endptr, 16);
|
|
|
|
src = ++endptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2019-04-14 13:30:46 +02:00
|
|
|
|
|
|
|
const char *net_family2str(sa_family_t family)
|
|
|
|
{
|
|
|
|
switch (family) {
|
|
|
|
case AF_UNSPEC:
|
|
|
|
return "AF_UNSPEC";
|
|
|
|
case AF_INET:
|
|
|
|
return "AF_INET";
|
|
|
|
case AF_INET6:
|
|
|
|
return "AF_INET6";
|
|
|
|
case AF_PACKET:
|
|
|
|
return "AF_PACKET";
|
|
|
|
case AF_CAN:
|
|
|
|
return "AF_CAN";
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
2019-08-09 13:49:35 +02:00
|
|
|
|
|
|
|
const struct in_addr *net_ipv4_unspecified_address(void)
|
|
|
|
{
|
|
|
|
static const struct in_addr addr;
|
|
|
|
|
|
|
|
return &addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct in_addr *net_ipv4_broadcast_address(void)
|
|
|
|
{
|
|
|
|
static const struct in_addr addr = { { { 255, 255, 255, 255 } } };
|
|
|
|
|
|
|
|
return &addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* IPv6 wildcard and loopback address defined by RFC2553 */
|
|
|
|
const struct in6_addr in6addr_any = IN6ADDR_ANY_INIT;
|
|
|
|
const struct in6_addr in6addr_loopback = IN6ADDR_LOOPBACK_INIT;
|
|
|
|
|
|
|
|
const struct in6_addr *net_ipv6_unspecified_address(void)
|
|
|
|
{
|
|
|
|
return &in6addr_any;
|
|
|
|
}
|