예제 #1
0
        /// <summary>
        /// Handler for Mac sdu frames
        /// </summary>
        private void MacDataIndHandler(
            IMacDataSap sender,
            MacAddress srcAddr,
            UInt16 srcPanId,
            MacAddress dstAddr,
            UInt16 dstPanId,
            Frame sdu,
            Byte linkQuality,
            Byte DSN,
            UInt32 timeStamp,
            SecurityOptions securityOptions)
        {
            bool ok = true;
            ok &= (srcPanId == _panId && dstPanId == _panId);
            switch (dstAddr.Mode)
            {
                case MacAddressingMode.ShortAddress:
                    ok &= (dstAddr.ShortAddress == _addrShort) || (dstAddr.ShortAddress == cBroadcastShortAddr);
                    break;
                case MacAddressingMode.ExtendedAddress:
                    ok &= (dstAddr.ExtendedAddress == _addrExt);
                    break;
                default:
                    ok &= false;
                    break;
            }

            if (srcAddr.Mode == MacAddressingMode.NoAddress)
                ok = false;
            if (sdu == null || sdu.LengthDataUsed == 0)
                ok = false;

            if (ok)
            {
                Messages6LoWPAN.Dispatch dispatch;
                ok = Messages6LoWPAN.GetType(sdu, out dispatch);
                if (ok)
                {
                    if (dispatch == Messages6LoWPAN.Dispatch.NonLowPan)
                    {
                        Message.Type type;
                        ok = Message.GetType(sdu, out type);
                        if (ok)
                        {
                            switch (type)
                            {
                                case Message.Type.AddressRequest:
                                    HandleAddressRequest(srcAddr, dstAddr, ref sdu);
                                    break;
                                case Message.Type.AddressReply:
                                    HandleAddressReply(srcAddr, dstAddr, ref sdu);
                                    break;
                                case Message.Type.RouteRequest:
                                case Message.Type.RouteReply:
                                    HandleRoutingMessage(srcAddr, dstAddr, ref sdu, linkQuality);
                                    break;
                                case Message.Type.RouteError:
                                    HandleRouteError(srcAddr, dstAddr, ref sdu);
                                    break;
                                case Message.Type.NeighborhoodDiscovery:
                                    HandleNeighborhoodDiscovery(srcAddr, dstAddr, ref sdu, linkQuality);
                                    break;
                                case Message.Type.Data:
                                    if (srcAddr.Mode == MacAddressingMode.ShortAddress && dstAddr.Mode == MacAddressingMode.ShortAddress)
                                    {
                                        HandleData(srcAddr.ShortAddress, dstAddr.ShortAddress, ref sdu);
                                    }
                                    break;
                                case Message.Type.DiscoveryReply:
                                    {
                                        if (srcAddr.Mode == MacAddressingMode.ShortAddress && _isAddrServer)
                                        {
                                            Message.DiscoveryReply drep = new Message.DiscoveryReply();
                                            if (drep.ReadFromFrame(sdu))
                                            {
                                                _addrServer.HandleDiscoveryReply(srcAddr.ShortAddress);
                                            }
                                        }
                                    }
                                    break;
                            }
                        }
                    }
                    else
                    {
                        if (srcAddr.Mode == MacAddressingMode.ShortAddress && dstAddr.Mode == MacAddressingMode.ShortAddress)
                        {
                            HandleData(srcAddr.ShortAddress, dstAddr.ShortAddress, ref sdu);
                        }
                    }
                }
            }

            Frame.Release(ref sdu);
        }
예제 #2
0
        private void HandleData(
            UInt16 srcAddr,
            UInt16 dstAddr,
            ref Frame sdu)
        {
            Messages6LoWPAN.Dispatch dispatch;
            if (!Messages6LoWPAN.GetType(sdu, out dispatch))
                return;

            bool consume = false; // receive locally
            bool forward = false; // forward to another node

            UInt16 origAddr = srcAddr; // originator
            UInt16 targetAddr = dstAddr; // target
            bool haveMeshHeader = false;
            Messages6LoWPAN.MeshHeader mh = new Messages6LoWPAN.MeshHeader();
            if (dispatch == Messages6LoWPAN.Dispatch.Mesh)
            {
                if (!mh.ReadFromFrameHeader(sdu) || !Messages6LoWPAN.GetType(sdu, out dispatch))
                    return;
                haveMeshHeader = true;
                origAddr = mh.originatorAddress;
                targetAddr = mh.finalAddress;
            }

            bool haveBroadcastHeader = false;
            Messages6LoWPAN.BroadcastHeader bh = new Messages6LoWPAN.BroadcastHeader();
            if (dispatch == Messages6LoWPAN.Dispatch.IpPacket && bh.ReadFromFrameHeader(sdu))
            {
                haveBroadcastHeader = true;
                bool usefull = _routingTable.CheckBroadcastSeqNo(origAddr, bh.seqNo) && (origAddr != _addrShort);
                if (!usefull || !haveMeshHeader || !Messages6LoWPAN.GetType(sdu, out dispatch))
                    return;
            }

            bool multicast = IsMulticast(targetAddr);
            consume = (targetAddr == _addrShort || multicast);
            forward = (targetAddr != _addrShort || multicast) && haveMeshHeader && (mh.HopsLeft > 0);

            if (forward)
            {
                mh.HopsLeft--;
                // currently we always need to reallocate a frame to be forwarded, since mac header space is worst case assumption
                Frame frame = Frame.GetFrame(_macHeader + Messages6LoWPAN.MeshHeader.cLengthMax + Messages6LoWPAN.BroadcastHeader.cLength,
                    sdu.LengthDataUsed + _macTailer);
                frame.WriteToBack(sdu);

                bool ok = true;
                UInt16 nextHop;
                if (multicast)
                {
                    nextHop = cBroadcastShortAddr;
                    ok = haveBroadcastHeader && bh.WriteToFrameHeader(frame);
                }
                else
                {
                    bool routeHasError;
                    ok = _routingTable.GetRoute(targetAddr, out nextHop, out routeHasError);
                    if (!ok || routeHasError)
                    {
                        SendRouteError(origAddr, targetAddr, !routeHasError);
                    }
                }

                ok &= mh.WriteToFrameHeader(frame);
                if (ok)
                {
                    MacDataRequest(nextHop, ref frame);
                }

                Frame.Release(ref frame);
            }

            if (consume)
            {
                if (dispatch == Messages6LoWPAN.Dispatch.NonLowPan)
                {
                    Message.Type type;
                    if (Message.GetType(sdu, out type))
                    {
                        switch (type)
                        {
                            case Message.Type.Data:
                                {
                                    Message.Data data = new Message.Data();
                                    if (data.ReadFromFrameHeader(sdu))
                                    {
                                        _data.HandleDataIndication(origAddr, targetAddr, ref sdu);
                                    }
                                    break;
                                }
                            case Message.Type.DiscoveryReply:
                                {
                                    Message.DiscoveryReply drep;
                                    if (_isAddrServer && drep.ReadFromFrame(sdu))
                                    {
                                        _addrServer.HandleDiscoveryReply(origAddr);
                                    }
                                    break;
                                }
                        }
                    }
                }
                else
                { // must be 6LoWPAN frame
                    _data.HandleDataIndication6Low(origAddr, targetAddr, ref sdu);
                }
            }

            Frame.Release(ref sdu);
        }