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

📄 cc2420controlp.nc

📁 tinyos-2.x.rar
💻 NC
字号:
/*
 * Copyright (c) 2005-2006 Arch Rock Corporation
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 * - 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.
 * - Neither the name of the Arch Rock Corporation 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 COPYRIGHT HOLDERS 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
 * ARCHED ROCK OR ITS 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
 */

/**
 * @author Jonathan Hui <jhui@archrock.com>
 * @author David Moss
 * @author Urs Hunkeler (ReadRssi implementation)
 * @author Jan Hauer <hauer@tkn.tu-berlin.de>
 * @version $Revision: 1.5 $ $Date: 2009/05/05 16:56:49 $
 */

#include "Timer.h"
#include "AM.h"
#include "TKN154_PIB.h"

module CC2420ControlP {

  provides interface Init;
  provides interface Resource;
  provides interface CC2420Config;
  provides interface CC2420Power;

  uses interface Alarm<T62500hz,uint32_t> as StartupAlarm;
  uses interface GeneralIO as CSN;
  uses interface GeneralIO as RSTN;
  uses interface GeneralIO as VREN;
  uses interface GpioInterrupt as InterruptCCA;
  uses interface GeneralIO as FIFO;

  uses interface CC2420Ram as IEEEADR;
  uses interface CC2420Register as FSCTRL;
  uses interface CC2420Register as IOCFG0;
  uses interface CC2420Register as IOCFG1;
  uses interface CC2420Register as MDMCTRL0;
  uses interface CC2420Register as MDMCTRL1;
  uses interface CC2420Register as RXCTRL1;
  uses interface CC2420Register as RSSI;
  uses interface CC2420Register as RXFIFO_REGISTER;
  uses interface CC2420Strobe as SNOP;
  uses interface CC2420Strobe as SRXON;
  uses interface CC2420Strobe as SRFOFF;
  uses interface CC2420Strobe as SXOSCOFF;
  uses interface CC2420Strobe as SXOSCON;
  uses interface CC2420Strobe as SACKPEND;
  uses interface CC2420Strobe as SFLUSHRX;
  uses interface CC2420Register as TXCTRL;
  uses interface AMPacket;
  uses interface Resource as SpiResource;
  uses interface FrameUtility;
}

implementation {

  typedef enum {
    S_VREG_STOPPED,
    S_VREG_STARTING,
    S_VREG_STARTED,
    S_XOSC_STARTING,
    S_XOSC_STARTED,
  } cc2420_control_state_t;

  uint8_t m_channel;
  uint16_t m_pan;
  uint16_t m_short_addr;
  bool autoAckEnabled;
  bool hwAutoAckDefault;
  bool addressRecognition;
  bool acceptReservedFrames;
  bool m_isPanCoord;
  uint8_t m_CCAMode;
  uint8_t m_txPower;

  bool m_needsSync;

  norace cc2420_control_state_t m_state = S_VREG_STOPPED;
  
  /***************** Prototypes ****************/

  void writeFsctrl();
  void writeMdmctrl0();
  void writeId();
  void writeTxPower();

  /***************** Init Commands ****************/
  command error_t Init.init() {
    call CSN.makeOutput();
    call RSTN.makeOutput();
    call VREN.makeOutput();
    autoAckEnabled = TRUE;
    hwAutoAckDefault = TRUE;
    addressRecognition = TRUE;
    acceptReservedFrames = FALSE;
    m_needsSync = FALSE;
    return SUCCESS;
  }

  /***************** Resource Commands ****************/
  /* This module never actively requests the SPI resource,
   * instead the caller MUST request the SPI through this module
   * before it calls any of the provided commands and it must
   * release it afterwards (the caller can call multiple  
   * commands in this module before it releases the SPI, though).
   */ 
  async command error_t Resource.immediateRequest() {
    error_t error = call SpiResource.immediateRequest();
    if ( error == SUCCESS ) {
/*      call CSN.clr();*/
    }
    return error;
  }

  async command error_t Resource.request() {
    return call SpiResource.request();
  }

  async command uint8_t Resource.isOwner() {
    return call SpiResource.isOwner();
  }

  async command error_t Resource.release() {
    atomic {
/*      call CSN.set();*/
      return call SpiResource.release();
    }
  }

  event void SpiResource.granted() {
/*    call CSN.clr();*/
    signal Resource.granted();
  }

  /***************** CC2420Power Commands ****************/
  async command error_t CC2420Power.startVReg() {
    atomic {
      if ( m_state != S_VREG_STOPPED ) {
        return FAIL;
      }
      m_state = S_VREG_STARTING;
    }
    call VREN.set();
    call StartupAlarm.start( CC2420_TIME_VREN * 2 ); // JH: changed from 32khz jiffies
    return SUCCESS;
  }

  async command error_t CC2420Power.stopVReg() {
    m_state = S_VREG_STOPPED;
    call RSTN.clr();
    call VREN.clr();
    call RSTN.set();
    return SUCCESS;
  }

  async command error_t CC2420Power.startOscillator() {
    atomic {
      if ( m_state != S_VREG_STARTED ) {
        return FAIL;
      }
        
      m_state = S_XOSC_STARTING;
      call CSN.set();
      call CSN.clr();
      call IOCFG1.write( CC2420_SFDMUX_XOSC16M_STABLE << 
                         CC2420_IOCFG1_CCAMUX );
                         
      call InterruptCCA.enableRisingEdge();
      call SXOSCON.strobe();
      
      call IOCFG0.write( ( 1 << CC2420_IOCFG0_FIFOP_POLARITY ) |
          ( 127 << CC2420_IOCFG0_FIFOP_THR ) );
                         
      writeFsctrl();
      writeMdmctrl0();
  
      call RXCTRL1.write( ( 1 << CC2420_RXCTRL1_RXBPF_LOCUR ) |
          ( 1 << CC2420_RXCTRL1_LOW_LOWGAIN ) |
          ( 1 << CC2420_RXCTRL1_HIGH_HGM ) |
          ( 1 << CC2420_RXCTRL1_LNA_CAP_ARRAY ) |
          ( 1 << CC2420_RXCTRL1_RXMIX_TAIL ) |
          ( 1 << CC2420_RXCTRL1_RXMIX_VCM ) |
          ( 2 << CC2420_RXCTRL1_RXMIX_CURRENT ) );
      call CSN.set();
    }
    return SUCCESS;
  }


  async command error_t CC2420Power.stopOscillator() {
    atomic {
      if ( m_state != S_XOSC_STARTED ) {
        return FAIL;
      }
      m_state = S_VREG_STARTED;
      call CSN.set();
      call CSN.clr();
      call SXOSCOFF.strobe();
      call CSN.set();
    }
    return SUCCESS;
  }

  async command error_t CC2420Power.rxOn() {
    atomic {
      if ( !call SpiResource.isOwner() )
        return FAIL;
      call CSN.set();
      call CSN.clr();
      call SRXON.strobe();
      call SACKPEND.strobe();  // JH: ACKs need the pending bit set
      call CSN.set();
    }
    return SUCCESS;
  }

  async command error_t CC2420Power.rfOff() {
    atomic {
      if ( !call SpiResource.isOwner() )
        return FAIL;
      call CSN.set();
      call CSN.clr();
      call SRFOFF.strobe();
      call CSN.set();
    }
    return SUCCESS;
  }

  async command error_t CC2420Power.flushRxFifo()
  {
    uint16_t dummy;
    atomic {
      if ( !call SpiResource.isOwner() )
        return FAIL;
      if ( call FIFO.get() ){ // check if there is something in the RXFIFO
        // SFLUSHRX: "Flush the RX FIFO buffer and reset the demodulator. 
        // Always read at least one byte from the RXFIFO before 
        // issuing the SFLUSHRX command strobe" (CC2420 Datasheet)
        call CSN.clr();
        call RXFIFO_REGISTER.read(&dummy); // reading the byte
        call CSN.set();
        call CSN.clr();
        // "SFLUSHRX command strobe should be issued twice to ensure 
        // that the SFD pin goes back to its idle state." (CC2420 Datasheet)
        call SFLUSHRX.strobe();
        call SFLUSHRX.strobe();
        call CSN.set();
      }
    }
    return SUCCESS;
  }
  
  /***************** CC2420Config Commands ****************/
  command uint8_t CC2420Config.getChannel() {
    atomic return m_channel;
  }

  command void CC2420Config.setChannel( uint8_t channel ) {
    atomic {
      m_needsSync = TRUE;
      m_channel = channel;
    }
  }

  async command uint16_t CC2420Config.getShortAddr() {
    atomic return m_short_addr;
  }

  command void CC2420Config.setShortAddr( uint16_t addr ) {
    atomic {
      m_needsSync = TRUE;
      m_short_addr = addr;
    }
  }

  async command uint16_t CC2420Config.getPanAddr() {
    atomic return m_pan;
  }

  command void CC2420Config.setPanAddr( uint16_t pan ) {
    atomic {
      m_needsSync = TRUE;
      m_pan = pan;
    }
  }

  async command bool CC2420Config.getPanCoordinator() {
    atomic return m_isPanCoord;
  }

  command void CC2420Config.setPanCoordinator( bool pCoord ) {
    atomic {
      m_needsSync = TRUE;
      m_isPanCoord = pCoord;
    }
  }

  command void CC2420Config.setPromiscuousMode(bool on)
  {
    atomic {
      m_needsSync = TRUE;
      if (on){
        addressRecognition = FALSE;
        acceptReservedFrames = TRUE;
        autoAckEnabled = FALSE;
      } else {
        addressRecognition = TRUE;
        acceptReservedFrames = FALSE;
        autoAckEnabled = TRUE;
      }
    }
  }

  async command bool CC2420Config.isPromiscuousModeEnabled()
  {
    return acceptReservedFrames;
  }

  async command uint8_t CC2420Config.getCCAMode()
  {
    atomic return m_CCAMode;
  }

  command void CC2420Config.setCCAMode(uint8_t mode)
  {
    atomic {
      m_needsSync = TRUE;
      m_CCAMode = mode;
    }
  }

  async command uint8_t CC2420Config.getTxPower()
  {
    atomic return m_txPower;
  }

  command void CC2420Config.setTxPower(uint8_t txPower)
  {
    atomic {
      m_needsSync = TRUE;
      m_txPower = txPower;
    }
  }

  async command bool CC2420Config.needsSync(){
    atomic return m_needsSync;
  }

  /**
   * Sync must be called to commit software parameters configured on
   * the microcontroller (through the CC2420Config interface) to the
   * CC2420 radio chip.
   */
  async command error_t CC2420Config.sync() {
    atomic {
      if ( !call SpiResource.isOwner() )
        return FAIL;
      if (m_needsSync){
        call CSN.set();
        call CSN.clr();
        call SRFOFF.strobe();
        call CSN.set();
        call CSN.clr();
        writeFsctrl();
        writeMdmctrl0();
        writeTxPower();
        call CSN.set();
        call CSN.clr();
        writeId();
        call CSN.set();
        m_needsSync = FALSE;
      }
    }
    return SUCCESS;
  }

  /***************** ReadRssi Commands ****************/
  
  async command error_t CC2420Power.rssi(int8_t *rssi) {
    uint16_t data;
    cc2420_status_t status;
    atomic {
      if ( !call SpiResource.isOwner() )
        return FAIL;
      call CSN.set();
      call CSN.clr();
      status = call RSSI.read(&data);
      call CSN.set();
      if ((status & 0x02)){
        *rssi = (data & 0x00FF);
        return SUCCESS;
      } else
        return FAIL;
    }
  }
  
  /***************** StartupAlarm Events ****************/
  async event void StartupAlarm.fired() {
    if ( m_state == S_VREG_STARTING ) {
      m_state = S_VREG_STARTED;
      call RSTN.clr();
      call RSTN.set();
      signal CC2420Power.startVRegDone();
    }
  }

  /***************** InterruptCCA Events ****************/
  async event void InterruptCCA.fired() {
    m_state = S_XOSC_STARTED;
    call InterruptCCA.disable();
    call CSN.set();
    call CSN.clr();
    call IOCFG1.write( 0 );
    writeId();
    call CSN.set();
    signal CC2420Power.startOscillatorDone();
  }
  
  /***************** Functions ****************/
  /**
   * Write teh FSCTRL register
   */
  void writeFsctrl() {
    uint8_t channel;
    
    atomic {
      channel = m_channel;
    }
    
    call FSCTRL.write( ( 1 << CC2420_FSCTRL_LOCK_THR ) |
          ( ( (channel - 11)*5+357 ) << CC2420_FSCTRL_FREQ ) );
  }

  /**
   * Write the MDMCTRL0 register
   */
  void writeMdmctrl0() {
    atomic {
      uint8_t _acceptReservedFrames = (acceptReservedFrames ? 1: 0);
      uint8_t _panCoord = (m_isPanCoord ? 1: 0);
      uint8_t _addressRecognition = (addressRecognition ? 1: 0);
      uint8_t _autoAck = ((autoAckEnabled && hwAutoAckDefault) ? 1 : 0);
      call MDMCTRL0.write( ( _acceptReservedFrames << CC2420_MDMCTRL0_RESERVED_FRAME_MODE ) |
          ( _panCoord << CC2420_MDMCTRL0_PAN_COORDINATOR ) | 
          ( _addressRecognition << CC2420_MDMCTRL0_ADR_DECODE ) |
          ( 2 << CC2420_MDMCTRL0_CCA_HYST ) |
          ( m_CCAMode << CC2420_MDMCTRL0_CCA_MOD ) |
          ( 1 << CC2420_MDMCTRL0_AUTOCRC ) |
          ( _autoAck << CC2420_MDMCTRL0_AUTOACK ) |
          ( 2 << CC2420_MDMCTRL0_PREAMBLE_LENGTH ) );
    }
    // Jon Green:
    // MDMCTRL1.CORR_THR is defaulted to 20 instead of 0 like the datasheet says
    // If we add in changes to MDMCTRL1, be sure to include this fix.
  }
  
  /**
   * Write the IEEEADR register
   */
  void writeId() {
    uint16_t bcnAccept = 0;
    nxle_uint16_t id[ 6 ];

    atomic {
      call FrameUtility.copyLocalExtendedAddressLE((uint8_t*) &id);
      id[ 4 ] = m_pan;
      id[ 5 ] = m_short_addr;
    }
    if (m_pan == 0xFFFF)
      bcnAccept = 1;
    
    
    call IOCFG0.write( (bcnAccept << CC2420_IOCFG0_BCN_ACCEPT) |
        ( 1 << CC2420_IOCFG0_FIFOP_POLARITY ) |
          ( 127 << CC2420_IOCFG0_FIFOP_THR ) );
    // ext.adr, PANID and short adr are located at consecutive addresses in RAM
    call IEEEADR.write(0, (uint8_t*)&id, sizeof(id));
  }

  void writeTxPower(){
    call TXCTRL.write( 
        ( 2 << CC2420_TXCTRL_TXMIXBUF_CUR ) |
        ( 3 << CC2420_TXCTRL_PA_CURRENT ) |
        ( 1 << CC2420_TXCTRL_RESERVED ) |
      ( (m_txPower & 0x1F) << CC2420_TXCTRL_PA_LEVEL ) );
  }
}

⌨️ 快捷键说明

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