/* * Copyright (c) 2008, Swedish Institute of Computer Science. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the Institute nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * This file is part of the Contiki operating system. * * $Id: sicslowmac.c,v 1.1 2008/10/14 09:43:40 adamdunkels Exp $ */ /** * \file * Example glue code between the existing MAC code and the * Contiki mac interface * * \author * Adam Dunkels * Eric Gnoske * Blake Leverett * * \addtogroup rf230mac */ #include #include #include #include #include #include #include "net/rime/rimebuf.h" #include "mac.h" #include "frame.h" #include "radio.h" #include "tcpip.h" #include "sicslowmac.h" #include "sicslowpan.h" #include "ieee-15-4-manager.h" /* Macros */ #define DEBUG 0 #define MAX_EVENTS 10 #if DEBUG #define PRINTF(...) printf(__VA_ARGS__) #else #define PRINTF(...) #endif /* Globals */ static mac_driver_t mac_driver; static mac_driver_t *pmac_driver = &mac_driver; extern ieee_15_4_manager_t ieee15_4ManagerAddress; static parsed_frame_t *parsed_frame; const mac_driver_t sicslowmac_driver = { sicslowmac_dataRequest, /* read_packet, */ /* set_receive_function, */ /* on, */ /* off, */ }; static struct { uint8_t head; uint8_t tail; event_object_t event_object[MAX_EVENTS]; } event_queue; /* Prototypes */ static void setinput(void (*r)(const mac_driver_t *d)); static void (*pinput)(const mac_driver_t *r); /*---------------------------------------------------------------------------*/ /** * \brief Checks for any pending events in the queue. * * \return True if there is a pending event, else false. */ uint8_t mac_event_pending(void) { return (event_queue.head != event_queue.tail); } /*---------------------------------------------------------------------------*/ /** * \brief Puts an event into the queue of events. * * \param object is a pointer to the event to add to queue. */ void mac_put_event(event_object_t *object) { uint8_t newhead; if ((event_queue.head + 1) % MAX_EVENTS == event_queue.tail){ /* queue full, get outta here */ return; } newhead = event_queue.head; /* store in queue */ event_queue.event_object[newhead] = *object; /* calculate new head index */ newhead++; if (newhead >= MAX_EVENTS){ newhead = 0; } event_queue.head = newhead; } /*---------------------------------------------------------------------------*/ /** * \brief Pulls an event from the event queue. * Assumes that there is an event in the queue. See mac_event_pending(). * * \return Pointer to the event object, or NULL in the event of empty queue. */ event_object_t *mac_get_event(void) { event_object_t *object = NULL; volatile uint8_t newtail; newtail = event_queue.tail; object = &(event_queue.event_object[newtail]); /* calculate new tail */ newtail++; if (newtail >= MAX_EVENTS){ newtail = 0; } event_queue.tail = newtail; return(object); } void mac_pollhandler(void) { mac_task(NULL, NULL); } /*---------------------------------------------------------------------------*/ /** * \brief This is the main loop task for the MAC. Called by the * main application loop. */ void mac_task(process_event_t ev, process_data_t data) { /* check for event in queue */ event_object_t *event; if(mac_event_pending()){ event = mac_get_event(); /* Handle events from radio */ if (event){ if (event->event == MAC_EVENT_RX){ /* got a frame, find out with kind of frame */ parsed_frame = (parsed_frame_t *)event->data; if (parsed_frame->fcf->frameType == DATAFRAME){ sicslowmac_dataIndication(); } /* Frame no longer in use */ parsed_frame->in_use = false; } if (event->event == MAC_EVENT_DROPPED){ /* Frame was dropped */ printf("sicslowmac: Frame Dropped!\n"); } } } } /*---------------------------------------------------------------------------*/ static void setinput(void (*r)(const mac_driver_t *d)) { pinput = r; } /*---------------------------------------------------------------------------*/ void sicslowmac_dataIndication(void) { rimebuf_clear(); /* Finally, get the stuff into the rime buffer.... */ rimebuf_copyfrom(parsed_frame->payload, parsed_frame->payload_length); rimebuf_set_datalen(parsed_frame->payload_length); /* Change addresses to expected byte order */ byte_reverse((uint8_t *)parsed_frame->dest_addr, 8); byte_reverse((uint8_t *)parsed_frame->src_addr, 8); rimebuf_set_addr(RIMEBUF_ADDR_RECEIVER, (const rimeaddr_t *)parsed_frame->dest_addr); rimebuf_set_addr(RIMEBUF_ADDR_SENDER, (const rimeaddr_t *)parsed_frame->src_addr); PRINTF("sicslowmac: hand off frame to sicslowpan \n"); pinput(pmac_driver); } /*---------------------------------------------------------------------------*/ /** * \brief This is the implementation of the 15.4 MAC Data Request * primitive. * * \return Integer denoting success or failure. * \retval 0 Failure. * \retval 1 Success. * * The data request primitive creates the frame header based * on static and dynamic data. The static data will be refined * in phase II of the project. The frame payload and length are * retrieved from the rime buffer and rime length respectively. * * When the header and payload are assembled into the * frame_create_params structure, the frame is created * by a call to frame_tx_create and then transmited via * radio_send_data. */ /*---------------------------------------------------------------------------*/ int sicslowmac_dataRequest(void) { /* create structure to store result. */ frame_create_params_t params; frame_result_t result; /* Save the msduHandle in a global variable. */ msduHandle = rimebuf_attr(RIMEBUF_ATTR_PACKET_ID); /* Build the FCF. */ params.fcf.frameType = DATAFRAME; params.fcf.securityEnabled = false; params.fcf.framePending = false; params.fcf.ackRequired = rimebuf_attr(RIMEBUF_ATTR_RELIABLE); params.fcf.panIdCompression = false; /* Insert IEEE 802.15.4 (2003) version bit. */ params.fcf.frameVersion = IEEE802154_2003; /* Increment and set the data sequence number. */ params.seq = macDSN++; /* Complete the addressing fields. */ /** \todo For phase 1 the addresses are all long. We'll need a mechanism in the rime attributes to tell the mac to use long or short for phase 2. */ params.fcf.srcAddrMode = LONGADDRMODE; params.dest_pid = ieee15_4ManagerAddress.get_dst_panid(); /* * If the output address is NULL in the Rime buf, then it is broadcast * on the 802.15.4 network. */ if(rimeaddr_cmp(rimebuf_addr(RIMEBUF_ADDR_RECEIVER), &rimeaddr_null) ) { /* Broadcast requires short address mode. */ params.fcf.destAddrMode = SHORTADDRMODE; params.dest_pid = BROADCASTPANDID; params.dest_addr.addr16 = BROADCASTADDR; } else { /* Phase 1.5 - end nodes send to anyone? */ memcpy(¶ms.dest_addr, (uint8_t *)rimebuf_addr(RIMEBUF_ADDR_RECEIVER), LONG_ADDR_LEN); /* Hack to allow Ethernet to send us data, we have to limit addresses to have zero in upper */ /* two bytes */ /* params.dest_addr.addr64 &= 0x0000ffffffffffffUL; */ /* Phase 1 - end nodes only sends to pan coordinator node. */ /* params.dest_addr.addr64 = ieee15_4ManagerAddress.get_coord_long_addr(); */ params.fcf.destAddrMode = LONGADDRMODE; } /* Set the source PAN ID to the global variable. */ params.src_pid = ieee15_4ManagerAddress.get_src_panid(); /* * Set up the source address using only the long address mode for * phase 1. */ params.src_addr.addr64 = ieee15_4ManagerAddress.get_long_addr(); /* Copy the payload data. */ params.payload_len = rimebuf_datalen(); params.payload = rimebuf_dataptr(); /* Create transmission frame. */ frame_tx_create(¶ms, &result); /* Retry up to this many times to send the packet if radio is busy */ uint8_t retry_count = 3; while(retry_count) { PRINTF("sicslowmac: sending packet of length %d to radio, result:", result.length); /* Send data to radio. */ radio_status_t rv = radio_send_data(result.length, result.frame); if (rv == RADIO_SUCCESS) { PRINTF(" Success\n"); return 1; /* True says that the packet could be sent */ } if (rv != RADIO_WRONG_STATE) { PRINTF(" Failed\n"); return 0; } PRINTF(" Radio busy, retrying\n"); _delay_ms(10); //We have blocking delay here, it is safest this way retry_count--; } PRINTF("sicslowmac: Unable to send packet, dropped\n"); return 0; } /*---------------------------------------------------------------------------*/ /** * \brief Stub function that will be implemented in phase 2 to cause * end nodes to sleep. */ int mac_wake(void) { return 1; } /*---------------------------------------------------------------------------*/ /** * \brief Stub function that will be implemented in phase 2 to cause * end nodes to sleep. */ int mac_sleep(void) { return 1; } /*---------------------------------------------------------------------------*/ const mac_driver_t * sicslowmac_init(const struct radio_driver *d) { /* AD: commented out the radio_driver code for now.*/ /* radio = d; radio->set_receive_function(input_packet); radio->on();*/ return &sicslowmac_driver; } /*---------------------------------------------------------------------------*/ /** * \brief This is the implementation of the 15.4 MAC Reset Request * primitive. * \param setDefaultPIB True if the default PIB values should be set. * \return Integer denoting success or failure. * \retval 0 Failure. * \retval 1 Success. * * Sets all PIB values to default. */ void sicslowmac_resetRequest (bool setDefaultPIB) { if(setDefaultPIB){ /* initialize all of the MAC PIB variables to their default values */ macCoordShortAddress = 0xffff; macDSN = rand() % 256; macSrcPANId = SOURCE_PAN_ID; macDstPANId = DEST_PAN_ID; macShortAddress = 0xffff; /* Setup the address of this device by reading a stored address from eeprom. */ /** \todo This might be read from the serial eeprom onboard Raven. */ AVR_ENTER_CRITICAL_REGION(); eeprom_read_block ((void *)&macLongAddr, EEPROMMACADDRESS, 8); byte_reverse((uint8_t *) &macLongAddr, 8); AVR_LEAVE_CRITICAL_REGION(); } } /*---------------------------------------------------------------------------*/ PROCESS(mac_process, "802.15.4 MAC process"); PROCESS_THREAD(mac_process, ev, data) { PROCESS_POLLHANDLER(mac_pollhandler()); PROCESS_BEGIN(); radio_status_t return_value; /* init radio */ /** \todo: this screws up if calosc is set to TRUE, find out why? */ return_value = radio_init(false, NULL, NULL, NULL); #if DEBUG if (return_value == RADIO_SUCCESS) { printf("Radio init successful.\n"); } else { printf("Radio init failed with return: %d\n", return_value); } #endif radio_set_operating_channel(24); radio_use_auto_tx_crc(true); radio_set_trx_state(TRX_OFF); mac_init(); /* Set up MAC function pointers and sicslowpan callback. */ pmac_driver->set_receive_function = setinput; pmac_driver->send = sicslowmac_dataRequest; pmac_driver->wake = mac_wake; pmac_driver->sleep = mac_sleep; sicslowpan_init(pmac_driver); ieee_15_4_init(&ieee15_4ManagerAddress); radio_set_trx_state(RX_AACK_ON); while(1) { PROCESS_YIELD(); mac_task(ev, data); } PROCESS_END(); } void byte_reverse(uint8_t * bytes, uint8_t num) { uint8_t tempbyte; uint8_t i, j; i = 0; j = num - 1; while(i < j) { tempbyte = bytes[i]; bytes[i] = bytes[j]; bytes[j] = tempbyte; j--; i++; } return; }