All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.tinyos.packet.Packetizer Maven / Gradle / Ivy

The newest version!
// $Id: Packetizer.java,v 1.14 2004/11/18 19:02:18 idgay Exp $

/*									tab:4
 * "Copyright (c) 2000-2003 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, the following
 * two paragraphs and the author 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."
 *
 * Copyright (c) 2002-2003 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.
 */
package net.tinyos.packet;

import net.tinyos.util.*;
import net.tinyos.message.*;
import java.io.*;
import java.util.*;

/**
 * The Packetizer class implements the new mote-PC protocol, using
 * a ByteSource for low-level I/O
 */
public class Packetizer extends AbstractSource implements Runnable {
    /* Protocol inspired by, but not identical to, RFC 1663.
     * There is currently no protocol establishment phase, and a single
     * byte ("packet type") to identify the kind/target/etc of each packet.
     *
     * The protocol is really, really not aiming for high performance.
     * 
     * There is however a hook for future extensions: implementations are
     * required to answer all unknown packet types with a P_UNKNOWN packet.
     *
     * To summarise the protocol:
     * - the two sides (A & B) are connected by a (potentially unreliable)
     *   byte stream
     * - the two sides exchange packets framed by 0x7e (SYNC_BYTE) bytes
     * - each packet has the form
     *       <16-bit crc>
     *   where the crc (see net.tinyos.util.Crc) covers the packet type
     *   and bytes 1..n
     * - bytes can be escaped by preceding them with 0x7d and their
     *   value xored with 0x20; 0x7d and 0x7e bytes must be escaped, 
     *   0x00 - 0x1f and 0x80-0x9f may be optionally escaped
     * - There are currently 5 packet types:
     *   P_PACKET_NO_ACK: A user-packet, with no ack required
     *   P_PACKET_ACK: A user-packet with a prefix byte, ack required. 
     *     The receiver must send a P_ACK packet with the prefix byte
     *     as its contents.
     *   P_ACK: ack for a previous P_PACKET_ACK packet
     *   P_UNKNOWN: unknown packet type received. On reception of an
     *     unknown packet type, the receicer must send a P_UNKNOWN packet,
     *     the first byte must be the unknown packet type.
     * - Packets that are greater than a (private) MTU are silently dropped.
     */
    final static boolean DEBUG = false;

    final static int SYNC_BYTE = 0x7e;
    final static int ESCAPE_BYTE = 0x7d;
    final static int MTU = 256;
    final static int ACK_TIMEOUT = 1000; // in milliseconds

    final static int P_ACK = 64;
    final static int P_PACKET_ACK = 65;
    final static int P_PACKET_NO_ACK = 66;
    final static int P_UNKNOWN = 255;

    private ByteSource io;
    private boolean inSync;
    private byte[] receiveBuffer = new byte[MTU];
    private int seqNo;

    // Packets are received by a separate thread and placed in a
    // per-packet-type queue. If received[x] is null, then x is an
    // unknown protocol (but P_UNKNOWN and P_PACKET_ACK are handled
    // specially)
    private Thread reader;
    private LinkedList[] received;

    /**
     * Packetizers are built using the makeXXX methods in BuildSource
     */
    Packetizer(String name, ByteSource io, int plat) {
	super(name);
	this.io = io;
	platform = plat;
	inSync = false;
	seqNo = 13;
	reader = new Thread(this);
	received = new LinkedList[256];
	received[P_ACK] = new LinkedList();
	received[P_PACKET_NO_ACK] = new LinkedList();
    }

    synchronized public void open(Messenger messages) throws IOException {
	super.open(messages);
	reader.start();
    }

    protected void openSource() throws IOException {
	io.open();
    }

    protected void closeSource() {
        io.close();
    }

    protected byte[] readProtocolPacket(int packetType, long deadline) throws IOException {
	LinkedList inPackets = received[packetType];

	// Wait for a packet on inPackets
	synchronized (inPackets) {
	    while (inPackets.isEmpty()) {
		long now = System.currentTimeMillis();
		if (deadline != 0 && now >= deadline) {
		    return null;
		}
		try {
		    inPackets.wait(deadline != 0 ? deadline - now : 0);
		}
		catch (InterruptedException e) {
		    throw new IOException("interrupted");
		}
	    }
	    return (byte [])inPackets.removeFirst();
	}
    }

    // Place a packet in its packet queue, or reject unknown packet
    // types (which don't have a queue)
    protected void pushProtocolPacket(int packetType, byte[] packet) {
	LinkedList inPackets = received[packetType];

	if (inPackets != null) {
	    synchronized (inPackets) {
		inPackets.add(packet);
		inPackets.notify();
	    }
	}
	else if (packetType != P_UNKNOWN) {
	    try {
		writeFramedPacket(P_UNKNOWN, packetType, ackPacket, 0);
	    }
	    catch (IOException e) { }
	    message(name + ": ignoring unknown packet type " +
		    Integer.toHexString(packetType));
	}
    }

    protected byte[] readSourcePacket() throws IOException {
	// Packetizer packet format is identical to PacketSource's
	return readProtocolPacket(P_PACKET_NO_ACK, 0);
    }

    // Write an ack-ed packet
    protected boolean writeSourcePacket(byte[] packet) throws IOException {
	seqNo++;
	writeFramedPacket(P_PACKET_ACK, seqNo, packet, packet.length);
	long deadline = System.currentTimeMillis() + ACK_TIMEOUT;
	for (;;) {
	    byte[] ack = readProtocolPacket(P_ACK, deadline);
	    if (ack == null) {
		if (DEBUG) {
		    message(name + ": ACK timed out");
		}
		return false;
	    }
	    if (ack[0] == (byte)seqNo) {
		if (DEBUG) {
		    message(name + ": Rcvd ACK");
		}
		return true;
	    }
	}
    }

    static private byte ackPacket[] = new byte[0];

    public void run() {
	try {
	    for (;;) {
		byte[] packet = readFramedPacket();
		int packetType = packet[0] & 0xff;
		int pdataOffset = 1;

		if (packetType == P_PACKET_ACK) {
		    // send ack
		    writeFramedPacket(P_ACK, packet[1], ackPacket, 0);
		    // And merge with un-acked packets
		    packetType = P_PACKET_NO_ACK;
		    pdataOffset = 2;
		}
		int dataLength = packet.length - pdataOffset;
		byte[] dataPacket = new byte[dataLength];
		System.arraycopy(packet, pdataOffset, dataPacket, 0, dataLength);
		pushProtocolPacket(packetType, dataPacket);
	    }
	}
	catch (IOException e) { }
    }

    // Read system-level packet. If inSync is false, we currently don't
    // have sync
    private byte[] readFramedPacket() throws IOException {
	int count = 0;
	boolean escaped = false;

	for (;;) {
	    if (!inSync) {
		message(name + ": resynchronising");
		// re-synchronise
		while (io.readByte() != SYNC_BYTE)
		    ;
		inSync = true;
		count = 0;
		escaped = false;
	    }

	    if (count >= MTU) {
		// Packet too long, give up and try to resync
		message(name + ": packet too long");
		inSync = false;
		continue;
	    }

	    byte b = io.readByte();
	    if (escaped) {
		if (b == SYNC_BYTE) {
		    // sync byte following escape is an error, resync
		    message(name + ": unexpected sync byte");
		    inSync = false;
		    continue;
		}
		b ^= 0x20;
		escaped = false;
	    }
	    else if (b == ESCAPE_BYTE) {
		escaped = true;
		continue;
	    }
	    else if (b == SYNC_BYTE) {
		if (count < 4) {
		    // too-small frames are ignored
		    count = 0;
		    continue;
		}
		byte[] packet = new byte[count - 2];
		System.arraycopy(receiveBuffer, 0, packet, 0, count - 2);

		int readCrc = (receiveBuffer[count - 2]  & 0xff) |
		  (receiveBuffer[count - 1] & 0xff) << 8;
		int computedCrc = Crc.calc(packet, packet.length);

		if (DEBUG) {
		    System.err.println("received: ");
		    Dump.printPacket(System.err, packet);
		    System.err.println(" rcrc: " + Integer.toHexString(readCrc) +
				       " ccrc: " + Integer.toHexString(computedCrc));
		}

		if (readCrc == computedCrc) {
		    return packet;
		}
		else {
		    message(name + ": bad packet");
		    /* We don't lose sync here. If we did, garbage on the line
		       at startup will cause loss of the first packet. */
		    count = 0;
		    continue;
		}
	    }

	    receiveBuffer[count++] = b;
	}
    }

    // Class to build a framed, escaped and crced packet byte stream
    static class Escaper {
	byte[] escaped;
	int escapePtr;
	int crc;

	// We're building a length-byte packet
	Escaper(int length) {
	    escaped = new byte[2 * length];
	    escapePtr = 0;
	    crc = 0;
	    escaped[escapePtr++] = SYNC_BYTE;
	}

	static private boolean needsEscape(int b) {
	    return b == SYNC_BYTE || b == ESCAPE_BYTE;
	}

	void nextByte(int b) {
	    b = b & 0xff;
	    crc = Crc.calcByte(crc, b);
	    if (needsEscape(b)) {
		escaped[escapePtr++] = ESCAPE_BYTE;
		escaped[escapePtr++] = (byte)(b ^ 0x20);
	    }
	    else {
		escaped[escapePtr++] = (byte)b;
	    }
	}

	void terminate() {
	    escaped[escapePtr++] = SYNC_BYTE;
	}
    }

    // Write a packet of type 'packetType', first byte 'firstByte'
    // and bytes 2..'count'+1 in 'packet'
    private synchronized void writeFramedPacket(int packetType, int firstByte,
						byte[] packet, int count) throws IOException {
	if (DEBUG) {
	    System.err.println("sending: ");
	    Dump.printByte(System.err, packetType);
	    Dump.printByte(System.err, firstByte);
	    Dump.printPacket(System.err, packet);
	    System.err.println();
	}

	Escaper buffer = new Escaper(count + 6);
	
	buffer.nextByte(packetType);
	buffer.nextByte(firstByte);
	for (int i = 0; i < count; i++) {
	    buffer.nextByte(packet[i]);
	}

	int crc = buffer.crc;
	buffer.nextByte(crc & 0xff);
	buffer.nextByte(crc >> 8);

	buffer.terminate();

	byte[] realPacket = new byte[buffer.escapePtr];
	System.arraycopy(buffer.escaped, 0, realPacket, 0, buffer.escapePtr);

	if (DEBUG) {
	    Dump.dump("encoded", realPacket);
	}
	io.writeBytes(realPacket);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy