mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
161 lines
5.1 KiB
C
161 lines
5.1 KiB
C
/*
|
|
* Copyright (C) 2019 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup tests
|
|
* @{
|
|
*
|
|
* @file
|
|
*
|
|
* @author Martine Lenders <m.lenders@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include "net/ipv6/addr.h"
|
|
#include "net/ipv6/hdr.h"
|
|
#include "net/sock/ip.h"
|
|
#include "net/sock/udp.h"
|
|
#include "net/gnrc.h"
|
|
#include "net/gnrc/ipv6/hdr.h"
|
|
#include "net/gnrc/pktdump.h"
|
|
#include "net/gnrc/udp.h"
|
|
#include "net/protnum.h"
|
|
#include "od.h"
|
|
#include "thread.h"
|
|
|
|
#define RECV_BUFFER_SIZE (128)
|
|
|
|
#define TEST_PORT (38664U)
|
|
#define TEST_LOCAL { 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01 }
|
|
#define TEST_REMOTE { 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02 }
|
|
#define TEST_PAYLOAD { 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef }
|
|
|
|
static const uint8_t _test_local[] = TEST_LOCAL;
|
|
static const uint8_t _test_remote[] = TEST_REMOTE;
|
|
static const uint8_t _test_payload[] = TEST_PAYLOAD;
|
|
|
|
static gnrc_netreg_entry_t _pktdump;
|
|
|
|
static char _addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
static uint8_t _buffer[128];
|
|
static sock_ip_t _ip_sock;
|
|
static sock_udp_t _udp_sock;
|
|
|
|
/* module is not compiled in, so provide this function for the test */
|
|
ipv6_hdr_t *gnrc_ipv6_get_header(gnrc_pktsnip_t *pkt)
|
|
{
|
|
gnrc_pktsnip_t *tmp = gnrc_pktsnip_search_type(pkt, GNRC_NETTYPE_IPV6);
|
|
if (tmp == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
assert(tmp->data != NULL);
|
|
assert(tmp->size >= sizeof(ipv6_hdr_t));
|
|
assert(ipv6_hdr_is(tmp->data));
|
|
|
|
return ((ipv6_hdr_t*) tmp->data);
|
|
}
|
|
|
|
static void _recv_udp(sock_udp_t *sock, sock_async_flags_t flags)
|
|
{
|
|
printf("UDP event triggered: %04X\n", flags);
|
|
if (flags & SOCK_ASYNC_MSG_RECV) {
|
|
sock_udp_ep_t remote;
|
|
ssize_t res;
|
|
|
|
if ((res = sock_udp_recv(sock, _buffer, sizeof(_buffer), 0,
|
|
&remote)) >= 0) {
|
|
printf("Received UDP packet from [%s]:%u:\n",
|
|
ipv6_addr_to_str(_addr_str, (ipv6_addr_t *)remote.addr.ipv6,
|
|
sizeof(_addr_str)),
|
|
remote.port);
|
|
od_hex_dump(_buffer, res, OD_WIDTH_DEFAULT);
|
|
}
|
|
}
|
|
if (flags & SOCK_ASYNC_MSG_SENT) {
|
|
puts("UDP message successfully sent");
|
|
}
|
|
}
|
|
|
|
static void _recv_ip(sock_ip_t *sock, sock_async_flags_t flags)
|
|
{
|
|
printf("IP event triggered: %04X\n", flags);
|
|
if (flags & SOCK_ASYNC_MSG_RECV) {
|
|
sock_ip_ep_t remote;
|
|
ssize_t res;
|
|
|
|
if ((res = sock_ip_recv(sock, _buffer, sizeof(_buffer), 0,
|
|
&remote)) >= 0) {
|
|
printf("Received IP packet from [%s]:\n",
|
|
ipv6_addr_to_str(_addr_str, (ipv6_addr_t *)remote.addr.ipv6,
|
|
sizeof(_addr_str)));
|
|
od_hex_dump(_buffer, res, OD_WIDTH_DEFAULT);
|
|
}
|
|
}
|
|
if (flags & SOCK_ASYNC_MSG_SENT) {
|
|
puts("IP message successfully sent");
|
|
}
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
gnrc_pktsnip_t *pkt;
|
|
sock_udp_ep_t local = SOCK_IPV6_EP_ANY;
|
|
sock_udp_ep_t remote = SOCK_IPV6_EP_ANY;
|
|
|
|
/* register for IPv6 to have a target */
|
|
gnrc_netreg_entry_init_pid(&_pktdump, GNRC_NETREG_DEMUX_CTX_ALL,
|
|
gnrc_pktdump_pid);
|
|
gnrc_netreg_register(GNRC_NETTYPE_IPV6, &_pktdump);
|
|
|
|
local.port = TEST_PORT;
|
|
sock_udp_create(&_udp_sock, &local, NULL, 0);
|
|
sock_ip_create(&_ip_sock, (sock_ip_ep_t *)&local, NULL, PROTNUM_UDP, 0);
|
|
|
|
|
|
/* XXX don't do it like this in production and use a proper `sock_async`
|
|
* frontend! This is just for testing. */
|
|
sock_udp_set_cb(&_udp_sock, _recv_udp);
|
|
sock_ip_set_cb(&_ip_sock, _recv_ip);
|
|
memcpy(remote.addr.ipv6, _test_remote, sizeof(_test_remote));
|
|
remote.port = TEST_PORT - 1;
|
|
|
|
sock_udp_send(&_udp_sock, _test_payload, sizeof(_test_payload), &remote);
|
|
sock_ip_send(&_ip_sock, _test_payload, sizeof(_test_payload),
|
|
PROTNUM_RESERVED, (sock_ip_ep_t *)&remote);
|
|
|
|
/* create packet to inject for reception */
|
|
pkt = gnrc_netif_hdr_build(NULL, 0, NULL, 0);
|
|
assert(pkt != NULL);
|
|
memset(pkt->data, 0, pkt->size);
|
|
pkt = gnrc_ipv6_hdr_build(pkt, (ipv6_addr_t *)&_test_remote,
|
|
(ipv6_addr_t *)&_test_local);
|
|
assert(pkt != NULL);
|
|
/* module is not compiled in, so set header type manually */
|
|
pkt->type = GNRC_NETTYPE_IPV6;
|
|
pkt = gnrc_udp_hdr_build(pkt, TEST_PORT - 1, TEST_PORT);
|
|
assert(pkt != NULL);
|
|
pkt = gnrc_pktbuf_add(pkt, _test_payload, sizeof(_test_payload),
|
|
GNRC_NETTYPE_UNDEF);
|
|
assert(pkt != NULL);
|
|
/* we dispatch twice, so hold one time */
|
|
gnrc_pktbuf_hold(pkt, 1);
|
|
|
|
/* trigger receive on UDP sock */
|
|
gnrc_netapi_dispatch_receive(GNRC_NETTYPE_UDP, TEST_PORT, pkt);
|
|
/* trigger receive on IP sock */
|
|
gnrc_netapi_dispatch_receive(GNRC_NETTYPE_IPV6, PROTNUM_UDP, pkt);
|
|
return 0;
|
|
}
|
|
|
|
/** @} */
|