1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/net/sixlowpan/sixlownd.c
2010-11-08 10:33:23 +01:00

163 lines
5.4 KiB
C

#include "sixlowip.h"
#include "sixlownd.h"
#include "sixlowmac.h"
#include <stdio.h>
#define PRINTF(...) printf(__VA_ARGS__)
#define PRINT6ADDR(addr) PRINTF("%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n", ((uint8_t *)addr)[0], ((uint8_t *)addr)[1], ((uint8_t *)addr)[2], ((uint8_t *)addr)[3], ((uint8_t *)addr)[4], ((uint8_t *)addr)[5], ((uint8_t *)addr)[6], ((uint8_t *)addr)[7], ((uint8_t *)addr)[8], ((uint8_t *)addr)[9], ((uint8_t *)addr)[10], ((uint8_t *)addr)[11], ((uint8_t *)addr)[12], ((uint8_t *)addr)[13], ((uint8_t *)addr)[14], ((uint8_t *)addr)[15])
static uint8_t option_field_length;
uint8_t ipv6_ext_hdr_len = 0;
uint16_t packet_length;
#define IP_BUFFER ((struct ipv6_hdr*)&buffer[LL_HEADER_LENGTH])
#define ICMP_BUFFER ((struct icmpv6_hdr*)&buffer[LLHDR_IPV6HDR_LENGTH])
/* fields after icmp header*/
#define RA_BUFFER ((struct router_advertisement*)&buffer[LLHDR_ICMPV6HDR_LENGTH])
#define IPH_LLH_BUFFER &buffer[LL_HEADER_LENGTH + IPV6_HEADER_LENGTH]
#define OPT_ST_FIELD_BUFFER ((struct option_src_trgt_lla*)&buffer[LLHDR_IPV6HDR_LENGTH + option_field_length])
#define OPT_MTU_BUFFER ((struct option_mtu*)&buffer[LLHDR_IPV6HDR_LENGTH + option_field_length])
uint8_t rs_count;
option_prefix_info *prefix; /** prefix list pointer */
/* send router solicitation message - RFC4861 section 4.1 */
void send_rs(void){
//uint8_t ipv6_ext_hdr_len = 0;
if(rs_count < MAX_RTR_SOL){
packet_length = 0;
ICMP_BUFFER->type = ICMP_RTR_SOL;
ICMP_BUFFER->code = 0;
IP_BUFFER->version_trafficclass = IPV6_VERSION;
IP_BUFFER->trafficclass_flowlabel = 0;
IP_BUFFER->flowlabel = 0;
IP_BUFFER->nextheader = ICMP_NEXTHEADER;
IP_BUFFER->hoplimit = NEIGHBOR_DISCOVERY_HOPLIMIT;
create_all_routers_mcast_addr(&IP_BUFFER->destaddr);
PRINT6ADDR(&IP_BUFFER->destaddr);
/* set payload length */
IP_BUFFER->length = ICMP_HEADER_LENGTH + RTR_SOL_LENGTH + SLLAO_OPT_LENGTH;
set_llao(&buffer[LLHDR_ICMPV6HDR_LENGTH + RTR_SOL_LENGTH], SLLAO_OPT_LENGTH);
ICMP_BUFFER->checksum = 0;
ICMP_BUFFER->checksum = ~chksum_calc(ICMP_NEXTHEADER);
printf("%x\n",ICMP_BUFFER->checksum);
rs_count++;
// sleep 4 sec
swtimer_usleep(RTR_SOL_INTERVAL * 1000000);
}
}
void recv_rs(void){
option_field_length = RTR_SOL_LENGTH;
uint8_t *llao;
/* get link layer address option from buf */
if(OPT_ST_FIELD_BUFFER->type == SLLAO_OPT_LENGTH){
llao = OPT_ST_FIELD_BUFFER;
}
if(llao != NULL){
// TODO: Neighbor lookup
}
send_ra();
}
void send_ra(ipv6_addr *addr){
IP_BUFFER->version_trafficclass = IPV6_VERSION;
IP_BUFFER->trafficclass_flowlabel = 0;
IP_BUFFER->flowlabel = 0;
IP_BUFFER->nextheader = ICMP_NEXTHEADER;
IP_BUFFER->hoplimit = NEIGHBOR_DISCOVERY_HOPLIMIT;
// not solicited
create_all_nodes_mcast_addr(&IP_BUFFER->destaddr);
ICMP_BUFFER->type = ICMP_RTR_ADV;
ICMP_BUFFER->code = 0;
// set current ttl
RA_BUFFER->hoplimit = MULTIHOP_HOPLIMIT;
// set M and O flag, last 6 bits are zero
RA_BUFFER->autoconfig_flags = (ND_M_FLAG << 7) | (ND_O_FLAG << 6);
RA_BUFFER->router_lifetime = MAX_RTR_ADV_INTERVAL * MAX_RTR_ADV;
RA_BUFFER->reachable_time = 0;
RA_BUFFER->retrans_timer = 0;
set_llao((uint8_t *)OPT_ST_FIELD_BUFFER,SLLAO_OPT_LENGTH);
/* set MTU options */
OPT_MTU_BUFFER->type = MTU_OPTION_TYPE;
OPT_MTU_BUFFER->length = MTU_OPTION_LENGTH;
OPT_MTU_BUFFER->reserved = 0;
// 1500 octets mtu
OPT_MTU_BUFFER->mtu = HTONL(1500);
/* set packet length */
packet_length = IPV6_HEADER_LENGTH + ICMP_HEADER_LENGTH +
RTR_ADV_LENGTH + SLLAO_OPT_LEN + MTU_OPTION_HDR_LENGTH;
/* set payload length field */
/* set prefix option */
for(){
}
IP_BUFFER->length = packet_length - IPV6_HEADER_LENGTH;
/* calculate checksum */
ICMP_BUFFER->checksum = 0;
ICMP_BUFFER->checksum = ~chksum_calc(ICMP_NEXTHEADER);
}
/* link-layer address option - RFC4861 section 4.6.1/ RFC4944 8. */
void set_llao(uint8_t *llao, uint8_t type){
llao[ND_OPT_TYPE] = type;
if(SLLAO_OPT_LENGTH == 16){
// 802.15.4 long address
llao[ND_OPT_LENGTH] = 2;
} else
// 16-bit short address
llao[ND_OPT_LENGTH] = 1;
// get link layer address
link_layer_addr *mac = get_eui(&IP_BUFFER->srcaddr);
//get_link_layer_addr_from_ipaddr(&IP_BUFFER->srcaddr,&mac);
memcpy(&llao[ND_OPT_DATA], mac, SIXLOWPAN_IPV6_LL_ADDR_LEN);
// padding (all zeros) - 48bit
memset(&llao[ND_OPT_DATA + SIXLOWPAN_IPV6_LL_ADDR_LEN], 0,
SLLAO_OPT_LENGTH - SIXLOWPAN_IPV6_LL_ADDR_LEN - 2);
}
/* pseudo-header checksum calculation - RFC4443 Section 2.3*/
uint16_t chksum_calc(uint8_t type){
uint16_t length = IP_BUFFER->length;
uint16_t sum = length + type;
uint8_t *addrptr = (uint8_t)&IP_BUFFER->srcaddr;
uint16_t addr_fields_length = (2 * sizeof(ipv6_addr));
while(addr_fields_length > 1){
sum += (addrptr[0] << 8) + addrptr[1];
addrptr += 2;
addr_fields_length -= 2;
}
if(addr_fields_length > 1){
sum += (addrptr[0] << 8) + 0;
}
uint8_t *bufptr = IPH_LLH_BUFFER;
while(length > 1){
sum += (bufptr[0] << 8) + bufptr[1];
bufptr += 2;
length -= 2;
}
if(length){
sum += (bufptr[0] << 8) + 0;
}
return sum;
}