📄 multihoposcilloscopec.nc
字号:
/*
* Copyright (c) 2006 Intel Corporation
* All rights reserved.
*
* This file is distributed under the terms in the attached INTEL-LICENSE
* file. If you do not find these files, copies can be found by writing to
* Intel Research Berkeley, 2150 Shattuck Avenue, Suite 1300, Berkeley, CA,
* 94704. Attention: Intel License Inquiry.
*/
/**
* MultihopOscilloscope demo application using the collection layer.
* See README.txt file in this directory and TEP 119: Collection.
*
* @author David Gay
* @author Kyle Jamieson
*/
#include "Timer.h"
#include "MultihopOscilloscope.h"
module MultihopOscilloscopeC @safe(){
uses {
// Interfaces for initialization:
interface Boot;
interface SplitControl as RadioControl;
interface SplitControl as SerialControl;
interface StdControl as RoutingControl;
// Interfaces for communication, multihop and serial:
interface Send;
interface Receive as Snoop;
interface Receive;
interface AMSend as SerialSend;
interface CollectionPacket;
interface RootControl;
interface Queue<message_t *> as UARTQueue;
interface Pool<message_t> as UARTMessagePool;
// Miscalleny:
interface Timer<TMilli>;
interface Read<uint16_t>;
interface Leds;
}
}
implementation {
task void uartSendTask();
static void startTimer();
static void fatal_problem();
static void report_problem();
static void report_sent();
static void report_received();
uint8_t uartlen;
message_t sendbuf;
message_t uartbuf;
bool sendbusy=FALSE, uartbusy=FALSE;
/* Current local state - interval, version and accumulated readings */
oscilloscope_t local;
uint8_t reading; /* 0 to NREADINGS */
/* When we head an Oscilloscope message, we check it's sample count. If
it's ahead of ours, we "jump" forwards (set our count to the received
count). However, we must then suppress our next count increment. This
is a very simple form of "time" synchronization (for an abstract
notion of time). */
bool suppress_count_change;
//
// On bootup, initialize radio and serial communications, and our
// own state variables.
//
event void Boot.booted() {
local.interval = DEFAULT_INTERVAL;
local.id = TOS_NODE_ID;
local.version = 0;
// Beginning our initialization phases:
if (call RadioControl.start() != SUCCESS)
fatal_problem();
if (call RoutingControl.start() != SUCCESS)
fatal_problem();
}
event void RadioControl.startDone(error_t error) {
if (error != SUCCESS)
fatal_problem();
if (sizeof(local) > call Send.maxPayloadLength())
fatal_problem();
if (call SerialControl.start() != SUCCESS)
fatal_problem();
}
event void SerialControl.startDone(error_t error) {
if (error != SUCCESS)
fatal_problem();
// This is how to set yourself as a root to the collection layer:
if (local.id % 500 == 0)
call RootControl.setRoot();
startTimer();
}
static void startTimer() {
if (call Timer.isRunning()) call Timer.stop();
call Timer.startPeriodic(local.interval);
reading = 0;
}
event void RadioControl.stopDone(error_t error) { }
event void SerialControl.stopDone(error_t error) { }
//
// Only the root will receive messages from this interface; its job
// is to forward them to the serial uart for processing on the pc
// connected to the sensor network.
//
event message_t*
Receive.receive(message_t* msg, void *payload, uint8_t len) {
oscilloscope_t* in = (oscilloscope_t*)payload;
oscilloscope_t* out;
if (uartbusy == FALSE) {
out = (oscilloscope_t*)call SerialSend.getPayload(&uartbuf, sizeof(oscilloscope_t));
if (len != sizeof(oscilloscope_t) || out == NULL) {
return msg;
}
else {
memcpy(out, in, sizeof(oscilloscope_t));
}
uartlen = sizeof(oscilloscope_t);
post uartSendTask();
} else {
// The UART is busy; queue up messages and service them when the
// UART becomes free.
message_t *newmsg = call UARTMessagePool.get();
if (newmsg == NULL) {
// drop the message on the floor if we run out of queue space.
report_problem();
return msg;
}
//Serial port busy, so enqueue.
out = (oscilloscope_t*)call SerialSend.getPayload(newmsg, sizeof(oscilloscope_t));
if (out == NULL) {
return msg;
}
memcpy(out, in, sizeof(oscilloscope_t));
if (call UARTQueue.enqueue(newmsg) != SUCCESS) {
// drop the message on the floor and hang if we run out of
// queue space without running out of queue space first (this
// should not occur).
call UARTMessagePool.put(newmsg);
fatal_problem();
return msg;
}
}
return msg;
}
task void uartSendTask() {
if (call SerialSend.send(0xffff, &uartbuf, uartlen) != SUCCESS) {
report_problem();
} else {
uartbusy = TRUE;
}
}
event void SerialSend.sendDone(message_t *msg, error_t error) {
uartbusy = FALSE;
if (call UARTQueue.empty() == FALSE) {
// We just finished a UART send, and the uart queue is
// non-empty. Let's start a new one.
message_t *queuemsg = call UARTQueue.dequeue();
if (queuemsg == NULL) {
fatal_problem();
return;
}
memcpy(&uartbuf, queuemsg, sizeof(message_t));
if (call UARTMessagePool.put(queuemsg) != SUCCESS) {
fatal_problem();
return;
}
post uartSendTask();
}
}
//
// Overhearing other traffic in the network.
//
event message_t*
Snoop.receive(message_t* msg, void* payload, uint8_t len) {
oscilloscope_t *omsg = payload;
report_received();
// If we receive a newer version, update our interval.
if (omsg->version > local.version) {
local.version = omsg->version;
local.interval = omsg->interval;
startTimer();
}
// If we hear from a future count, jump ahead but suppress our own
// change.
if (omsg->count > local.count) {
local.count = omsg->count;
suppress_count_change = TRUE;
}
return msg;
}
/* At each sample period:
- if local sample buffer is full, send accumulated samples
- read next sample
*/
event void Timer.fired() {
if (reading == NREADINGS) {
if (!sendbusy) {
oscilloscope_t *o = (oscilloscope_t *)call Send.getPayload(&sendbuf, sizeof(oscilloscope_t));
if (o == NULL) {
fatal_problem();
return;
}
memcpy(o, &local, sizeof(local));
if (call Send.send(&sendbuf, sizeof(local)) == SUCCESS)
sendbusy = TRUE;
else
report_problem();
}
reading = 0;
/* Part 2 of cheap "time sync": increment our count if we didn't
jump ahead. */
if (!suppress_count_change)
local.count++;
suppress_count_change = FALSE;
}
if (call Read.read() != SUCCESS)
fatal_problem();
}
event void Send.sendDone(message_t* msg, error_t error) {
if (error == SUCCESS)
report_sent();
else
report_problem();
sendbusy = FALSE;
}
event void Read.readDone(error_t result, uint16_t data) {
if (result != SUCCESS) {
data = 0xffff;
report_problem();
}
local.readings[reading++] = data;
}
// Use LEDs to report various status issues.
static void fatal_problem() {
call Leds.led0On();
call Leds.led1On();
call Leds.led2On();
call Timer.stop();
}
static void report_problem() { call Leds.led0Toggle(); }
static void report_sent() { call Leds.led1Toggle(); }
static void report_received() { call Leds.led2Toggle(); }
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -