#include "ns3/core-module.h" #include "ns3/applications-module.h" #include "ns3/command-line.h" #include "ns3/simulator.h" #include "ns3/node.h" #include "ns3/drop-tail-queue.h" #include "ns3/timestamp-tag.h" #include "ns3/tsn-multidrop-net-device.h" #include "ns3/tsn-multidrop-channel.h" #include "ns3/ethernet-generator.h" #include "ns3/ethernet-header2.h" #include "ns3/cbs.h" /** * \file * * Example of the use of tsn-net-device.cc tsn-multidrop-channel.cc on a network * composed of four end-stations connected by a 10Base-T1S with a CBS on the * highest priority FIFO of ES1 * ES1 ====== ES2 * || * ||== ES3 * || * ||== ES4 */ using namespace ns3; NS_LOG_COMPONENT_DEFINE("Example"); //A callback to log pkt fifo entry time static void MacTxCallback(std::string context, Ptr p) { Ptr originalPacket = p->Copy(); EthernetHeader2 ethHeader; originalPacket->RemoveHeader(ethHeader); NS_LOG_INFO((Simulator::Now()).As(Time::S) << " \t" << context << " : Pkt #" << p->GetUid() << "(VID:" << ethHeader.GetVid() << ") entering the FIFO on the producer"); } //A callback to log the pkt latency static void LatencyCallback(std::string context, Ptr p) { TimestampTag tag; if (!p->FindFirstMatchingByteTag(tag)) { return; } Time arrival = Simulator::Now(); Time latency = arrival - tag.GetTimestamp(); Ptr originalPacket = p->Copy(); EthernetHeader2 ethHeader; originalPacket->RemoveHeader(ethHeader); NS_LOG_INFO((Simulator::Now()).As(Time::S) << " \t" << context << " : Pkt #" << p->GetUid() << " received from " << ethHeader.GetSrc() << "(VID:" << ethHeader.GetVid() << ") with a latency=" << latency.GetNanoSeconds() <<"ns"); } int main(int argc, char* argv[]) { //Enable logging LogComponentEnable("Example", LOG_LEVEL_INFO); LogComponentEnable("EthernetGenerator", LOG_LEVEL_INFO); // LogComponentEnable("TsnMultidropNetDevice", LOG_LEVEL_INFO); // LogComponentEnable("TsnMultidropChannel", LOG_LEVEL_INFO); CommandLine cmd(__FILE__); cmd.Parse(argc, argv); //Create four nodes Ptr n0 = CreateObject(); Names::Add("ES1", n0); Ptr n1 = CreateObject(); Names::Add("ES2", n1); Ptr n2 = CreateObject(); Names::Add("ES3", n2); Ptr n3 = CreateObject(); Names::Add("ES4", n3); //Create and add a netDevice to each node Ptr net0 = CreateObject(); net0->SetAttribute("PLCALocalNodeId", UintegerValue(0)); net0->SetAttribute("PLCANodeCount", UintegerValue(4)); // net0->SetAttribute("PLCAMaxBurstCount", UintegerValue(1)); n0->AddDevice(net0); Names::Add("ES1#01", net0); Ptr net1 = CreateObject(); net1->SetAttribute("PLCALocalNodeId", UintegerValue(1)); net1->SetAttribute("PLCANodeCount", UintegerValue(4)); n1->AddDevice(net1); Names::Add("ES2#01", net1); Ptr net2 = CreateObject(); net2->SetAttribute("PLCALocalNodeId", UintegerValue(2)); net2->SetAttribute("PLCANodeCount", UintegerValue(4)); n2->AddDevice(net2); Names::Add("ES3#01", net2); Ptr net3 = CreateObject(); net3->SetAttribute("PLCALocalNodeId", UintegerValue(3)); net3->SetAttribute("PLCANodeCount", UintegerValue(4)); n3->AddDevice(net3); Names::Add("ES4#01", net3); //Create a 10Base-T1S Channel and attach it two the netDevices Ptr channel = CreateObject(); net0->Attach(channel); net1->Attach(channel); net2->Attach(channel); net3->Attach(channel); //Allocate a Mac address and create a FIFO (for the output port) //for each netDevice. net0->SetAddress(Mac48Address::Allocate()); Ptr cbs = CreateObject(); cbs->SetTsnNetDevice(net0); cbs->SetAttribute("IdleSlope", DataRateValue(DataRate("8Mb/s"))); cbs->SetAttribute("portTransmitRate", DataRateValue(DataRate("10Mb/s"))); net0->SetQueue(CreateObject>()); net0->SetQueue(CreateObject>(), cbs); net1->SetAddress(Mac48Address::Allocate()); net1->SetQueue(CreateObject>()); net1->SetQueue(CreateObject>()); net2->SetAddress(Mac48Address::Allocate()); net2->SetQueue(CreateObject>()); net2->SetQueue(CreateObject>()); net3->SetAddress(Mac48Address::Allocate()); net2->SetQueue(CreateObject>()); net3->SetQueue(CreateObject>()); //Application description Ptr app0 = CreateObject(); app0->Setup(net0); app0->SetAttribute("BurstSize", UintegerValue(2)); app0->SetAttribute("PayloadSize", UintegerValue(1400)); app0->SetAttribute("Period", TimeValue(Seconds(5))); app0->SetAttribute("PCP", UintegerValue(1)); app0->SetAttribute("VlanID", UintegerValue(1)); n0->AddApplication(app0); app0->SetStartTime(Seconds(0)); app0->SetStopTime(Seconds(10)); Ptr app1 = CreateObject(); app1->Setup(net0); app1->SetAttribute("BurstSize", UintegerValue(2)); app1->SetAttribute("PayloadSize", UintegerValue(1400)); app1->SetAttribute("Period", TimeValue(Seconds(5))); app1->SetAttribute("PCP", UintegerValue(0)); app1->SetAttribute("VlanID", UintegerValue(2)); n0->AddApplication(app1); app1->SetStartTime(Seconds(0)); app1->SetStopTime(Seconds(10)); //Callback to log pkt fifo entry time std::string context = Names::FindName(n0) + ":" + Names::FindName(net0); net0->TraceConnectWithoutContext("MacTx", MakeBoundCallback(&MacTxCallback, context)); context = Names::FindName(n1) + ":" + Names::FindName(net1); net1->TraceConnectWithoutContext("MacTx", MakeBoundCallback(&MacTxCallback, context)); context = Names::FindName(n2) + ":" + Names::FindName(net2); net2->TraceConnectWithoutContext("MacTx", MakeBoundCallback(&MacTxCallback, context)); context = Names::FindName(n3) + ":" + Names::FindName(net3); net3->TraceConnectWithoutContext("MacTx", MakeBoundCallback(&MacTxCallback, context)); //Callback to display the packet latency net0->TraceConnectWithoutContext("Latency", MakeBoundCallback(&LatencyCallback, context)); context = Names::FindName(n1) + ":" + Names::FindName(net1); net1->TraceConnectWithoutContext("Latency", MakeBoundCallback(&LatencyCallback, context)); context = Names::FindName(n2) + ":" + Names::FindName(net2); net2->TraceConnectWithoutContext("Latency", MakeBoundCallback(&LatencyCallback, context)); context = Names::FindName(n3) + ":" + Names::FindName(net3); net3->TraceConnectWithoutContext("Latency", MakeBoundCallback(&LatencyCallback, context)); //Execute the simulation Simulator::Stop(MilliSeconds(5)); Simulator::Run(); Simulator::Destroy(); return 0; }