ee5e331f3c
add tcp and udp port remeber issue 533 add udp listern wait issue 534 fix wp receive on mono part issue 530 allow logging of unknown packets mav 0.9 do_speed_change fix issue 531 remember last video res issue 521 fix issue 528 - ch 6 and ch 7 options
300 lines
9.1 KiB
C#
300 lines
9.1 KiB
C#
using System.Reflection;
|
|
using System.Text;
|
|
using System.Net; // dns, ip address
|
|
using System.Net.Sockets; // tcplistner
|
|
using log4net;
|
|
using ArdupilotMega.Controls;
|
|
|
|
namespace System.IO.Ports
|
|
{
|
|
public class UdpSerial : ArdupilotMega.ICommsSerial
|
|
{
|
|
private static readonly ILog log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
|
|
UdpClient client = new UdpClient();
|
|
IPEndPoint RemoteIpEndPoint = new IPEndPoint(IPAddress.Any, 0);
|
|
byte[] rbuffer = new byte[0];
|
|
int rbufferread = 0;
|
|
|
|
public int WriteBufferSize { get; set; }
|
|
public int WriteTimeout { get; set; }
|
|
public int ReceivedBytesThreshold { get; set; }
|
|
public bool RtsEnable { get; set; }
|
|
|
|
~UdpSerial()
|
|
{
|
|
this.Close();
|
|
client = null;
|
|
}
|
|
|
|
public UdpSerial()
|
|
{
|
|
//System.Threading.Thread.CurrentThread.CurrentUICulture = new System.Globalization.CultureInfo("en-US");
|
|
//System.Threading.Thread.CurrentThread.CurrentCulture = new System.Globalization.CultureInfo("en-US");
|
|
|
|
Port = "14550";
|
|
}
|
|
|
|
public void toggleDTR()
|
|
{
|
|
}
|
|
|
|
public string Port { get; set; }
|
|
|
|
public int ReadTimeout
|
|
{
|
|
get;// { return client.ReceiveTimeout; }
|
|
set;// { client.ReceiveTimeout = value; }
|
|
}
|
|
|
|
public int ReadBufferSize {get;set;}
|
|
|
|
public int BaudRate { get; set; }
|
|
public StopBits StopBits { get; set; }
|
|
public Parity Parity { get; set; }
|
|
public int DataBits { get; set; }
|
|
|
|
public string PortName { get; set; }
|
|
|
|
public int BytesToRead
|
|
{
|
|
get { return client.Available + rbuffer.Length - rbufferread; }
|
|
}
|
|
|
|
public bool IsOpen { get { if (client.Client == null) return false; return client.Client.Connected; } }
|
|
|
|
public bool DtrEnable
|
|
{
|
|
get;
|
|
set;
|
|
}
|
|
|
|
public void Open()
|
|
{
|
|
if (client.Client.Connected)
|
|
{
|
|
log.Info("udpserial socket already open");
|
|
return;
|
|
}
|
|
|
|
ProgressReporterDialogue frmProgressReporter = new ProgressReporterDialogue
|
|
{
|
|
StartPosition = System.Windows.Forms.FormStartPosition.CenterScreen,
|
|
Text = "Connecting Mavlink UDP"
|
|
};
|
|
|
|
frmProgressReporter.DoWork += frmProgressReporter_DoWork;
|
|
|
|
frmProgressReporter.UpdateProgressAndStatus(-1, "Connecting Mavlink UDP");
|
|
|
|
ArdupilotMega.MainV2.fixtheme(frmProgressReporter);
|
|
|
|
frmProgressReporter.RunBackgroundOperationAsync();
|
|
|
|
|
|
}
|
|
|
|
void frmProgressReporter_DoWork(object sender, ArdupilotMega.Controls.ProgressWorkerEventArgs e)
|
|
{
|
|
string dest = Port;
|
|
|
|
if (ArdupilotMega.MainV2.config["UDP_port"] != null)
|
|
dest = ArdupilotMega.MainV2.config["UDP_port"].ToString();
|
|
|
|
ArdupilotMega.Common.InputBox("Listern Port", "Enter Local port (ensure remote end is already sending)", ref dest);
|
|
Port = dest;
|
|
|
|
ArdupilotMega.MainV2.config["UDP_port"] = Port;
|
|
|
|
client = new UdpClient(int.Parse(Port));
|
|
|
|
while (true)
|
|
{
|
|
((ProgressReporterDialogue)sender).UpdateProgressAndStatus(-1, "Waiting for UDP");
|
|
System.Threading.Thread.Sleep(500);
|
|
|
|
if (((ProgressReporterDialogue)sender).doWorkArgs.CancelRequested)
|
|
{
|
|
((ProgressReporterDialogue)sender).doWorkArgs.CancelAcknowledged = true;
|
|
try
|
|
{
|
|
client.Close();
|
|
}
|
|
catch { }
|
|
return;
|
|
}
|
|
|
|
if (BytesToRead > 0)
|
|
break;
|
|
}
|
|
|
|
if (BytesToRead == 0)
|
|
return;
|
|
|
|
try
|
|
{
|
|
client.Receive(ref RemoteIpEndPoint);
|
|
log.InfoFormat("NetSerial connecting to {0} : {1}", RemoteIpEndPoint.Address, RemoteIpEndPoint.Port);
|
|
client.Connect(RemoteIpEndPoint);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
if (client != null && client.Client.Connected)
|
|
{
|
|
client.Close();
|
|
}
|
|
log.Info(ex.ToString());
|
|
System.Windows.Forms.MessageBox.Show("Please check your Firewall settings\nPlease try running this command\n1. Run the following command in an elevated command prompt to disable Windows Firewall temporarily:\n \nNetsh advfirewall set allprofiles state off\n \nNote: This is just for test; please turn it back on with the command 'Netsh advfirewall set allprofiles state on'.\n");
|
|
throw new Exception("The socket/serialproxy is closed " + e);
|
|
}
|
|
}
|
|
|
|
void VerifyConnected()
|
|
{
|
|
if (client == null || !IsOpen)
|
|
{
|
|
throw new Exception("The socket/serialproxy is closed");
|
|
}
|
|
}
|
|
|
|
public int Read(byte[] readto,int offset,int length)
|
|
{
|
|
VerifyConnected();
|
|
try
|
|
{
|
|
if (length < 1) { return 0; }
|
|
|
|
if (rbufferread == rbuffer.Length)
|
|
{
|
|
MemoryStream r = new MemoryStream();
|
|
while (client.Available > 0)
|
|
{
|
|
Byte[] b = client.Receive(ref RemoteIpEndPoint);
|
|
r.Write(b, 0, b.Length);
|
|
}
|
|
rbuffer = r.ToArray();
|
|
rbufferread = 0;
|
|
}
|
|
|
|
Array.Copy(rbuffer, rbufferread, readto, offset, length);
|
|
|
|
rbufferread += length;
|
|
|
|
return length;
|
|
}
|
|
catch { throw new Exception("Socket Closed"); }
|
|
}
|
|
|
|
public int ReadByte()
|
|
{
|
|
VerifyConnected();
|
|
int count = 0;
|
|
while (this.BytesToRead == 0)
|
|
{
|
|
System.Threading.Thread.Sleep(1);
|
|
if (count > ReadTimeout)
|
|
throw new Exception("NetSerial Timeout on read");
|
|
count++;
|
|
}
|
|
byte[] buffer = new byte[1];
|
|
Read(buffer, 0, 1);
|
|
return buffer[0];
|
|
}
|
|
|
|
public int ReadChar()
|
|
{
|
|
return ReadByte();
|
|
}
|
|
|
|
public string ReadExisting()
|
|
{
|
|
VerifyConnected();
|
|
byte[] data = new byte[client.Available];
|
|
if (data.Length > 0)
|
|
Read(data, 0, data.Length);
|
|
|
|
string line = Encoding.ASCII.GetString(data, 0, data.Length);
|
|
|
|
return line;
|
|
}
|
|
|
|
public void WriteLine(string line)
|
|
{
|
|
VerifyConnected();
|
|
line = line + "\n";
|
|
Write(line);
|
|
}
|
|
|
|
public void Write(string line)
|
|
{
|
|
VerifyConnected();
|
|
byte[] data = new System.Text.ASCIIEncoding().GetBytes(line);
|
|
Write(data, 0, data.Length);
|
|
}
|
|
|
|
public void Write(byte[] write, int offset, int length)
|
|
{
|
|
VerifyConnected();
|
|
try
|
|
{
|
|
client.Send(write, length);
|
|
}
|
|
catch { }//throw new Exception("Comport / Socket Closed"); }
|
|
}
|
|
|
|
public void DiscardInBuffer()
|
|
{
|
|
VerifyConnected();
|
|
int size = client.Available;
|
|
byte[] crap = new byte[size];
|
|
log.InfoFormat("UdpSerial DiscardInBuffer {0}",size);
|
|
Read(crap, 0, size);
|
|
}
|
|
|
|
public string ReadLine() {
|
|
byte[] temp = new byte[4000];
|
|
int count = 0;
|
|
int timeout = 0;
|
|
|
|
while (timeout <= 100)
|
|
{
|
|
if (!this.IsOpen) { break; }
|
|
if (this.BytesToRead > 0)
|
|
{
|
|
byte letter = (byte)this.ReadByte();
|
|
|
|
temp[count] = letter;
|
|
|
|
if (letter == '\n') // normal line
|
|
{
|
|
break;
|
|
}
|
|
|
|
|
|
count++;
|
|
if (count == temp.Length)
|
|
break;
|
|
timeout = 0;
|
|
} else {
|
|
timeout++;
|
|
System.Threading.Thread.Sleep(5);
|
|
}
|
|
}
|
|
|
|
Array.Resize<byte>(ref temp, count + 1);
|
|
|
|
return Encoding.ASCII.GetString(temp, 0, temp.Length);
|
|
}
|
|
|
|
public void Close()
|
|
{
|
|
if (client.Client != null && client.Client.Connected)
|
|
{
|
|
client.Client.Close();
|
|
client.Close();
|
|
}
|
|
|
|
client = new UdpClient();
|
|
}
|
|
}
|
|
}
|