⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 radio.c

📁 无线传感器网络操作系统源代码
💻 C
字号:
/*									tab:4 * * * "Copyright (c) 2000 and The Regents of the University  * of California.  All rights reserved. * * Permission to use, copy, modify, and distribute this software and its * documentation for any purpose, without fee, and without written agreement is * hereby granted, provided that the above copyright notice and the following * two paragraphs appear in all copies of this software. *  * IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY FOR * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES ARISING OUT * OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF THE UNIVERSITY OF * CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *  * THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY * AND FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS * ON AN "AS IS" BASIS, AND THE UNIVERSITY OF CALIFORNIA HAS NO OBLIGATION TO * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS." * * Authors:		Jason Hill, Philip Levis * Description:         New radio bit interface to hardware * Major Modifications: * * * This component performs bit level control over the RF Monolitics radio. * Addtionally, it controls the amount of time per bit by using TCNT1. * It exposes the radio timer for higher level component use. * * The bitrate or timer value passed to functions represent clock ticks on * the 4 MHz clock. Passing a timer of 0 to POWER_OFF will result in * interrupts being disabled until POWER_ON is called. * */#include "tos.h"#include "RADIO.h"#include "dbg.h"#include <stdlib.h>#define TOS_FRAME_TYPE RFM_frameTOS_FRAME_BEGIN(RFM_frame) {  char mode;  short rate;  char pendingChange;  event_t* radioTickEvent;}TOS_FRAME_END(RFM_frame);#define RADIO_RX_MODE        0#define RADIO_TX_MODE        1#define RADIO_OFF_MODE       2#define RX_PENDING      0x01#define TX_PENDING      0x02#define OFF_PENDING     0x04#define ON_PENDING      0x08#define BITRATE_PENDING 0x10#define RATE_LIMIT         0x0190/* This is a SIGNAL handler that timer1 generates to trigger this   component to sample on the radio */TOS_SIGNAL_HANDLER(SIG_OUTPUT_COMPARE1A, ()){  char in;  //dbg(DBG_RADIO, ("mote in state %d\n", NODE_NUM, VAR(mode)));  tos_state.rfm->stop_transmit(NODE_NUM);  if(VAR(mode) == RADIO_TX_MODE){    TOS_SIGNAL_EVENT(RADIO_TX_BIT_EVENT)();  }  else if(VAR(mode) == RADIO_RX_MODE){    in = tos_state.rfm->hears(NODE_NUM);    if (in) {dbg(DBG_RADIO, ("Received %x\n", in));}    TOS_SIGNAL_EVENT(RADIO_RX_BIT_EVENT)(in);  }  else if (VAR(mode) == RADIO_OFF_MODE) {    VAR(mode) = RADIO_RX_MODE;    in = tos_state.rfm->hears(NODE_NUM);    TOS_SIGNAL_EVENT(RADIO_RX_BIT_EVENT)(in);  }}/* This command sets timer1 to different sampling level */char TOS_COMMAND(RADIO_SET_BIT_RATE)(short rate){  event_t* event;  int time;  dbg(DBG_RADIO, ("RADIO: bit rate set to %i.\n", (int)rate));  cli();  if (VAR(mode) == RADIO_OFF_MODE) {    sei();    return 0;  }  else if (rate != VAR(rate)) {    if (VAR(radioTickEvent) != NULL) {      event_radiotick_invalidate(VAR(radioTickEvent));    }    event = (event_t*)malloc(sizeof(event_t));    dbg(DBG_MEM, ("Allocating radio clock bit event: 0x%x.\n", (unsigned int)event));        time = tos_state.tos_time + (rate / 2);    event_radiotick_create(event, NODE_NUM, time, (rate / 2));    TOS_queue_insert_event(event);        VAR(radioTickEvent) = event;      VAR(rate) = rate;    sei();  }  return 1;}/* This command sets the RADIO component (radio) to transmit bit "data" */char TOS_COMMAND(RADIO_TX_BIT)(char data){  char rval;  //if not in the transmit mote fail.  cli();  if(VAR(mode) != RADIO_TX_MODE) {    dbg(DBG_RADIO, ("RADIO: Trying to transmit in non-transmit state.\n"));    rval = 0;  }  //sent the output pin accordingly.  else {    tos_state.rfm->transmit(NODE_NUM, (char)(data & 0x01));    dbg(DBG_RADIO, ("RADIO:sending bit %x\n", data & 0x01));    rval = 1;  }    sei();  return rval;}/* * All of these mode-switching functions need to be executed with interrupts * disabled, as the values they modify are checked in the signal handler. * * Specifically, if VAR(mode) or VAR(rate) is modified, it must be done * with interrupts disabled. Also, as some of these operations mess * with the timing of interrupts, keeping them disabled for those operations * is a paranoid precaution. *//* This command sets the RADIO component (radio) into different power * mode */char TOS_COMMAND(RADIO_PWR_OFF)(short timer){  cli();  VAR(mode) = RADIO_OFF_MODE;  VAR(rate) = timer;    //turn off the RADIO chip.  CLR_RFM_CTL0_PIN();  CLR_RFM_CTL1_PIN();    if (timer == 0 && VAR(radioTickEvent) != NULL) {    // If timer is 0, disable interrupts    event_radiotick_invalidate(VAR(radioTickEvent));  }  else {  // Give wakeup time    TOS_CALL_COMMAND(RADIO_SET_BIT_RATE)(timer);  }    sei();  return 1;}/* This command sets the RADIO component (radio) into transmit mode */char TOS_COMMAND(RADIO_TX_MODE)(){  cli();  if(VAR(mode) == RADIO_OFF_MODE) {    sei();    return 0;  }  //set the RADIO chip to TX mode.  SET_RFM_CTL0_PIN();  CLR_RFM_CTL1_PIN();    dbg(DBG_RADIO, ("RADIO: set TX mode....\n"));    //record the current mode.  VAR(mode) = RADIO_TX_MODE;    sei();  return 1;}/* This command sets the RADIO component (radio) into receiving mode */char TOS_COMMAND(RADIO_RX_MODE)(){  cli();  if(VAR(mode) == RADIO_OFF_MODE) {    sei();    return 0;  }  //set the RADIO to RX mode.  SET_RFM_CTL0_PIN();  SET_RFM_CTL1_PIN();  CLR_RFM_TXD_PIN();    dbg(DBG_RADIO, ("RADIO: set RX mode....\n"));    //record the current mode.  VAR(mode) = RADIO_RX_MODE;  sei();  return 1;}// Turns the radio on into RX modechar TOS_COMMAND(RADIO_PWR_ON)(short bitrate) {  char val;  cli();  if (VAR(rate) == 0) { // radio was fully turned off; turn back on    sbi(TIMSK, OCIE1A);  }    VAR(mode) = RADIO_RX_MODE;  TOS_CALL_COMMAND(RADIO_RX_MODE)();  val = TOS_CALL_COMMAND(RADIO_SET_BIT_RATE)(bitrate);    sei();  return 1;}short TOS_COMMAND(RADIO_GET_BIT_RATE)(void) {  return VAR(rate);}/* Initialization of the Component */char TOS_COMMAND(RADIO_INIT)(){  //Reset to idle mode.  VAR(mode) = RADIO_RX_MODE;    //set the RADIO pins.  SET_RFM_CTL0_PIN();  SET_RFM_CTL1_PIN();  CLR_RFM_TXD_PIN();    cbi(TIMSK, OCIE1A); //clear interrupts  cbi(TIMSK, TICIE1); //clear interrupts  cbi(TIMSK, TOIE1);  //clear interrupts  cbi(TIMSK, OCIE1B); //clear interrupts  outp(0x09, TCCR1B); //scale the counter  outp(0x00, TCCR1A);  outp(0x00, OCR1AH); // set upper byte of comp reg.  outp(0xc8, OCR1AL); // set the lower byte compare  sbi(TIMSK, OCIE1A); // enable timer1 interupt  outp(0x00, TCNT1H); // clear current counter value  outp(0x00, TCNT1L); // clear current couter high byte value  TOS_CALL_COMMAND(RADIO_SET_BIT_RATE)(200);    dbg(DBG_BOOT, ("RADIO initialized\n"));  return 1;}void event_radiotick_handle(event_t* event,			    struct TOS_state* state) {    event_queue_t* queue = &(state->queue);  radio_tick_data_t* data = (radio_tick_data_t*)event->data;  if (data->valid) {    //dbg(DBG_RADIO, ("RADIO: tick event handled for mote %i at %lli with interval of %i.\n", event->mote, event->time, data->interval));        event->time = event->time + data->interval;    queue_insert_event(queue, event);        TOS_ISSUE_SIGNAL(SIG_OUTPUT_COMPARE1A)();  }  else {    //dbg(DBG_RADIO, ("RADIO: invalid tick event for mote %i at %lli discarded.\n", data->mote, event->time));    event->cleanup(event);  }}void event_radiotick_create(event_t* event, int mote, long long time, int interval) {  //int time = THIS_NODE.time;  clock_tick_data_t* data = (clock_tick_data_t*)malloc(sizeof(radio_tick_data_t));  dbg(DBG_MEM, ("Allocating radio clock bit event data: 0x%x.\n", (unsigned int)data));  data->interval = interval;  data->mote = mote;  data->valid = 1;    event->mote = mote;  event->data = data;  event->time = time;  event->handle = event_radiotick_handle;  event->cleanup = event_total_cleanup;  event->pause = 0;}void event_radiotick_invalidate(event_t* event) {  clock_tick_data_t* data = event->data;  data->valid = 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -