/* 6LoWPAN MAC - layer 2 implementations */ #include #include #include "sixlowmac.h" #include "sixlowip.h" #include "sixlownd.h" #include "sixlowpan.h" #include #include "thread.h" #include "msg.h" #include "radio/radio.h" #include "transceiver.h" #include "vtimer.h" #include "ieee802154_frame.h" char radio_stack_buffer[RADIO_STACK_SIZE]; msg msg_q[RADIO_RCV_BUF_SIZE]; uint8_t snd_buffer[RADIO_SND_BUF_SIZE][PAYLOAD_SIZE]; uint8_t r_src_addr; uint8_t buf[PAYLOAD_SIZE]; uint16_t packet_length; static uint8_t macdsn; //static uint8_t macbsn; static radio_packet_t p; static msg mesg; int transceiver_type; static transceiver_command_t tcmd; uint8_t get_radio_address(void){ uint16_t addr; tcmd.transceivers = transceiver_type; tcmd.data = &addr; mesg.content.ptr = (char*)&tcmd; mesg.type = GET_ADDRESS; msg_send_receive(&mesg,&mesg,transceiver_pid); return (uint8_t)addr; } void set_radio_address(uint8_t addr){ tcmd.transceivers = transceiver_type; tcmd.data = &addr; mesg.content.ptr = (char*)&tcmd; mesg.type = SET_ADDRESS; msg_send_receive(&mesg, &mesg, transceiver_pid); } void set_radio_channel(uint8_t channel){ tcmd.transceivers = transceiver_type; tcmd.data = &channel; mesg.content.ptr = (char*)&tcmd; mesg.type = SET_CHANNEL; msg_send_receive(&mesg,&mesg,transceiver_pid); } void switch_to_rx(void){ mesg.type = SWITCH_RX; mesg.content.ptr = (char*) &tcmd; tcmd.transceivers = TRANSCEIVER_CC1100; msg_send(&mesg, transceiver_pid, 1); } void init_802154_short_addr(ieee_802154_short_t *saddr){ saddr->uint8[0] = 0; saddr->uint8[1] = get_radio_address(); } ieee_802154_long_t* mac_get_eui(ipv6_addr_t *ipaddr){ return ((ieee_802154_long_t *) &(ipaddr->uint8[8])); } void init_802154_long_addr(ieee_802154_long_t *laddr){ // 16bit Pan-ID:16-zero-bits:16-bit-short-addr = 48bit laddr->uint16[0] = IEEE_802154_PAN_ID; /* RFC 4944 Section 6 / RFC 2464 Section 4 */ laddr->uint8[0] ^= 0x02; laddr->uint8[2] = 0; laddr->uint8[3] = 0xFF; laddr->uint8[4] = 0xFE; laddr->uint8[5] = 0; laddr->uint8[6] = 0; laddr->uint8[7] = get_radio_address(); } void recv_ieee802154_frame(void){ msg m; radio_packet_t *p; uint8_t hdrlen, length; ieee802154_frame_t frame; msg_init_queue(msg_q, RADIO_RCV_BUF_SIZE); while (1) { msg_receive(&m); if (m.type == PKT_PENDING) { p = (radio_packet_t*) m.content.ptr; hdrlen = read_802154_frame(p->data, &frame, p->length); length = p->length - hdrlen; /* deliver packet to network(6lowpan)-layer */ input(frame.payload, length, (ieee_802154_long_t*)&frame.src_addr, (ieee_802154_long_t*)&frame.dest_addr); p->processing--; } else if (m.type == ENOBUFFER) { puts("Transceiver buffer full"); } else { puts("Unknown packet received"); } } } void set_ieee802154_fcf_values(ieee802154_frame_t *frame, uint8_t dest_mode, uint8_t src_mode){ frame->fcf.frame_type = IEEE_802154_DATA_FRAME; frame->fcf.sec_enb = 0; frame->fcf.frame_pend = 0; frame->fcf.ack_req = 0; frame->fcf.panid_comp = 0; frame->fcf.frame_ver = 0; frame->fcf.src_addr_m = src_mode; frame->fcf.dest_addr_m = dest_mode; } void set_ieee802154_frame_values(ieee802154_frame_t *frame){ // TODO: addresse aus ip paket auslesen und in frame einfuegen frame->dest_pan_id = IEEE_802154_PAN_ID; frame->src_pan_id = IEEE_802154_PAN_ID; frame->seq_nr = macdsn; macdsn++; } void send_ieee802154_frame(ieee_802154_long_t *addr, uint8_t *payload, uint8_t length, uint8_t mcast){ uint16_t daddr; r_src_addr = get_radio_address(); mesg.type = SND_PKT; mesg.content.ptr = (char*) &tcmd; tcmd.transceivers = transceiver_type; tcmd.data = &p; ieee802154_frame_t frame; memset(&frame, 0, sizeof(frame)); set_ieee802154_fcf_values(&frame, IEEE_802154_LONG_ADDR_M, IEEE_802154_LONG_ADDR_M); set_ieee802154_frame_values(&frame); /* ONLY FOR TESTING */ memcpy(&(frame.dest_addr[0]), &(addr->uint8[0]), 8); //frame.dest_addr[0] = *addr & 0xff; //frame.dest_addr[1] = (*addr >> 8) & 0xff; memcpy(&(frame.src_addr[0]), &(iface.laddr.uint8[0]), 8); //frame.src_addr[0] = 0; //frame.src_addr[1] = r_src_addr; /* check if destination address is NULL => broadcast */ //if(addr[0] == 0 && addr[1] == 0){ // frame.dest_addr[0] = 0xff; // frame.dest_addr[1] = 0xff; //} daddr = HTONS(addr->uint16[3]); //memcpy(&daddr, &addr->uint8[6], 2); //printf("blub: %02x\n", addr->uint8[6]); //uint8_t test = 30; frame.payload = payload; frame.payload_len = length; //printf("length: %x\n",frame.payload_len); uint8_t hdrlen = get_802154_hdr_len(&frame); memset(&buf,0,PAYLOAD_SIZE); init_802154_frame(&frame,(uint8_t*)&buf); memcpy(&buf[hdrlen],frame.payload,frame.payload_len); p.length = hdrlen + frame.payload_len; if(mcast == 0){ p.dst = daddr; } else { p.dst = 0; } // TODO: geeignete ring-bufferung nötig p.data = buf; //p.data = snd_buffer[i % RADIO_SND_BUF_SIZE]; msg_send(&mesg, transceiver_pid, 1); hwtimer_wait(5000); //switch_to_rx(); } void sixlowmac_init(transceiver_type_t type){ int recv_pid = thread_create(radio_stack_buffer, RADIO_STACK_SIZE, PRIORITY_MAIN-2, CREATE_STACKTEST, recv_ieee802154_frame , "radio"); transceiver_type = type; transceiver_init(transceiver_type); transceiver_start(); transceiver_register(type, recv_pid); macdsn = rand() % 256; }