diff --git a/.idea/.gitignore b/.idea/.gitignore
new file mode 100644
index 000000000..13566b81b
--- /dev/null
+++ b/.idea/.gitignore
@@ -0,0 +1,8 @@
+# Default ignored files
+/shelf/
+/workspace.xml
+# Editor-based HTTP Client requests
+/httpRequests/
+# Datasource local storage ignored files
+/dataSources/
+/dataSources.local.xml
diff --git a/.idea/modules.xml b/.idea/modules.xml
new file mode 100644
index 000000000..d1d6d71f1
--- /dev/null
+++ b/.idea/modules.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/vcs.xml b/.idea/vcs.xml
new file mode 100644
index 000000000..94a25f7f4
--- /dev/null
+++ b/.idea/vcs.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/xiuos.iml b/.idea/xiuos.iml
new file mode 100644
index 000000000..bc2cd8740
--- /dev/null
+++ b/.idea/xiuos.iml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Kconfig b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Kconfig
new file mode 100755
index 000000000..b2e70eee0
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Kconfig
@@ -0,0 +1,41 @@
+menuconfig XiZi_USING_NETDEV
+ bool "Enable network interface device"
+ default y
+
+if RT_USING_NETDEV
+
+ config NETDEV_USING_IFCONFIG
+ bool "Enable ifconfig features"
+ default y
+
+ config NETDEV_USING_PING
+ bool "Enable ping features"
+ default y
+
+ config NETDEV_USING_NETSTAT
+ bool "Enable netstat features"
+ default y
+
+ config NETDEV_USING_AUTO_DEFAULT
+ bool "Enable default netdev automatic change features"
+ default y
+
+ config NETDEV_USING_IPV6
+ bool "Enable IPV6 protocol support"
+ default n
+
+ config NETDEV_IPV4
+ int
+ default 1
+
+ config NETDEV_IPV6
+ int
+ default 1 if NETDEV_USING_IPV6
+ default 0 if !NETDEV_USING_IPV6
+
+ config NETDEV_IPV6_SCOPES
+ bool
+ help
+ Defined to synchronize the ip6_addr structure state
+ default n
+endif
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Makefile b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Makefile
new file mode 100755
index 000000000..a7ad646c3
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/Makefile
@@ -0,0 +1,4 @@
+SRC_DIR := src
+
+
+include $(KERNEL_ROOT)/compiler.mk
\ No newline at end of file
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/SConscript b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/SConscript
new file mode 100755
index 000000000..ec1d2b290
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/SConscript
@@ -0,0 +1,13 @@
+# RT-Thread building script for component
+
+from building import *
+
+cwd = GetCurrentDir()
+
+src = Glob('src/*.c')
+
+CPPPATH = [cwd + '/include']
+
+group = DefineGroup('SAL', src, depend = ['RT_USING_NETDEV'], CPPPATH = CPPPATH)
+
+Return('group')
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/arpa/inet.h b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/arpa/inet.h
new file mode 100755
index 000000000..ba439e282
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/arpa/inet.h
@@ -0,0 +1,16 @@
+/*
+ * Copyright (c) 2006-2021, RT-Thread Development Team
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Change Logs:
+ * Date Author Notes
+ * 2022-8-22 zhaoyun for XiZi
+ */
+
+#ifndef __INET_H__
+#define __INET_H__
+
+#include
+
+#endif
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev.h b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev.h
new file mode 100755
index 000000000..5dbc26e38
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev.h
@@ -0,0 +1,214 @@
+/*
+ * Copyright (c) 2006-2021, RT-Thread Development Team
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Change Logs:
+ * Date Author Notes
+ * 2022-8-22 zhaoyun for XiZi
+ */
+
+#ifndef __NETDEV_H__
+#define __NETDEV_H__
+
+
+#include
+#include "netdev_ipaddr.h"
+
+#include "sys_arch.h"
+#include
+#include "lwip/sys.h"
+
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/* the maximum of all used hardware address lengths */
+#ifndef NETDEV_HWADDR_MAX_LEN
+#define NETDEV_HWADDR_MAX_LEN 8U
+#endif
+
+/* the maximum of dns server number supported */
+#ifndef NETDEV_DNS_SERVERS_NUM
+#define NETDEV_DNS_SERVERS_NUM 2U
+#endif
+
+#if NETDEV_IPV6
+/* the maximum of dns server number supported */
+#ifndef NETDEV_IPV6_NUM_ADDRESSES
+#define NETDEV_IPV6_NUM_ADDRESSES 3U
+#endif
+#endif /* NETDEV_IPV6 */
+
+/* whether the network interface device is 'up' (set by the network interface driver or application) */
+#define NETDEV_FLAG_UP 0x01U
+/* if set, the network interface device has broadcast capability, only supported in the 'lwIP' stack */
+#define NETDEV_FLAG_BROADCAST 0x02U
+/* if set, the network interface device has an active link (set by the network interface driver) */
+#define NETDEV_FLAG_LINK_UP 0x04U
+/* if set, the network interface device is an ethernet device using ARP, only supported in the 'lwIP' stack */
+#define NETDEV_FLAG_ETHARP 0x08U
+/* if set, the network interface device is an ethernet device, only supported in the 'lwIP' stack */
+#define NETDEV_FLAG_ETHERNET 0x10U
+/* if set, the network interface device has IGMP capability, only supported in the 'lwIP' stack */
+#define NETDEV_FLAG_IGMP 0x20U
+/* if set, the network interface device has MLD6 capability, only supported in the 'lwIP' stack */
+#define NETDEV_FLAG_MLD6 0x40U
+/* if set, the network interface device connected to internet successfully (set by the network interface driver) */
+#define NETDEV_FLAG_INTERNET_UP 0x80U
+/* if set, the network interface device has DHCP capability (set by the network interface device driver or application) */
+#define NETDEV_FLAG_DHCP 0x100U
+
+enum netdev_cb_type
+{
+ NETDEV_CB_ADDR_IP, /* IP address */
+ NETDEV_CB_ADDR_NETMASK, /* subnet mask */
+ NETDEV_CB_ADDR_GATEWAY, /* netmask */
+ NETDEV_CB_ADDR_DNS_SERVER, /* dns server */
+ NETDEV_CB_STATUS_UP, /* changed to 'up' */
+ NETDEV_CB_STATUS_DOWN, /* changed to 'down' */
+ NETDEV_CB_STATUS_LINK_UP, /* changed to 'link up' */
+ NETDEV_CB_STATUS_LINK_DOWN, /* changed to 'link down' */
+ NETDEV_CB_STATUS_INTERNET_UP, /* changed to 'internet up' */
+ NETDEV_CB_STATUS_INTERNET_DOWN, /* changed to 'internet down' */
+ NETDEV_CB_STATUS_DHCP_ENABLE, /* enable DHCP capability */
+ NETDEV_CB_STATUS_DHCP_DISABLE, /* disable DHCP capability */
+};
+
+struct netdev;
+
+/* function prototype for network interface device status or address change callback functions */
+typedef void (*netdev_callback_fn )(struct netdev *netdev, enum netdev_cb_type type);
+
+struct netdev_ops;
+
+/* network interface device object */
+struct xizi_slist_node
+{
+ struct xizi_slist_node *next; /**< point to next node. */
+};
+
+
+typedef struct xizi_slist_node xizi_slist_t; /**< Type for single list. */
+
+
+#define XiZi_NAME_MAX 8
+struct netdev
+{
+ xizi_slist_t list;//
+
+ char name[XiZi_NAME_MAX]; /* network interface device name */
+ ip_addr_t ip_addr; /* IP address */
+ ip_addr_t netmask; /* subnet mask */
+ ip_addr_t gw; /* gateway */
+#if NETDEV_IPV6
+ ip_addr_t ip6_addr[NETDEV_IPV6_NUM_ADDRESSES]; /* array of IPv6 addresses */
+#endif /* NETDEV_IPV6 */
+ ip_addr_t dns_servers[NETDEV_DNS_SERVERS_NUM]; /* DNS server */
+ uint8_t hwaddr_len; /* hardware address length */
+ uint8_t hwaddr[NETDEV_HWADDR_MAX_LEN]; /* hardware address */
+
+ uint16_t flags; /* network interface device status flag */
+ uint16_t mtu; /* maximum transfer unit (in bytes) */
+ const struct netdev_ops *ops; /* network interface device operations */
+
+ netdev_callback_fn status_callback; /* network interface device flags change callback */
+ netdev_callback_fn addr_callback; /* network interface device address information change callback */
+
+#ifdef RT_USING_SAL
+ void *sal_user_data; /* user-specific data for SAL */
+#endif /* RT_USING_SAL */
+ void *user_data; /* user-specific data */
+};
+
+/* The list of network interface device */
+extern struct netdev *netdev_list;
+/* The default network interface device */
+extern struct netdev *netdev_default;
+
+/* The network interface device ping response object */
+struct netdev_ping_resp
+{
+ ip_addr_t ip_addr; /* response IP address */
+ uint16_t data_len; /* response data length */
+ uint16_t ttl; /* time to live */
+ uint32_t ticks; /* response time, unit tick */
+ void *user_data; /* user-specific data */
+};
+
+/* The network interface device operations */
+struct netdev_ops
+{
+ /* set network interface device hardware status operations */
+ int (*set_up)(struct netdev *netdev);
+ int (*set_down)(struct netdev *netdev);
+
+ /* set network interface device address information operations */
+ int (*set_addr_info)(struct netdev *netdev, ip_addr_t *ip_addr, ip_addr_t *netmask, ip_addr_t *gw);
+ int (*set_dns_server)(struct netdev *netdev, uint8_t dns_num, ip_addr_t *dns_server);
+ int (*set_dhcp)(struct netdev *netdev, _Bool is_enabled);
+
+#ifdef RT_USING_FINSH
+ /* set network interface device common network interface device operations */
+ int (*ping)(struct netdev *netdev, const char *host, size_t data_len, uint32_t timeout, struct netdev_ping_resp *ping_resp);
+ void (*netstat)(struct netdev *netdev);
+#endif
+
+ /* set default network interface device in current network stack*/
+ int (*set_default)(struct netdev *netdev);
+};
+
+/* The network interface device registered and unregistered*/
+int netdev_register(struct netdev *netdev, const char *name, void *user_data);
+int netdev_unregister(struct netdev *netdev);
+
+/* Get network interface device object */
+struct netdev *netdev_get_first_by_flags(uint16_t flags);
+struct netdev *netdev_get_by_ipaddr(ip_addr_t *ip_addr);
+struct netdev *netdev_get_by_name(const char *name);
+#ifdef RT_USING_SAL
+struct netdev *netdev_get_by_family(int family);
+int netdev_family_get(struct netdev *netdev);
+#endif /* RT_USING_SAL */
+
+/* Set default network interface device in list */
+void netdev_set_default(struct netdev *netdev);
+
+/* Set network interface device status */
+int netdev_set_up(struct netdev *netdev);
+int netdev_set_down(struct netdev *netdev);
+int netdev_dhcp_enabled(struct netdev *netdev, _Bool is_enabled);
+
+/* Get network interface device status */
+#define netdev_is_up(netdev) (((netdev)->flags & NETDEV_FLAG_UP) ? (uint8_t)1 : (uint8_t)0)
+#define netdev_is_link_up(netdev) (((netdev)->flags & NETDEV_FLAG_LINK_UP) ? (uint8_t)1 : (uint8_t)0)
+#define netdev_is_internet_up(netdev) (((netdev)->flags & NETDEV_FLAG_INTERNET_UP) ? (uint8_t)1 : (uint8_t)0)
+#define netdev_is_dhcp_enabled(netdev) (((netdev)->flags & NETDEV_FLAG_DHCP) ? (uint8_t)1 : (uint8_t)0)
+
+/* Set network interface device address */
+int netdev_set_ipaddr(struct netdev *netdev, const ip_addr_t *ipaddr);
+int netdev_set_netmask(struct netdev *netdev, const ip_addr_t *netmask);
+int netdev_set_gw(struct netdev *netdev, const ip_addr_t *gw);
+int netdev_set_dns_server(struct netdev *netdev, uint8_t dns_num, const ip_addr_t *dns_server);
+
+/* Set network interface device callback, it can be called when the status or address changed */
+void netdev_set_status_callback(struct netdev *netdev, netdev_callback_fn status_callback);
+void netdev_set_addr_callback(struct netdev *netdev, netdev_callback_fn addr_callback);
+
+/* Set network interface device status and address, this function can only be called in the network interface device driver */
+void netdev_low_level_set_ipaddr(struct netdev *netdev, const ip_addr_t *ipaddr);
+void netdev_low_level_set_netmask(struct netdev *netdev, const ip_addr_t *netmask);
+void netdev_low_level_set_gw(struct netdev *netdev, const ip_addr_t *gw);
+void netdev_low_level_set_dns_server(struct netdev *netdev, uint8_t dns_num, const ip_addr_t *dns_server);
+void netdev_low_level_set_status(struct netdev *netdev, _Bool is_up);
+void netdev_low_level_set_link_status(struct netdev *netdev, _Bool is_up);
+void netdev_low_level_set_internet_status(struct netdev *netdev, _Bool is_up);
+void netdev_low_level_set_dhcp_status(struct netdev *netdev, _Bool is_enable);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __NETDEV_H__ */
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev_ipaddr.h b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev_ipaddr.h
new file mode 100755
index 000000000..176c91939
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/include/netdev_ipaddr.h
@@ -0,0 +1,319 @@
+/*
+ * Copyright (c) 2006-2021, RT-Thread Development Team
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Change Logs:
+ * Date Author Notes
+ * 2022-8-22 zhaoyun for XiZi
+ */
+
+#ifndef __NETDEV_IPADDR_H__
+#define __NETDEV_IPADDR_H__
+
+#include
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+
+
+
+/*
+ * Definitions of bits in internet address integers.
+ * On subnets, the decomposition of addresses to host and net parts
+ * is done according to subnet mask, not the masks here.
+ */
+//#define IN_CLASSA(i) (((long)(i) & 0x80000000) == 0)
+//#define IN_CLASSA_NET 0xff000000
+//#define IN_CLASSA_NSHIFT 24
+//#define IN_CLASSA_HOST 0x00ffffff
+//#define IN_CLASSA_MAX 128
+
+//#define IN_CLASSB(i) (((long)(i) & 0xc0000000) == 0x80000000)
+//#define IN_CLASSB_NET 0xffff0000
+//#define IN_CLASSB_NSHIFT 16
+//#define IN_CLASSB_HOST 0x0000ffff
+//#define IN_CLASSB_MAX 65536
+
+//#define IN_CLASSC(i) (((long)(i) & 0xe0000000) == 0xc0000000)
+//#define IN_CLASSC_NET 0xffffff00
+//#define IN_CLASSC_NSHIFT 8
+//#define IN_CLASSC_HOST 0x000000ff
+
+//#define IN_CLASSD(i) (((long)(i) & 0xf0000000) == 0xe0000000)
+//#define IN_MULTICAST(i) IN_CLASSD(i)
+
+//#define IN_EXPERIMENTAL(i) (((long)(i) & 0xe0000000) == 0xe0000000)
+//#define IN_BADCLASS(i) (((long)(i) & 0xf0000000) == 0xf0000000)
+
+//#define IN_LOOPBACKNET 127 /* official! */
+
+/* IP address types for use in ip_addr_t.type member */
+//enum netdev_ip_addr_type {
+ /** IPv4 */
+ //IPADDR_TYPE_V4 = 0U,
+ /** IPv6 */
+ //IPADDR_TYPE_V6 = 6U,
+ /** IPv4+IPv6 ("dual-stack") */
+ //IPADDR_TYPE_ANY = 46U/
+//};
+
+#if NETDEV_IPV4
+/** IPv4 only: set the IP address given as an u32_t */
+#define ip4_addr_set_u32(dest_ipaddr, src_u32) ((dest_ipaddr)->addr = (src_u32))
+/** IPv4 only: get the IP address as an u32_t */
+#define ip4_addr_get_u32(src_ipaddr) ((src_ipaddr)->addr)
+
+#define IP4ADDR_STRLEN_MAX 16
+#endif /* NETIF_IPV4 */
+
+/* These macros should be calculated by the preprocessor and are used
+ with compile-time constants only (so that there is no little-endian
+ overhead at runtime). */
+//#define PP_HTONS(x) ((((x) & 0x00ffUL) << 8) | (((x) & 0xff00UL) >> 8))
+#define PP_NTOHS(x) PP_HTONS(x)
+//#define PP_HTONL(x) ((((x) & 0x000000ffUL) << 24) | \
+ // (((x) & 0x0000ff00UL) << 8) | \
+ // (((x) & 0x00ff0000UL) >> 8) | \
+ // (((x) & 0xff000000UL) >> 24))
+#define PP_NTOHL(x) PP_HTONL(x)
+
+//#define htons(x) (uint16_t)PP_HTONS(x)
+//#define ntohs(x) (uint16_t)PP_NTOHS(x)
+//#define htonl(x) (uint32_t)PP_HTONL(x)
+//#define ntohl(x) (uint32_t)PP_NTOHL(x)
+
+/* If your port already typedef's in_addr_t, define IN_ADDR_T_DEFINED
+ to prevent this code from redefining it. */
+#if !defined(in_addr_t) && !defined(IN_ADDR_T_DEFINED)
+typedef uint32_t in_addr_t;
+#endif
+
+#if NETDEV_IPV4
+struct in_addr
+{
+ in_addr_t s_addr;
+};
+
+typedef struct ip4_addr
+{
+ uint32_t addr;
+} ip4_addr_t;
+
+/** 255.255.255.255 */
+#define IPADDR_NONE ((uint32_t)0xffffffffUL)
+/** 127.0.0.1 */
+#define IPADDR_LOOPBACK ((uint32_t)0x7f000001UL)
+/** 0.0.0.0 */
+#define IPADDR_ANY ((uint32_t)0x00000000UL)
+/** 255.255.255.255 */
+#define IPADDR_BROADCAST ((uint32_t)0xffffffffUL)
+
+/** 255.255.255.255 */
+#define INADDR_NONE IPADDR_NONE
+/** 127.0.0.1 */
+#define INADDR_LOOPBACK IPADDR_LOOPBACK
+/** 0.0.0.0 */
+#define INADDR_ANY IPADDR_ANY
+/** 255.255.255.255 */
+#define INADDR_BROADCAST IPADDR_BROADCAST
+
+#define IPADDR_BROADCAST_STRING "255.255.255.255"
+
+/** Copy IP address - faster than ip4_addr_set: no NULL check */
+#define ip4_addr_copy(dest, src) ((dest).addr = (src).addr)
+#define ip4_addr_cmp(addr1, addr2) ((addr1)->addr == (addr2)->addr)
+/** Safely copy one IP address to another (src may be NULL) */
+#define ip4_addr_set(dest, src) ((dest)->addr = ((src) == NULL ? 0 : (src)->addr))
+/** Set complete address to zero */
+#define ip4_addr_set_zero(ipaddr) ((ipaddr)->addr = 0)
+/** Set address to IPADDR_ANY (no need for htonl()) */
+#define ip4_addr_set_any(ipaddr) ((ipaddr)->addr = IPADDR_ANY)
+#define ip4_addr_isany_val(ipaddr) ((ipaddr).addr == IPADDR_ANY)
+#define ip4_addr_isany(ipaddr) ((ipaddr) == NULL || ip4_addr_isany_val(*(ipaddr)))
+
+in_addr_t netdev_ipaddr_addr(const char *cp);
+int netdev_ip4addr_aton(const char *cp, ip4_addr_t *addr);
+char *netdev_ip4addr_ntoa(const ip4_addr_t *addr);
+char *netdev_ip4addr_ntoa_r(const ip4_addr_t *addr, char *buf, int buflen);
+#endif /* NETIF_IPV4 */
+
+#if NETDEV_IPV6
+struct in6_addr
+{
+ union
+ {
+ uint32_t u32_addr[4];
+ uint8_t u8_addr[16];
+ } un;
+#define s6_addr un.u8_addr
+};
+
+typedef struct ip6_addr
+{
+ uint32_t addr[4];
+#ifdef NETDEV_IPV6_SCOPES
+ uint8_t zone;
+#endif /* NETDEV_IPV6_SCOPES */
+} ip6_addr_t;
+
+/** This macro can be used to initialize a variable of type struct in6_addr
+ to the IPv6 wildcard address. */
+#define IN6ADDR_ANY_INIT {{{0,0,0,0}}}
+/** This macro can be used to initialize a variable of type struct in6_addr
+ to the IPv6 loopback address. */
+#define IN6ADDR_LOOPBACK_INIT {{{0,0,0,PP_HTONL(1)}}}
+
+/** This variable is initialized by the system to contain the wildcard IPv6 address.
+ */
+extern const struct in6_addr in6addr_any;
+
+#define ip6_addr_cmp(addr1, addr2) (((addr1)->addr[0] == (addr2)->addr[0]) && \
+ ((addr1)->addr[1] == (addr2)->addr[1]) && \
+ ((addr1)->addr[2] == (addr2)->addr[2]) && \
+ ((addr1)->addr[3] == (addr2)->addr[3]))
+/** Copy IPv6 address - faster than ip6_addr_set: no NULL check */
+#define ip6_addr_copy(dest, src) do{(dest).addr[0] = (src).addr[0]; \
+ (dest).addr[1] = (src).addr[1]; \
+ (dest).addr[2] = (src).addr[2]; \
+ (dest).addr[3] = (src).addr[3];}while(0)
+/** Safely copy one IPv6 address to another (src may be NULL) */
+#define ip6_addr_set(dest, src) do{(dest)->addr[0] = (src) == NULL ? 0 : (src)->addr[0]; \
+ (dest)->addr[1] = (src) == NULL ? 0 : (src)->addr[1]; \
+ (dest)->addr[2] = (src) == NULL ? 0 : (src)->addr[2]; \
+ (dest)->addr[3] = (src) == NULL ? 0 : (src)->addr[3];}while(0)
+/** Set complete address to zero */
+#define ip6_addr_set_zero(ip6addr) do{(ip6addr)->addr[0] = 0; \
+ (ip6addr)->addr[1] = 0; \
+ (ip6addr)->addr[2] = 0; \
+ (ip6addr)->addr[3] = 0;}while(0)
+/** Set address to ipv6 'any' (no need for lwip_htonl()) */
+#define ip6_addr_set_any(ip6addr) ip6_addr_set_zero(ip6addr)
+#define ip6_addr_isany_val(ip6addr) (((ip6addr).addr[0] == 0) && \
+ ((ip6addr).addr[1] == 0) && \
+ ((ip6addr).addr[2] == 0) && \
+ ((ip6addr).addr[3] == 0))
+#define ip6_addr_isany(ip6addr) (((ip6addr) == NULL) || ip6_addr_isany_val(*(ip6addr)))
+
+int netdev_ip6addr_aton(const char *cp, ip6_addr_t *addr);
+char *netdev_ip6addr_ntoa(const ip6_addr_t *addr);
+char *netdev_ip6addr_ntoa_r(const ip6_addr_t *addr, char *buf, int buflen);
+#endif /* NETIF_IPV6 */
+
+#if NETDEV_IPV4 && NETDEV_IPV6
+/* A union struct for both IP version's addresses */
+typedef struct _ip_addr
+{
+ union
+ {
+ ip6_addr_t ip6;
+ ip4_addr_t ip4;
+ } u_addr;
+ /** @ref netdev_ip_addr_type */
+ uint8_t type;
+} ip_addr_t;
+
+#define IP_SET_TYPE_VAL(ipaddr, iptype) do { (ipaddr).type = (iptype); }while(0)
+#define IP_SET_TYPE(ipaddr, iptype) do { if((ipaddr) != NULL) { IP_SET_TYPE_VAL(*(ipaddr), iptype); }}while(0)
+#define IP_GET_TYPE(ipaddr) ((ipaddr)->type)
+
+#define IP_IS_V4_VAL(ipaddr) (IP_GET_TYPE(&ipaddr) == IPADDR_TYPE_V4)
+#define IP_IS_V6_VAL(ipaddr) (IP_GET_TYPE(&ipaddr) == IPADDR_TYPE_V6)
+#define IP_IS_V4(ipaddr) (((ipaddr) == NULL) || IP_IS_V4_VAL(*(ipaddr)))
+#define IP_IS_V6(ipaddr) (((ipaddr) != NULL) && IP_IS_V6_VAL(*(ipaddr)))
+
+/** Convert generic ip address to specific protocol version */
+#define ip_2_ip6(ipaddr) (&((ipaddr)->u_addr.ip6))
+/** Convert generic ip address to specific protocol version */
+#define ip_2_ip4(ipaddr) (&((ipaddr)->u_addr.ip4))
+
+#define ip_addr_copy(dest, src) do{ IP_SET_TYPE_VAL(dest, IP_GET_TYPE(&src)); if(IP_IS_V6_VAL(src)){ \
+ ip6_addr_copy(*ip_2_ip6(&(dest)), *ip_2_ip6(&(src))); }else{ \
+ ip4_addr_copy(*ip_2_ip4(&(dest)), *ip_2_ip4(&(src))); }}while(0)
+#define ip_addr_cmp(addr1, addr2) ((IP_GET_TYPE(addr1) != IP_GET_TYPE(addr2)) ? 0 : (IP_IS_V6_VAL(*(addr1)) ? \
+ ip6_addr_cmp(ip_2_ip6(addr1), ip_2_ip6(addr2)) : \
+ ip4_addr_cmp(ip_2_ip4(addr1), ip_2_ip4(addr2))))
+#define ip_addr_set(dest, src) do{ IP_SET_TYPE(dest, IP_GET_TYPE(src)); if(IP_IS_V6(src)){ \
+ ip6_addr_set(ip_2_ip6(dest), ip_2_ip6(src)); }else{ \
+ ip4_addr_set(ip_2_ip4(dest), ip_2_ip4(src)); }}while(0)
+#define ip_addr_set_zero(ipaddr) do{ ip6_addr_set_zero(ip_2_ip6(ipaddr)); IP_SET_TYPE(ipaddr, 0); }while(0)
+#define ip_addr_set_any(is_ipv6, ipaddr) do{ if(is_ipv6){ \
+ ip6_addr_set_any(ip_2_ip6(ipaddr)); IP_SET_TYPE(ipaddr, IPADDR_TYPE_V6); }else{ \
+ ip4_addr_set_any(ip_2_ip4(ipaddr)); IP_SET_TYPE(ipaddr, IPADDR_TYPE_V4); }}while(0)
+
+#define ip_addr_isany_val(ipaddr) ((IP_IS_V6_VAL(ipaddr)) ? \
+ ip6_addr_isany_val(*ip_2_ip6(&(ipaddr))) : \
+ ip4_addr_isany_val(*ip_2_ip4(&(ipaddr))))
+#define ip_addr_isany(ipaddr) ((IP_IS_V6(ipaddr)) ? \
+ ip6_addr_isany(ip_2_ip6(ipaddr)) : \
+ ip4_addr_isany(ip_2_ip4(ipaddr)))
+
+/* directly map this to the lwip internal functions */
+#define inet_addr(cp) netdev_ipaddr_addr(cp)
+#define inet_aton(cp, addr) ((IP_IS_V6_VAL(*addr)) ? \
+ netdev_ip6addr_aton(cp, ip_2_ip6(addr)) : \
+ netdev_ip4addr_aton(cp, ip_2_ip4(addr)))
+#define inet_ntoa(addr) ((IP_IS_V6_VAL(addr)) ? \
+ netdev_ip6addr_ntoa(ip_2_ip6(&addr)) : \
+ netdev_ip4addr_ntoa(ip_2_ip4(&addr)))
+#define inet_ntoa_r(addr, buf, buflen) ((IP_IS_V6_VAL(addr)) ? \
+ netdev_ip6addr_ntoa_r(ip_2_ip6(&addr), buf, buflen) : \
+ netdev_ip4addr_ntoa_r(ip_2_ip4(&addr), buf, buflen))
+#elif NETDEV_IPV4 /* NETDEV_IPV4 */
+
+typedef ip4_addr_t ip_addr_t;
+
+#define IP_SET_TYPE_VAL(ipaddr, iptype)
+#define IP_SET_TYPE(ipaddr, iptype)
+#define IP_GET_TYPE(ipaddr) IPADDR_TYPE_V4
+
+#define ip_addr_copy(dest, src) ip4_addr_copy(dest, src)
+#define ip_addr_cmp(addr1, addr2) ip4_addr_cmp(addr1, addr2)
+#define ip_addr_set(dest, src) ip4_addr_set(dest, src)
+#define ip_addr_set_zero(ipaddr) ip4_addr_set_zero(ipaddr)
+#define ip_addr_set_any(is_ipv6, ipaddr) ip4_addr_set_any(ipaddr)
+#define ip_addr_isany_val(ipaddr) ip4_addr_isany_val(ipaddr)
+#define ip_addr_isany(ipaddr) ip4_addr_isany(ipaddr)
+
+/* directly map this to the lwip internal functions */
+#define inet_addr(cp) netdev_ipaddr_addr(cp)
+#define inet_aton(cp, addr) netdev_ip4addr_aton(cp,(ip4_addr_t*)addr)
+#define inet_ntoa(addr) netdev_ip4addr_ntoa((const ip4_addr_t*)&(addr))
+#define inet_ntoa_r(addr, buf, buflen) netdev_ip4addr_ntoa_r((const ip4_addr_t*)&(addr), buf, buflen)
+#else /* NETDEV_IPV6 */
+
+//typedef ip6_addr_t ip_addr_t;
+
+#define IP_SET_TYPE_VAL(ipaddr, iptype)
+#define IP_SET_TYPE(ipaddr, iptype)
+//#define IP_GET_TYPE(ipaddr) IPADDR_TYPE_V6
+
+//#define ip_addr_copy(dest, src) ip6_addr_copy(dest, src)
+//#define ip_addr_cmp(addr1, addr2) ip6_addr_cmp(addr1, addr2)
+//#define ip_addr_set(dest, src) ip6_addr_set(dest, src)
+//#define ip_addr_set_zero(ipaddr) ip6_addr_set_zero(ipaddr)
+//#define ip_addr_set_any(is_ipv6, ipaddr) ip6_addr_set_any(ipaddr)
+//#define ip_addr_isany_val(ipaddr) ip6_addr_isany_val(ipaddr)
+//#define ip_addr_isany(ipaddr) ip6_addr_isany(ipaddr)
+
+/* directly map this to the lwip internal functions */
+//#define inet_aton(cp, addr) netdev_ip6addr_aton(cp, (ip6_addr_t*)addr)
+//#define inet_ntoa(addr) netdev_ip6addr_ntoa((const ip6_addr_t*)&(addr))
+//#define inet_ntoa_r(addr, buf, buflen) netdev_ip6addr_ntoa_r((const ip6_addr_t*)&(addr), buf, buflen)
+#endif /* NTDEV_IPV4 && NTDEV_IPV6 */
+
+
+const char *netdev_inet_ntop(int af, const void *src, char *dst, int32_t size);
+int netdev_inet_pton(int af, const char *src, void *dst);
+
+//#define inet_ntop(af, src, dst, size) netdev_inet_ntop(af, src, dst, size)
+//#define inet_pton(af, src, dst) netdev_inet_pton(af, src, dst)
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __NETDEV_IPADDR_H__ */
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/Makefile b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/Makefile
new file mode 100755
index 000000000..bf3fa4082
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/Makefile
@@ -0,0 +1,3 @@
+SRC_FILES := netdev.c netdev_ipaddr.c
+
+include $(KERNEL_ROOT)/compiler.mk
\ No newline at end of file
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev.c b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev.c
new file mode 100755
index 000000000..28aa8ea63
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev.c
@@ -0,0 +1,1339 @@
+/*
+ * Copyright (c) 2006-2022, RT-Thread Development Team
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Change Logs:
+ * Date Author Notes
+ * 2022-08-22 zhaoyun for XiZi
+ */
+
+#include
+#include
+#include
+#include
+#include
+
+
+#include "../include/netdev_ipaddr.h"
+#include "../include/netdev.h"
+
+#include "sys_arch.h"
+#include
+#include "lwip/sys.h"
+
+
+
+#define DBG_TAG "netdev"
+#define DBG_LVL DBG_INFO
+
+
+/* The list of network interface device */
+struct netdev *netdev_list = NULL;
+/* The default network interface device */
+struct netdev *netdev_default = NULL;
+
+/**
+ * This function will register network interface device and
+ * add it to network interface device list.
+ *
+ * @param netdev the network interface device object
+ * @param name the network interface device name
+ * @param user_data user-specific data
+ *
+ * @return 0: registered successfully
+ * -1: registered failed
+ */
+
+void xizi_slist_init(xizi_slist_t *l)
+{
+ l->next = NULL;
+}
+
+void xizi_slist_append(xizi_slist_t *l, xizi_slist_t *n)
+{
+ struct xizi_slist_node *node;
+
+ node = l;
+ while (node->next) node = node->next;
+
+ /* append the node to the tail */
+ node->next = n;
+ n->next = NULL;
+}
+xizi_slist_t *xizi_slist_next(xizi_slist_t *n)
+{
+ return n->next;
+}
+
+
+#define xizi_container_of(ptr, type, member) ((type *)((char *)(ptr) - (unsigned long)(&((type *)0)->member)))
+
+#define xizi_slist_entry(node, type, member) xizi_container_of(node, type, member)
+
+xizi_slist_t *xizi_slist_remove(xizi_slist_t *l, xizi_slist_t *n)
+{
+ /* remove slist head */
+ struct xizi_slist_node *node = l;
+ while (node->next && node->next != n) node = node->next;
+
+ /* remove node */
+ if (node->next != (xizi_slist_t *)0) node->next = node->next->next;
+
+ return l;
+}
+
+
+
+
+
+int netdev_register(struct netdev *netdev, const char *name, void *user_data)
+{
+ long level;
+ uint16_t flags_mask;
+ uint16_t index;
+
+ assert(netdev);
+ assert(name);
+
+ /* clean network interface device */
+ flags_mask = NETDEV_FLAG_UP | NETDEV_FLAG_LINK_UP | NETDEV_FLAG_INTERNET_UP | NETDEV_FLAG_DHCP;
+ netdev->flags &= ~flags_mask;
+
+ ip_addr_set_zero(&(netdev->ip_addr));
+ ip_addr_set_zero(&(netdev->netmask));
+ ip_addr_set_zero(&(netdev->gw));
+
+ IP_SET_TYPE_VAL(netdev->ip_addr, IPADDR_TYPE_V4);
+ IP_SET_TYPE_VAL(netdev->netmask, IPADDR_TYPE_V4);
+ IP_SET_TYPE_VAL(netdev->gw, IPADDR_TYPE_V4);
+#if NETDEV_IPV6
+ for (index = 0; index < NETDEV_IPV6_NUM_ADDRESSES; index++)
+ {
+ ip_addr_set_zero(&(netdev->ip6_addr[index]));
+ IP_SET_TYPE_VAL(netdev->ip_addr, IPADDR_TYPE_V6);
+ }
+#endif /* NETDEV_IPV6 */
+ for (index = 0; index < NETDEV_DNS_SERVERS_NUM; index++)
+ {
+ ip_addr_set_zero(&(netdev->dns_servers[index]));
+ IP_SET_TYPE_VAL(netdev->ip_addr, IPADDR_TYPE_V4);
+ }
+ netdev->status_callback = NULL;
+ netdev->addr_callback = NULL;
+
+ if(strlen(name) > XiZi_NAME_MAX)
+ {
+ char netdev_name[XiZi_NAME_MAX + 1] = {0};
+
+ strncpy(netdev_name, name, XiZi_NAME_MAX);
+ printf("netdev name[%s] length is so long that have been cut into [%s].", name, netdev_name);
+ }
+
+ /* fill network interface device */
+ strncpy(netdev->name, name, XiZi_NAME_MAX);
+ netdev->user_data = user_data;
+
+ /* initialize current network interface device single list */
+
+
+
+
+ xizi_slist_init(&(netdev->list));
+
+ level = CriticalAreaLock();
+
+ if (netdev_list == NULL)
+ {
+ netdev_list = netdev;
+ netdev_default = netdev;
+ }
+ else
+ {
+ /* tail insertion */
+ xizi_slist_append(&(netdev_list->list), &(netdev->list));
+ }
+
+ CriticalAreaUnLock(level);
+
+ return 0;
+}
+
+/**
+ * This function will unregister network interface device and
+ * delete it from network interface device list.
+ *
+ * @param netdev the network interface device object
+ *
+ * @return 0: unregistered successfully
+ * -1: unregistered failed
+ */
+int netdev_unregister(struct netdev *netdev)
+{
+ long level;
+ xizi_slist_t *node = NULL;
+ struct netdev *cur_netdev = NULL;
+
+ assert(netdev);
+
+ if (netdev_list == NULL)
+ {
+ return -1;
+ }
+
+ level = CriticalAreaLock();
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ cur_netdev = xizi_slist_entry(node, struct netdev, list);
+ if (cur_netdev == netdev)
+ {
+ /* find this network interface device in network interface device list */
+ if (netdev_list == netdev && xizi_slist_next(&netdev_list->list) == NULL)
+ {
+ netdev_list = NULL;
+ }
+ else
+ {
+ xizi_slist_remove(&(netdev_list->list), &(cur_netdev->list));
+ }
+ if (netdev_default == netdev)
+ {
+ netdev_default = netdev_list;
+ }
+ break;
+ }
+ }
+ CriticalAreaUnLock(level);
+
+ if (cur_netdev == netdev)
+ {
+#ifdef RT_USING_SAL
+ extern int sal_netdev_cleanup(struct netdev *netdev);
+ sal_netdev_cleanup(netdev);
+#endif
+ memset(netdev, 0, sizeof(*netdev));
+ }
+
+ return -1;
+}
+
+/**
+ * This function will get the first network interface device
+ * with the flags in network interface device list.
+ *
+ * @param flags the network interface device flags
+ *
+ * @return != NULL: network interface device object
+ * NULL: get failed
+ */
+struct netdev *netdev_get_first_by_flags(uint16_t flags)
+{
+ long level;
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+
+ if (netdev_list == NULL)
+ {
+ return NULL;
+ }
+
+ level = CriticalAreaLock();
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = xizi_slist_entry(node, struct netdev, list);
+ if (netdev && (netdev->flags & flags) != 0)
+ {
+ CriticalAreaUnLock(level);
+ return netdev;
+ }
+ }
+
+ CriticalAreaUnLock(level);
+
+ return NULL;
+}
+
+/**
+ * This function will get the first network interface device
+ * in network interface device list by IP address.
+ *
+ * @param addr the network interface device IP address
+ *
+ * @return != NULL: network interface device object
+ * NULL: get failed
+ */
+struct netdev *netdev_get_by_ipaddr(ip_addr_t *ip_addr)
+{
+ long level;
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+
+ if (netdev_list == NULL)
+ {
+ return NULL;
+ }
+
+ level = CriticalAreaLock();
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = xizi_slist_entry(node, struct netdev, list);
+ if (netdev && ip_addr_cmp(&(netdev->ip_addr), ip_addr))
+ {
+ CriticalAreaUnLock(level);
+ return netdev;
+ }
+ }
+
+ CriticalAreaUnLock(level);
+
+ return NULL;
+}
+
+/**
+ * This function will get network interface device
+ * in network interface device list by netdev name.
+ *
+ * @param name the network interface device name
+ *
+ * @return != NULL: network interface device object
+ * NULL: get failed
+ */
+struct netdev *netdev_get_by_name(const char *name)
+{
+ long level;
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+
+ if (netdev_list == NULL)
+ {
+ return NULL;
+ }
+
+ level = CriticalAreaLock();
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = xizi_slist_entry(node, struct netdev, list);
+ if (netdev && (strncmp(netdev->name, name, strlen(netdev->name) < XiZi_NAME_MAX ? strlen(netdev->name) : XiZi_NAME_MAX) == 0))
+ {
+ CriticalAreaUnLock(level);
+ return netdev;
+ }
+ }
+
+ CriticalAreaUnLock(level);
+
+ return NULL;
+}
+
+#ifdef RT_USING_SAL
+/**
+ * This function will get the first network interface device
+ * in network interface device list by protocol family type.
+ *
+ * @param family the network interface device protocol family type
+ *
+ * @return != NULL: network interface device object
+ * NULL: get failed
+ */
+struct netdev *netdev_get_by_family(int family)
+{
+ long level;
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+ struct sal_proto_family *pf = NULL;
+
+ if (netdev_list == NULL)
+ {
+ return NULL;
+ }
+
+ level = CriticalAreaLock();
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = xizi_slist_entry(node, struct netdev, list);
+ pf = (struct sal_proto_family *) netdev->sal_user_data;
+ if (pf && pf->skt_ops && pf->family == family && netdev_is_up(netdev))
+ {
+ CriticalAreaUnLock(level);
+ return netdev;
+ }
+ }
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = xizi_slist_entry(node, struct netdev, list);
+ pf = (struct sal_proto_family *) netdev->sal_user_data;
+ if (pf && pf->skt_ops && pf->sec_family == family && netdev_is_up(netdev))
+ {
+ CriticalAreaUnLock(level);
+ return netdev;
+ }
+ }
+
+ CriticalAreaUnLock(level);
+
+ return NULL;
+}
+
+/**
+ * This function will get the family type from network interface device
+ *
+ * @param netdev network interface device object
+ *
+ * @return the network interface device family type
+ */
+int netdev_family_get(struct netdev *netdev)
+{
+ assert(netdev);
+
+ return ((struct sal_proto_family *)netdev->sal_user_data)->family;
+}
+
+#endif /* RT_USING_SAL */
+
+/**
+ * This function will set default network interface device.
+ *
+ * @param netdev the network interface device to change
+ */
+void netdev_set_default(struct netdev *netdev)
+{
+ if (netdev)
+ {
+ netdev_default = netdev;
+
+ if (netdev->ops->set_default)
+ {
+ /* set default network interface device in the current network stack */
+ netdev->ops->set_default(netdev);
+ }
+ printf("Setting default network interface device name(%s) successfully.", netdev->name);
+ }
+}
+
+/**
+ * This function will enable network interface device .
+ *
+ * @param netdev the network interface device to change
+ *
+ * @return 0: set status successfully
+ * -1: set status failed
+ */
+int netdev_set_up(struct netdev *netdev)
+{
+ assert(netdev);
+
+ if (!netdev->ops || !netdev->ops->set_up)
+ {
+ printf("The network interface device(%s) not support to set status.", netdev->name);
+ return -1;
+ }
+
+ /* network interface device status flags check */
+ if (netdev_is_up(netdev))
+ {
+ return 0;
+ }
+
+ /* execute enable network interface device operations by network interface device driver */
+ return netdev->ops->set_up(netdev);
+}
+/**
+ * This function will disable network interface device.
+ *
+ * @param netdev the network interface device to change
+ *
+ * @return 0: set status successfully
+ * -1: set sttaus failed
+ */
+int netdev_set_down(struct netdev *netdev)
+{
+ assert(netdev);
+
+ if (!netdev->ops || !netdev->ops->set_down)
+ {
+ printf("The network interface device(%s) not support to set status.", netdev->name);
+ return -1;
+ }
+
+ /* network interface device status flags check */
+ if (!netdev_is_up(netdev))
+ {
+ return 0;
+ }
+
+ /* execute disable network interface device operations by network interface driver */
+ return netdev->ops->set_down(netdev);
+}
+
+/**
+ * This function will control network interface device DHCP capability enable or disable.
+ *
+ * @param netdev the network interface device device to change
+ * @param is_enable the new DHCP status
+ *
+ * @return 0: set DHCP status successfully
+ * -1: set DHCP status failed
+ */
+int netdev_dhcp_enabled(struct netdev *netdev, _Bool is_enabled)
+{
+ assert(netdev);
+
+ if (!netdev->ops || !netdev->ops->set_dhcp)
+ {
+ printf("The network interface device(%s) not support to set DHCP status.", netdev->name);
+ return -1;
+ }
+
+ /* network interface device DHCP flags check */
+ if (netdev_is_dhcp_enabled(netdev) == is_enabled)
+ {
+ return 0;
+ }
+
+ /* execute network interface device DHCP capability control operations */
+ return netdev->ops->set_dhcp(netdev, is_enabled);
+}
+
+/**
+ * This function will set network interface device IP address.
+ *
+ * @param netdev the network interface device to change
+ * @param ipaddr the new IP address
+ *
+ * @return 0: set IP address successfully
+ * -1: set IP address failed
+ */
+int netdev_set_ipaddr(struct netdev *netdev, const ip_addr_t *ip_addr)
+{
+ assert(netdev);
+ assert(ip_addr);
+
+ if (!netdev->ops || !netdev->ops->set_addr_info)
+ {
+ printf("The network interface device(%s) not support to set IP address.", netdev->name);
+ return -1;
+ }
+
+ if (netdev_is_dhcp_enabled(netdev))
+ {
+ printf("The network interface device(%s) DHCP capability is enable, not support set IP address.", netdev->name);
+ return -1;
+ }
+
+ /* execute network interface device set IP address operations */
+ return netdev->ops->set_addr_info(netdev, (ip_addr_t *)ip_addr, NULL, NULL);
+}
+
+/**
+ * This function will set network interface device netmask address.
+ *
+ * @param netdev the network interface device to change
+ * @param netmask the new netmask address
+ *
+ * @return 0: set netmask address successfully
+ * -1: set netmask address failed
+ */
+int netdev_set_netmask(struct netdev *netdev, const ip_addr_t *netmask)
+{
+ assert(netdev);
+ assert(netmask);
+
+ if (!netdev->ops || !netdev->ops->set_addr_info)
+ {
+ printf("The network interface device(%s) not support to set netmask address.", netdev->name);
+ return -1;
+ }
+
+ if (netdev_is_dhcp_enabled(netdev))
+ {
+ printf("The network interface device(%s) DHCP capability is enable, not support set netmask address.", netdev->name);
+ return -1;
+ }
+
+ /* execute network interface device set netmask address operations */
+ return netdev->ops->set_addr_info(netdev, NULL, (ip_addr_t *)netmask, NULL);
+}
+
+/**
+ * This function will set network interface device gateway address.
+ *
+ * @param netdev the network interface device to change
+ * @param gateway the new gateway address
+ *
+ * @return 0: set gateway address successfully
+ * -1: set gateway address failed
+ */
+int netdev_set_gw(struct netdev *netdev, const ip_addr_t *gw)
+{
+ assert(netdev);
+ assert(gw);
+
+ if (!netdev->ops || !netdev->ops->set_addr_info)
+ {
+ printf("The network interface device(%s) not support to set gateway address.", netdev->name);
+ return -1;
+ }
+
+ if (netdev_is_dhcp_enabled(netdev))
+ {
+ printf("The network interface device(%s) DHCP capability is enable, not support set gateway address.", netdev->name);
+ return -1;
+ }
+
+ /* execute network interface device set gateway address operations */
+ return netdev->ops->set_addr_info(netdev, NULL, NULL, (ip_addr_t *)gw);
+}
+
+/**
+ * This function will set network interface device DNS server address.
+ *
+ * @param netdev the network interface device to change
+ * @param dns_server the new DNS server address
+ *
+ * @return 0: set netmask address successfully
+ * -1: set netmask address failed
+ */
+int netdev_set_dns_server(struct netdev *netdev, uint8_t dns_num, const ip_addr_t *dns_server)
+{
+ assert(netdev);
+ assert(dns_server);
+
+ if (dns_num >= NETDEV_DNS_SERVERS_NUM)
+ {
+ printf("The number of DNS servers(%d) set exceeds the maximum number(%d).", dns_num + 1, NETDEV_DNS_SERVERS_NUM);
+ return -1;
+ }
+
+ if (!netdev->ops || !netdev->ops->set_dns_server)
+ {
+ printf("The network interface device(%s) not support to set DNS server address.", netdev->name);
+ return -1;
+ }
+
+ /* execute network interface device set DNS server address operations */
+ return netdev->ops->set_dns_server(netdev, dns_num, (ip_addr_t *)dns_server);
+}
+
+/**
+ * This function will set callback to be called when the network interface device status has been changed.
+ *
+ * @param netdev the network interface device to change
+ * @param status_callback the callback be called when the status has been changed.
+ */
+void netdev_set_status_callback(struct netdev *netdev, netdev_callback_fn status_callback)
+{
+ assert(netdev);
+ assert(status_callback);
+
+ netdev->status_callback = status_callback;
+}
+
+/**
+ * This function will set callback to be called when the network interface device address has been changed.
+ *
+ * @param netdev the network interface device to change
+ * @param addr_callback the callback be called when the address has been changed.
+ */
+void netdev_set_addr_callback(struct netdev *netdev, netdev_callback_fn addr_callback)
+{
+ assert(netdev);
+ assert(addr_callback);
+
+ netdev->addr_callback = addr_callback;
+}
+
+
+/**
+ * This function will set network interface device IP address.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param ipaddr the new IP address
+ */
+void netdev_low_level_set_ipaddr(struct netdev *netdev, const ip_addr_t *ip_addr)
+{
+ assert(ip_addr);
+
+ if (netdev && ip_addr_cmp(&(netdev->ip_addr), ip_addr) == 0)
+ {
+ ip_addr_copy(netdev->ip_addr, *ip_addr);
+
+#ifdef RT_USING_SAL
+ /* set network interface device flags to internet up */
+ if (netdev_is_up(netdev) && netdev_is_link_up(netdev))
+ {
+ sal_check_netdev_internet_up(netdev);
+ }
+#endif /* RT_USING_SAL */
+
+ /* execute IP address change callback function */
+ if (netdev->addr_callback)
+ {
+ netdev->addr_callback(netdev, NETDEV_CB_ADDR_IP);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device netmask address.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param netmask the new netmask address
+ */
+void netdev_low_level_set_netmask(struct netdev *netdev, const ip_addr_t *netmask)
+{
+ assert(netmask);
+
+ if (netdev && ip_addr_cmp(&(netdev->netmask), netmask) == 0)
+ {
+ ip_addr_copy(netdev->netmask, *netmask);
+
+#ifdef RT_USING_SAL
+ /* set network interface device flags to internet up */
+ if (netdev_is_up(netdev) && netdev_is_link_up(netdev) &&
+ !ip_addr_isany(&(netdev->ip_addr)))
+ {
+ sal_check_netdev_internet_up(netdev);
+ }
+#endif /* RT_USING_SAL */
+
+ /* execute netmask address change callback function */
+ if (netdev->addr_callback)
+ {
+ netdev->addr_callback(netdev, NETDEV_CB_ADDR_NETMASK);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device gateway address.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param gateway the new gateway address
+ */
+void netdev_low_level_set_gw(struct netdev *netdev, const ip_addr_t *gw)
+{
+ assert(gw);
+
+ if (netdev && ip_addr_cmp(&(netdev->gw), gw) == 0)
+ {
+ ip_addr_copy(netdev->gw, *gw);
+
+#ifdef RT_USING_SAL
+ /* set network interface device flags to internet up */
+ if (netdev_is_up(netdev) && netdev_is_link_up(netdev) &&
+ !ip_addr_isany(&(netdev->ip_addr)))
+ {
+ sal_check_netdev_internet_up(netdev);
+ }
+#endif /* RT_USING_SAL */
+
+ /* execute gateway address change callback function */
+ if (netdev->addr_callback)
+ {
+ netdev->addr_callback(netdev, NETDEV_CB_ADDR_GATEWAY);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device DNS server address.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param dns_server the new DNS server address
+ *
+ */
+void netdev_low_level_set_dns_server(struct netdev *netdev, uint8_t dns_num, const ip_addr_t *dns_server)
+{
+ unsigned int index;
+
+ assert(dns_server);
+
+ if (netdev == NULL)
+ {
+ return;
+ }
+ /* check DNS servers is exist */
+ for (index = 0; index < NETDEV_DNS_SERVERS_NUM; index++)
+ {
+ if (ip_addr_cmp(&(netdev->dns_servers[index]), dns_server))
+ {
+ return;
+ }
+ }
+
+ if (dns_num < NETDEV_DNS_SERVERS_NUM)
+ {
+ ip_addr_copy(netdev->dns_servers[dns_num], *dns_server);
+
+ /* execute DNS servers address change callback function */
+ if (netdev->addr_callback)
+ {
+ netdev->addr_callback(netdev, NETDEV_CB_ADDR_DNS_SERVER);
+ }
+ }
+}
+
+#ifdef NETDEV_USING_AUTO_DEFAULT
+/* Change to the first link_up network interface device automatically */
+static void netdev_auto_change_default(struct netdev *netdev)
+{
+ struct netdev *new_netdev = NULL;
+
+ if (rt_memcmp(netdev, netdev_default, sizeof(struct netdev)) == 0)
+ {
+ new_netdev = netdev_get_first_by_flags(NETDEV_FLAG_LINK_UP);
+ if (new_netdev)
+ {
+ netdev_set_default(new_netdev);
+ }
+ }
+}
+#endif /* NETDEV_USING_AUTO_DEFAULT */
+
+/**
+ * This function will set network interface device status.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param is_up the new status
+ */
+void netdev_low_level_set_status(struct netdev *netdev, _Bool is_up)
+{
+ if (netdev && netdev_is_up(netdev) != is_up)
+ {
+ if (is_up)
+ {
+ netdev->flags |= NETDEV_FLAG_UP;
+ }
+ else
+ {
+ netdev->flags &= ~NETDEV_FLAG_UP;
+
+#ifdef NETDEV_USING_AUTO_DEFAULT
+ /* change to the first link_up network interface device automatically */
+ netdev_auto_change_default(netdev);
+#endif /* NETDEV_USING_AUTO_DEFAULT */
+ }
+
+ /* execute network interface device status change callback function */
+ if (netdev->status_callback)
+ {
+ netdev->status_callback(netdev, is_up ? NETDEV_CB_STATUS_UP : NETDEV_CB_STATUS_DOWN);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device active link status.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param is_up the new link status
+ */
+void netdev_low_level_set_link_status(struct netdev *netdev, _Bool is_up)
+{
+ if (netdev && netdev_is_link_up(netdev) != is_up)
+ {
+ if (is_up)
+ {
+ netdev->flags |= NETDEV_FLAG_LINK_UP;
+
+#ifdef RT_USING_SAL
+ /* set network interface device flags to internet up */
+ if (netdev_is_up(netdev) && !ip_addr_isany(&(netdev->ip_addr)))
+ {
+ sal_check_netdev_internet_up(netdev);
+ }
+#endif /* RT_USING_SAL */
+ }
+ else
+ {
+ netdev->flags &= ~NETDEV_FLAG_LINK_UP;
+
+ /* set network interface device flags to internet down */
+ netdev->flags &= ~NETDEV_FLAG_INTERNET_UP;
+
+#ifdef NETDEV_USING_AUTO_DEFAULT
+ /* change to the first link_up network interface device automatically */
+ netdev_auto_change_default(netdev);
+#endif /* NETDEV_USING_AUTO_DEFAULT */
+ }
+
+ /* execute link status change callback function */
+ if (netdev->status_callback)
+ {
+ netdev->status_callback(netdev, is_up ? NETDEV_CB_STATUS_LINK_UP : NETDEV_CB_STATUS_LINK_DOWN);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device active internet status.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param is_up the new internet status
+ */
+void netdev_low_level_set_internet_status(struct netdev *netdev, _Bool is_up)
+{
+ if (netdev && netdev_is_internet_up(netdev) != is_up)
+ {
+ if (is_up)
+ {
+ netdev->flags |= NETDEV_FLAG_INTERNET_UP;
+ }
+ else
+ {
+ netdev->flags &= ~NETDEV_FLAG_INTERNET_UP;
+ }
+
+ /* execute network interface device status change callback function */
+ if (netdev->status_callback)
+ {
+ netdev->status_callback(netdev, is_up ? NETDEV_CB_STATUS_INTERNET_UP : NETDEV_CB_STATUS_INTERNET_DOWN);
+ }
+ }
+}
+
+/**
+ * This function will set network interface device DHCP status.
+ * @NOTE it can only be called in the network interface device driver.
+ *
+ * @param netdev the network interface device to change
+ * @param is_up the new DHCP status
+ */
+void netdev_low_level_set_dhcp_status(struct netdev *netdev, _Bool is_enable)
+{
+ if (netdev && netdev_is_dhcp_enabled(netdev) != is_enable)
+ {
+ if (is_enable)
+ {
+ netdev->flags |= NETDEV_FLAG_DHCP;
+ }
+ else
+ {
+ netdev->flags &= ~NETDEV_FLAG_DHCP;
+ }
+
+ /* execute DHCP status change callback function */
+ if (netdev->status_callback)
+ {
+ netdev->status_callback(netdev, is_enable ? NETDEV_CB_STATUS_DHCP_ENABLE : NETDEV_CB_STATUS_DHCP_DISABLE);
+ }
+ }
+}
+
+#ifdef RT_USING_FINSH
+
+#include
+
+#ifdef NETDEV_USING_IFCONFIG
+static void netdev_list_if(void)
+{
+#define NETDEV_IFCONFIG_MAC_MAX_LEN 6
+#define NETDEV_IFCONFIG_IMEI_MAX_LEN 8
+
+ rt_ubase_t index;
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+ struct netdev *cur_netdev_list = netdev_list;
+
+ if (cur_netdev_list == NULL)
+ {
+ rt_kprintf("ifconfig: network interface device list error.\n");
+ return;
+ }
+
+ for (node = &(cur_netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = rt_list_entry(node, struct netdev, list);
+
+ rt_kprintf("network interface device: %.*s%s\n",
+ XiZi_NAME_MAX, netdev->name,
+ (netdev == netdev_default) ? " (Default)" : "");
+ rt_kprintf("MTU: %d\n", netdev->mtu);
+
+ /* 6 - MAC address, 8 - IEMI */
+ if (netdev->hwaddr_len == NETDEV_IFCONFIG_MAC_MAX_LEN)
+ {
+ rt_kprintf("MAC: ");
+ for (index = 0; index < netdev->hwaddr_len; index++)
+ {
+ rt_kprintf("%02x ", netdev->hwaddr[index]);
+ }
+ }
+ else if (netdev->hwaddr_len == NETDEV_IFCONFIG_IMEI_MAX_LEN)
+ {
+ rt_kprintf("IMEI: ");
+ for (index = 0; index < netdev->hwaddr_len; index++)
+ {
+ /* two numbers are displayed at one time*/
+ if (netdev->hwaddr[index] < 10 && index != netdev->hwaddr_len - 1)
+ {
+ rt_kprintf("%02d", netdev->hwaddr[index]);
+ }
+ else
+ {
+ rt_kprintf("%d", netdev->hwaddr[index]);
+ }
+
+
+ }
+ }
+
+ rt_kprintf("\nFLAGS:");
+ if (netdev->flags & NETDEV_FLAG_UP) rt_kprintf(" UP");
+ else rt_kprintf(" DOWN");
+ if (netdev->flags & NETDEV_FLAG_LINK_UP) rt_kprintf(" LINK_UP");
+ else rt_kprintf(" LINK_DOWN");
+#ifdef SAL_INTERNET_CHECK
+ if (netdev->flags & NETDEV_FLAG_INTERNET_UP) rt_kprintf(" INTERNET_UP");
+ else rt_kprintf(" INTERNET_DOWN");
+#endif
+ if (netdev->flags & NETDEV_FLAG_DHCP) rt_kprintf(" DHCP_ENABLE");
+ else rt_kprintf(" DHCP_DISABLE");
+ if (netdev->flags & NETDEV_FLAG_ETHARP) rt_kprintf(" ETHARP");
+ if (netdev->flags & NETDEV_FLAG_BROADCAST) rt_kprintf(" BROADCAST");
+ if (netdev->flags & NETDEV_FLAG_IGMP) rt_kprintf(" IGMP");
+ rt_kprintf("\n");
+ rt_kprintf("ip address: %s\n", inet_ntoa(netdev->ip_addr));
+ rt_kprintf("gw address: %s\n", inet_ntoa(netdev->gw));
+ rt_kprintf("net mask : %s\n", inet_ntoa(netdev->netmask));
+
+#if NETDEV_IPV6
+ {
+ ip_addr_t *addr;
+ int i;
+
+ addr = &netdev->ip6_addr[0];
+
+ if (!ip_addr_isany(addr))
+ {
+ rt_kprintf("ipv6 link-local: %s %s\n", inet_ntoa(*addr),
+ !ip_addr_isany(addr) ? "VALID" : "INVALID");
+
+ for (i = 1; i < NETDEV_IPV6_NUM_ADDRESSES; i++)
+ {
+ addr = &netdev->ip6_addr[i];
+ rt_kprintf("ipv6[%d] address: %s %s\n", i, inet_ntoa(*addr),
+ !ip_addr_isany(addr) ? "VALID" : "INVALID");
+ }
+ }
+ }
+#endif /* NETDEV_IPV6 */
+
+ for (index = 0; index < NETDEV_DNS_SERVERS_NUM; index++)
+ {
+ rt_kprintf("dns server #%d: %s\n", index, inet_ntoa(netdev->dns_servers[index]));
+ }
+
+ if (xizi_slist_next(node))
+ {
+ rt_kprintf("\n");
+ }
+ }
+}
+
+static void netdev_set_if(char* netdev_name, char* ip_addr, char* gw_addr, char* nm_addr)
+{
+ struct netdev *netdev = NULL;
+ ip_addr_t addr;
+
+ netdev = netdev_get_by_name(netdev_name);
+ if (netdev == NULL)
+ {
+ rt_kprintf("bad network interface device name(%s).\n", netdev_name);
+ return;
+ }
+
+ /* set IP address */
+ if ((ip_addr != NULL) && inet_aton(ip_addr, &addr))
+ {
+ netdev_set_ipaddr(netdev, &addr);
+ }
+
+ /* set gateway address */
+ if ((gw_addr != NULL) && inet_aton(gw_addr, &addr))
+ {
+ netdev_set_gw(netdev, &addr);
+ }
+
+ /* set netmask address */
+ if ((nm_addr != NULL) && inet_aton(nm_addr, &addr))
+ {
+ netdev_set_netmask(netdev, &addr);
+ }
+}
+
+int netdev_ifconfig(int argc, char **argv)
+{
+ if (argc == 1)
+ {
+ netdev_list_if();
+ }
+ else if (argc == 5)
+ {
+ rt_kprintf("config : %s\n", argv[1]);
+ rt_kprintf("IP addr: %s\n", argv[2]);
+ rt_kprintf("Gateway: %s\n", argv[3]);
+ rt_kprintf("netmask: %s\n", argv[4]);
+ netdev_set_if(argv[1], argv[2], argv[3], argv[4]);
+ }
+ else
+ {
+ rt_kprintf("bad parameter! e.g: ifconfig e0 192.168.1.30 192.168.1.1 255.255.255.0\n");
+ }
+
+ return 0;
+}
+MSH_CMD_EXPORT_ALIAS(netdev_ifconfig, ifconfig, list the information of all network interfaces);
+#endif /* NETDEV_USING_IFCONFIG */
+
+#ifdef NETDEV_USING_PING
+int netdev_cmd_ping(char* target_name, char *netdev_name, rt_uint32_t times, rt_size_t size)
+{
+#define NETDEV_PING_DATA_SIZE 32
+/** ping receive timeout - in milliseconds */
+#define NETDEV_PING_RECV_TIMEO (2 * RT_TICK_PER_SECOND)
+/** ping delay - in milliseconds */
+#define NETDEV_PING_DELAY (1 * RT_TICK_PER_SECOND)
+/* check netdev ping options */
+#define NETDEV_PING_IS_COMMONICABLE(netdev) \
+ ((netdev) && (netdev)->ops && (netdev)->ops->ping && \
+ netdev_is_up(netdev) && netdev_is_link_up(netdev)) \
+
+ struct netdev *netdev = NULL;
+ struct netdev_ping_resp ping_resp;
+ rt_uint32_t index;
+ int ret = 0;
+
+ if (size == 0)
+ {
+ size = NETDEV_PING_DATA_SIZE;
+ }
+
+ if (netdev_name != NULL)
+ {
+ netdev = netdev_get_by_name(netdev_name);
+ if (netdev == NULL)
+ {
+ netdev = netdev_default;
+ rt_kprintf("ping: not found specified netif, using default netdev %s.\n", netdev->name);
+ }
+ }
+
+ if (NETDEV_PING_IS_COMMONICABLE(netdev_default))
+ {
+ /* using default network interface device for ping */
+ netdev = netdev_default;
+ }
+ else
+ {
+ /* using first internet up status network interface device */
+ netdev = netdev_get_first_by_flags(NETDEV_FLAG_LINK_UP);
+ if (netdev == NULL)
+ {
+ rt_kprintf("ping: not found available network interface device.\n");
+ return -1;
+ }
+ else if (netdev->ops == NULL || netdev->ops->ping == NULL)
+ {
+ rt_kprintf("ping: network interface device(%s) not support ping feature.\n", netdev->name);
+ return -1;
+ }
+ else if (!netdev_is_up(netdev) || !netdev_is_link_up(netdev))
+ {
+ rt_kprintf("ping: network interface device(%s) status error.\n", netdev->name);
+ return -1;
+ }
+ }
+
+ for (index = 0; index < times; index++)
+ {
+ int delay_tick = 0;
+ rt_tick_t start_tick = 0;
+
+ memset(&ping_resp, 0x00, sizeof(struct netdev_ping_resp));
+ start_tick = rt_tick_get();
+ ret = netdev->ops->ping(netdev, (const char *)target_name, size, NETDEV_PING_RECV_TIMEO, &ping_resp);
+ if (ret == -RT_ETIMEOUT)
+ {
+ rt_kprintf("ping: from %s icmp_seq=%d timeout\n",
+ (ip_addr_isany(&(ping_resp.ip_addr))) ? target_name : inet_ntoa(ping_resp.ip_addr), index);
+ }
+ else if (ret == -1)
+ {
+ rt_kprintf("ping: unknown %s %s\n",
+ (ip_addr_isany(&(ping_resp.ip_addr))) ? "host" : "address",
+ (ip_addr_isany(&(ping_resp.ip_addr))) ? target_name : inet_ntoa(ping_resp.ip_addr));
+ }
+ else
+ {
+ if (ping_resp.ttl == 0)
+ {
+ rt_kprintf("%d bytes from %s icmp_seq=%d time=%d ms\n",
+ ping_resp.data_len, inet_ntoa(ping_resp.ip_addr), index, ping_resp.ticks);
+ }
+ else
+ {
+ rt_kprintf("%d bytes from %s icmp_seq=%d ttl=%d time=%d ms\n",
+ ping_resp.data_len, inet_ntoa(ping_resp.ip_addr), index, ping_resp.ttl, ping_resp.ticks);
+ }
+ }
+
+ /* if the response time is more than NETDEV_PING_DELAY, no need to delay */
+ delay_tick = ((rt_tick_get() - start_tick) > NETDEV_PING_DELAY) || (index == times) ? 0 : NETDEV_PING_DELAY;
+ rt_thread_delay(delay_tick);
+ }
+
+ return 0;
+}
+
+int netdev_ping(int argc, char **argv)
+{
+ if (argc == 1)
+ {
+ rt_kprintf("Please input: ping [netdev name] \n");
+ }
+ else if (argc == 2)
+ {
+ netdev_cmd_ping(argv[1], NULL, 4, 0);
+ }
+ else if (argc == 3)
+ {
+ netdev_cmd_ping(argv[1], argv[2], 4, 0);
+ }
+
+ return 0;
+}
+MSH_CMD_EXPORT_ALIAS(netdev_ping, ping, ping network host);
+#endif /* NETDEV_USING_IFCONFIG */
+
+static void netdev_list_dns(void)
+{
+ unsigned int index = 0;
+ struct netdev *netdev = NULL;
+ xizi_slist_t *node = NULL;
+
+ for (node = &(netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = rt_list_entry(node, struct netdev, list);
+
+ rt_kprintf("network interface device: %.*s%s\n",
+ XiZi_NAME_MAX, netdev->name,
+ (netdev == netdev_default)?" (Default)":"");
+
+ for(index = 0; index < NETDEV_DNS_SERVERS_NUM; index++)
+ {
+ rt_kprintf("dns server #%d: %s\n", index, inet_ntoa(netdev->dns_servers[index]));
+ }
+
+ if (xizi_slist_next(node))
+ {
+ rt_kprintf("\n");
+ }
+ }
+}
+
+static void netdev_set_dns(char *netdev_name, uint8_t dns_num, char *dns_server)
+{
+ struct netdev *netdev = NULL;
+ ip_addr_t dns_addr;
+
+ netdev = netdev_get_by_name(netdev_name);
+ if (netdev == NULL)
+ {
+ rt_kprintf("bad network interface device name(%s).\n", netdev_name);
+ return;
+ }
+
+ inet_aton(dns_server, &dns_addr);
+ if (netdev_set_dns_server(netdev, dns_num, &dns_addr) == 0)
+ {
+ rt_kprintf("set network interface device(%s) dns server #%d: %s\n", netdev_name, dns_num, dns_server);
+ }
+}
+
+int netdev_dns(int argc, char **argv)
+{
+ if (argc == 1)
+ {
+ netdev_list_dns();
+ }
+ else if (argc == 3)
+ {
+ netdev_set_dns(argv[1], 0, argv[2]);
+ }
+ else if (argc == 4)
+ {
+ netdev_set_dns(argv[1], atoi(argv[2]), argv[3]);
+ }
+ else
+ {
+ rt_kprintf("bad parameter! input: dns [dns_num] \n");
+ return -1;
+ }
+
+ return 0;
+}
+MSH_CMD_EXPORT_ALIAS(netdev_dns, dns, list and set the information of dns);
+#ifdef NETDEV_USING_NETSTAT
+static void netdev_cmd_netstat(void)
+{
+ xizi_slist_t *node = NULL;
+ struct netdev *netdev = NULL;
+ struct netdev *cur_netdev_list = netdev_list;
+
+ if (cur_netdev_list == NULL)
+ {
+ rt_kprintf("netstat: network interface device list error.\n");
+ return;
+ }
+
+ for (node = &(cur_netdev_list->list); node; node = xizi_slist_next(node))
+ {
+ netdev = rt_list_entry(node, struct netdev, list);
+
+ if (netdev && netdev->ops && netdev->ops->netstat)
+ {
+ break;
+ }
+ }
+
+ if (netdev->ops->netstat != NULL)
+ {
+ netdev->ops->netstat(netdev);
+ }
+ else
+ {
+ rt_kprintf("netstat: this command is not supported!\n");
+ }
+}
+
+int netdev_netstat(int argc, char **argv)
+{
+ if (argc != 1)
+ {
+ rt_kprintf("Please input: netstat \n");
+ }
+ else
+ {
+ netdev_cmd_netstat();
+ }
+
+ return 0;
+}
+MSH_CMD_EXPORT_ALIAS(netdev_netstat, netstat, list the information of TCP / IP);
+#endif /* NETDEV_USING_NETSTAT */
+
+#endif /* RT_USING_FINSH */
diff --git a/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev_ipaddr.c b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev_ipaddr.c
new file mode 100755
index 000000000..bb0f51a91
--- /dev/null
+++ b/Ubiquitous/XiZi/resources/ethernet/LwIP/netdev/src/netdev_ipaddr.c
@@ -0,0 +1,619 @@
+/*
+ * Copyright (c) 2006-2021, RT-Thread Development Team
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Change Logs:
+ * Date Author Notes
+ * 2022-8-22 zhaoyun for XiZi
+ */
+#include
+#include "../include/netdev_ipaddr.h"
+#include "../include/netdev.h"
+
+#include
+#include "sys_arch.h"
+#include
+#include "lwip/sys.h"
+
+/* Here for now until needed in other places in lwIP */
+#ifndef isprint
+#define in_range(c, lo, up) ((uint8_t)c >= lo && (uint8_t)c <= up)
+#define isprint(c) in_range(c, 0x20, 0x7f)
+#define isdigit(c) in_range(c, '0', '9')
+#define isxdigit(c) (isdigit(c) || in_range(c, 'a', 'f') || in_range(c, 'A', 'F'))
+#define islower(c) in_range(c, 'a', 'z')
+#define isspace(c) (c == ' ' || c == '\f' || c == '\n' || c == '\r' || c == '\t' || c == '\v')
+#define xchar(i) ((i) < 10 ? '0' + (i) : 'A' + (i) - 10)
+#endif
+
+#if NETDEV_IPV4
+
+/**
+ * Check whether "cp" is a valid ascii representation
+ * of an Internet address and convert to a binary address.
+ * Returns 1 if the address is valid, 0 if not.
+ * This replaces inet_addr, the return value from which
+ * cannot distinguish between failure and a local broadcast address.
+ *
+ * @param cp IP address in ascii representation (e.g. "127.0.0.1")
+ * @param addr pointer to which to save the ip address in network order
+ * @return 1 if cp could be converted to addr, 0 on failure
+ */
+int netdev_ip4addr_aton(const char *cp, ip4_addr_t *addr)
+{
+ uint32_t val;
+ uint8_t base;
+ char c;
+ uint32_t parts[4];
+ uint32_t *pp = parts;
+
+ c = *cp;
+ for (;;)
+ {
+ /*
+ * Collect number up to ``.''.
+ * Values are specified as for C:
+ * 0x=hex, 0=octal, 1-9=decimal.
+ */
+ if (!isdigit(c))
+ {
+ return 0;
+ }
+ val = 0;
+ base = 10;
+ if (c == '0')
+ {
+ c = *++cp;
+ if (c == 'x' || c == 'X')
+ {
+ base = 16;
+ c = *++cp;
+ }
+ else
+ {
+ base = 8;
+ }
+ }
+ for (;;)
+ {
+ if (isdigit(c))
+ {
+ val = (val * base) + (uint32_t) (c - '0');
+ c = *++cp;
+ }
+ else if (base == 16 && isxdigit(c))
+ {
+ val = (val << 4) | (uint32_t) (c + 10 - (islower(c) ? 'a' : 'A'));
+ c = *++cp;
+ }
+ else
+ {
+ break;
+ }
+ }
+ if (c == '.')
+ {
+ /*
+ * Internet format:
+ * a.b.c.d
+ * a.b.c (with c treated as 16 bits)
+ * a.b (with b treated as 24 bits)
+ */
+ if (pp >= parts + 3)
+ {
+ return 0;
+ }
+ *pp++ = val;
+ c = *++cp;
+ }
+ else
+ {
+ break;
+ }
+ }
+ /*
+ * Check for trailing characters.
+ */
+ if (c != '\0' && !isspace(c))
+ {
+ return 0;
+ }
+ /*
+ * Concoct the address according to
+ * the number of parts specified.
+ */
+ switch (pp - parts + 1)
+ {
+
+ case 0:
+ return 0; /* initial nondigit */
+
+ case 1: /* a -- 32 bits */
+ break;
+
+ case 2: /* a.b -- 8.24 bits */
+ if (val > 0xffffffUL)
+ {
+ return 0;
+ }
+ if (parts[0] > 0xff)
+ {
+ return 0;
+ }
+ val |= parts[0] << 24;
+ break;
+
+ case 3: /* a.b.c -- 8.8.16 bits */
+ if (val > 0xffff)
+ {
+ return 0;
+ }
+ if ((parts[0] > 0xff) || (parts[1] > 0xff))
+ {
+ return 0;
+ }
+ val |= (parts[0] << 24) | (parts[1] << 16);
+ break;
+
+ case 4: /* a.b.c.d -- 8.8.8.8 bits */
+ if (val > 0xff)
+ {
+ return 0;
+ }
+ if ((parts[0] > 0xff) || (parts[1] > 0xff) || (parts[2] > 0xff))
+ {
+ return 0;
+ }
+ val |= (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8);
+ break;
+ default:
+ assert(0);
+ break;
+ }
+ if (addr)
+ {
+ ip4_addr_set_u32(addr, htonl(val));
+ }
+ return 1;
+}
+
+/**
+ * Same as ipaddr_ntoa, but reentrant since a user-supplied buffer is used.
+ *
+ * @param addr ip address in network order to convert
+ * @param buf target buffer where the string is stored
+ * @param buflen length of buf
+ * @return either pointer to buf which now holds the ASCII
+ * representation of addr or NULL if buf was too small
+ */
+char *netdev_ip4addr_ntoa_r(const ip4_addr_t *addr, char *buf, int buflen)
+{
+ uint32_t s_addr;
+ char inv[3];
+ char *rp;
+ uint8_t *ap;
+ uint8_t rem;
+ uint8_t n;
+ uint8_t i;
+ int len = 0;
+
+ s_addr = ip4_addr_get_u32(addr);
+
+ rp = buf;
+ ap = (uint8_t *) &s_addr;
+ for (n = 0; n < 4; n++)
+ {
+ i = 0;
+ do
+ {
+ rem = *ap % (uint8_t) 10;
+ *ap /= (uint8_t) 10;
+ inv[i++] = (char) ('0' + rem);
+ } while (*ap);
+ while (i--)
+ {
+ if (len++ >= buflen)
+ {
+ return NULL;
+ }
+ *rp++ = inv[i];
+ }
+ if (len++ >= buflen)
+ {
+ return NULL;
+ }
+ *rp++ = '.';
+ ap++;
+ }
+ *--rp = 0;
+ return buf;
+}
+
+
+/**
+ * Convert numeric IP address into decimal dotted ASCII representation.
+ * returns ptr to static buffer; not reentrant!
+ *
+ * @param addr ip address in network order to convert
+ * @return pointer to a global static (!) buffer that holds the ASCII
+ * representation of addr
+ */
+char *netdev_ip4addr_ntoa(const ip4_addr_t *addr)
+{
+ static char str[IP4ADDR_STRLEN_MAX];
+ return netdev_ip4addr_ntoa_r(addr, str, IP4ADDR_STRLEN_MAX);
+}
+
+
+/**
+ * Ascii internet address interpretation routine.
+ * The value returned is in network order.
+ *
+ * @param cp IP address in ascii representation (e.g. "127.0.0.1")
+ * @return ip address in network order
+ */
+in_addr_t netdev_ipaddr_addr(const char *cp)
+{
+ ip4_addr_t val;
+
+ if (netdev_ip4addr_aton(cp, &val)) {
+ return ip4_addr_get_u32(&val);
+ }
+ return (IPADDR_NONE);
+}
+
+#endif /* NETDEV_IPV4 */
+
+
+#if NETDEV_IPV6
+
+RT_WEAK const struct in6_addr in6addr_any = IN6ADDR_ANY_INIT;
+
+/**
+ * Check whether "cp" is a valid ascii representation
+ * of an IPv6 address and convert to a binary address.
+ * Returns 1 if the address is valid, 0 if not.
+ *
+ * @param cp IPv6 address in ascii representation (e.g. "FF01::1")
+ * @param addr pointer to which to save the ip address in network order
+ * @return 1 if cp could be converted to addr, 0 on failure
+ */
+int
+netdev_ip6addr_aton(const char *cp, ip6_addr_t *addr)
+{
+ uint32_t addr_index, zero_blocks, current_block_index, current_block_value;
+ const char *s;
+
+ /* Count the number of colons, to count the number of blocks in a "::" sequence
+ zero_blocks may be 1 even if there are no :: sequences */
+ zero_blocks = 8;
+ for (s = cp; *s != 0; s++)
+ {
+ if (*s == ':')
+ {
+ zero_blocks--;
+ }
+ else if (!isxdigit(*s))
+ {
+ break;
+ }
+ }
+
+ /* parse each block */
+ addr_index = 0;
+ current_block_index = 0;
+ current_block_value = 0;
+ for (s = cp; *s != 0; s++)
+ {
+ if (*s == ':')
+ {
+ if (addr)
+ {
+ if (current_block_index & 0x1)
+ {
+ addr->addr[addr_index++] |= current_block_value;
+ }
+ else
+ {
+ addr->addr[addr_index] = current_block_value << 16;
+ }
+ }
+ current_block_index++;
+ current_block_value = 0;
+ if (current_block_index > 7)
+ {
+ /* address too long! */
+ return 0;
+ }
+ if (s[1] == ':')
+ {
+ if (s[2] == ':')
+ {
+ /* invalid format: three successive colons */
+ return 0;
+ }
+ s++;
+ /* "::" found, set zeros */
+ while (zero_blocks > 0)
+ {
+ zero_blocks--;
+ if (current_block_index & 0x1)
+ {
+ addr_index++;
+ }
+ else
+ {
+ if (addr)
+ {
+ addr->addr[addr_index] = 0;
+ }
+ }
+ current_block_index++;
+ if (current_block_index > 7)
+ {
+ /* address too long! */
+ return 0;
+ }
+ }
+ }
+ }
+ else if (isxdigit(*s))
+ {
+ /* add current digit */
+ current_block_value = (current_block_value << 4) +
+ (isdigit(*s) ? (uint32_t)(*s - '0') : (uint32_t)(10 + (islower(*s) ? *s - 'a' : *s - 'A')));
+ }
+ else
+ {
+ /* unexpected digit, space? CRLF? */
+ break;
+ }
+ }
+
+ if (addr)
+ {
+ if (current_block_index & 0x1)
+ {
+ addr->addr[addr_index++] |= current_block_value;
+ }
+ else
+ {
+ addr->addr[addr_index] = current_block_value << 16;
+ }
+ }
+
+ /* convert to network byte order. */
+ if (addr)
+ {
+ for (addr_index = 0; addr_index < 4; addr_index++)
+ {
+ addr->addr[addr_index] = htonl(addr->addr[addr_index]);
+ }
+ }
+
+ if (current_block_index != 7)
+ {
+ return 0;
+ }
+
+ return 1;
+}
+
+/**
+ * Same as ipaddr_ntoa, but reentrant since a user-supplied buffer is used.
+ *
+ * @param addr ip6 address in network order to convert
+ * @param buf target buffer where the string is stored
+ * @param buflen length of buf
+ * @return either pointer to buf which now holds the ASCII
+ * representation of addr or NULL if buf was too small
+ */
+char *
+netdev_ip6addr_ntoa_r(const ip6_addr_t *addr, char *buf, int buflen)
+{
+ uint32_t current_block_index, current_block_value, next_block_value;
+ int32_t i;
+ uint8_t zero_flag, empty_block_flag;
+
+ i = 0;
+ empty_block_flag = 0; /* used to indicate a zero chain for "::' */
+
+ for (current_block_index = 0; current_block_index < 8; current_block_index++)
+ {
+ /* get the current 16-bit block */
+ current_block_value = htonl(addr->addr[current_block_index >> 1]);
+ if ((current_block_index & 0x1) == 0)
+ {
+ current_block_value = current_block_value >> 16;
+ }
+ current_block_value &= 0xffff;
+
+ /* Check for empty block. */
+ if (current_block_value == 0)
+ {
+ if (current_block_index == 7 && empty_block_flag == 1)
+ {
+ /* special case, we must render a ':' for the last block. */
+ buf[i++] = ':';
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ break;
+ }
+ if (empty_block_flag == 0)
+ {
+ /* generate empty block "::", but only if more than one contiguous zero block,
+ * according to current formatting suggestions RFC 5952. */
+ next_block_value = htonl(addr->addr[(current_block_index + 1) >> 1]);
+ if ((current_block_index & 0x1) == 0x01)
+ {
+ next_block_value = next_block_value >> 16;
+ }
+ next_block_value &= 0xffff;
+ if (next_block_value == 0)
+ {
+ empty_block_flag = 1;
+ buf[i++] = ':';
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ continue; /* move on to next block. */
+ }
+ }
+ else if (empty_block_flag == 1)
+ {
+ /* move on to next block. */
+ continue;
+ }
+ }
+ else if (empty_block_flag == 1)
+ {
+ /* Set this flag value so we don't produce multiple empty blocks. */
+ empty_block_flag = 2;
+ }
+
+ if (current_block_index > 0)
+ {
+ buf[i++] = ':';
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ }
+
+ if ((current_block_value & 0xf000) == 0)
+ {
+ zero_flag = 1;
+ }
+ else
+ {
+ buf[i++] = xchar(((current_block_value & 0xf000) >> 12));
+ zero_flag = 0;
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ }
+
+ if (((current_block_value & 0xf00) == 0) && (zero_flag))
+ {
+ /* do nothing */
+ }
+ else
+ {
+ buf[i++] = xchar(((current_block_value & 0xf00) >> 8));
+ zero_flag = 0;
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ }
+
+ if (((current_block_value & 0xf0) == 0) && (zero_flag))
+ {
+ /* do nothing */
+ }
+ else
+ {
+ buf[i++] = xchar(((current_block_value & 0xf0) >> 4));
+ zero_flag = 0;
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ }
+
+ buf[i++] = xchar((current_block_value & 0xf));
+ if (i >= buflen)
+ {
+ return NULL;
+ }
+ }
+
+ buf[i] = 0;
+
+ return buf;
+}
+
+/**
+ * Convert numeric IPv6 address into ASCII representation.
+ * returns ptr to static buffer; not reentrant!
+ *
+ * @param addr ip6 address in network order to convert
+ * @return pointer to a global static (!) buffer that holds the ASCII
+ * representation of addr
+ */
+char *
+netdev_ip6addr_ntoa(const ip6_addr_t *addr)
+{
+ static char str[40];
+ return netdev_ip6addr_ntoa_r(addr, str, 40);
+}
+
+#endif /* NETDEV_IPV6 */
+
+const char *
+netdev_inet_ntop(int af, const void *src, char *dst, int32_t size)
+{
+#define AF_INET 2
+//#define AF_INET6 10
+
+ const char *ret = NULL;
+ int size_int = (int)size;
+ if (size_int < 0)
+ {
+ return NULL;
+ }
+ switch (af)
+ {
+#if NETDEV_IPV4
+ case AF_INET:
+ return netdev_ip4addr_ntoa_r((const ip4_addr_t *)src, dst, size_int);
+#endif
+#if NETDEV_IPV6
+ case AF_INET6:
+ return netdev_ip6addr_ntoa_r((const ip6_addr_t *)src, dst, size_int);
+#endif
+ default:
+ break;
+ }
+ return ret;
+}
+
+int
+netdev_inet_pton(int af, const char *src, void *dst)
+{
+#define AF_INET 2
+//#define AF_INET6 10
+
+ int err;
+ switch (af)
+ {
+#if NETDEV_IPV4
+ case AF_INET:
+ err = netdev_ip4addr_aton(src, (ip4_addr_t *)dst);
+ break;
+#endif
+#if NETDEV_IPV6
+ case AF_INET6:
+ {
+ /* convert into temporary variable since ip6_addr_t might be larger
+ than in6_addr when scopes are enabled */
+ ip6_addr_t addr;
+ err = netdev_ip6addr_aton(src, &addr);
+ if (err)
+ {
+ rt_memcpy(dst, &addr.addr, sizeof(addr.addr));
+ }
+ break;
+ }
+#endif
+ default:
+ err = -1;
+ break;
+ }
+ return err;
+}