Simple curses-based mesh display
This commit is contained in:
parent
2e27a60d19
commit
a38a5a9874
2 changed files with 227 additions and 0 deletions
60
tools/mesh/meshtrace.pl
Executable file
60
tools/mesh/meshtrace.pl
Executable file
|
@ -0,0 +1,60 @@
|
|||
#!/usr/bin/perl
|
||||
#
|
||||
# vim:set ts=4 sw=4:
|
||||
|
||||
use strict;
|
||||
use Curses;
|
||||
use POSIX qw(strftime);
|
||||
|
||||
use lib '.';
|
||||
use r0ket;
|
||||
|
||||
$|=1;
|
||||
|
||||
my $ser="<undef>";
|
||||
|
||||
do {$ser=$_ if ( -e $_ ) } for qw(/dev/ttyS3 /dev/ttyACM0);
|
||||
|
||||
if ($ARGV[0] eq "-s"){
|
||||
shift;
|
||||
$ser=shift;
|
||||
};
|
||||
|
||||
open(SER, "+<",$ser) || die "open serial: $!";
|
||||
|
||||
r0ket::readbeacon();
|
||||
|
||||
my $str;
|
||||
my %bdata;
|
||||
|
||||
initscr;
|
||||
END{endwin;}
|
||||
use constant WIDTH => 80;
|
||||
use constant m_height => 15;
|
||||
my $win_top=subwin(2,WIDTH,0,0);
|
||||
my $win=subwin(m_height,WIDTH,2,0);
|
||||
noecho;
|
||||
curs_set(0);
|
||||
$win_top->addstr(0,0,"r0ket Mesh-Trace 0.1");
|
||||
$win_top->addstr(1,0,"-"x20);
|
||||
$win_top->refresh;
|
||||
|
||||
my $beaconctr=0;
|
||||
while(1){
|
||||
$str=r0ket::get_packet(\*SER);
|
||||
my $p=r0ket::pkt_beauty($str);
|
||||
if(!$bdata{$p->{beacon}}){
|
||||
$bdata{$p->{beacon}}=++$beaconctr;
|
||||
};
|
||||
$win->addstr($bdata{$p->{beacon}},0,
|
||||
sprintf "%s | g=%d rel=%s time=%s =%+4d | %s",
|
||||
$p->{beacon},
|
||||
$p->{generation},
|
||||
$p->{release},
|
||||
strftime("%Y-%m-%d %H:%M:%S",gmtime $p->{time}),
|
||||
$p->{time}-(time+3600),
|
||||
r0ket::getbeacon($p->{beacon})
|
||||
);
|
||||
$win->refresh;
|
||||
};
|
||||
r0ket::rest();
|
167
tools/mesh/r0ket.pm
Executable file
167
tools/mesh/r0ket.pm
Executable file
|
@ -0,0 +1,167 @@
|
|||
#!/usr/bin/perl
|
||||
#
|
||||
# vim:set ts=4 sw=4:
|
||||
|
||||
use strict;
|
||||
|
||||
use IO::Select;
|
||||
package r0ket;
|
||||
|
||||
use Digest::CRC qw(crcccitt);
|
||||
use POSIX qw(strftime);
|
||||
|
||||
|
||||
### Utility
|
||||
sub sprint{
|
||||
return join("",map {
|
||||
if (ord($_)>30 && ord($_)<127){
|
||||
$_;
|
||||
}else{
|
||||
"[x".unpack("H*",$_)."]";
|
||||
}
|
||||
}split(//,shift));
|
||||
};
|
||||
|
||||
### Nickname/beacon helper functions
|
||||
our %beacon;
|
||||
sub readbeacon{
|
||||
return if( ! -f "BEACON" );
|
||||
open(B,"<","BEACON") || die "open: $!";
|
||||
while(<B>){
|
||||
/(\w+)\s+(.*)/ && do {
|
||||
$beacon{$1}=$2;
|
||||
};
|
||||
};
|
||||
close(B);
|
||||
};
|
||||
sub getbeacon{
|
||||
my $b=shift;
|
||||
if(!$beacon{$b}){
|
||||
return "";
|
||||
}else{
|
||||
return $beacon{$b};
|
||||
};
|
||||
};
|
||||
sub resolvebeacon{
|
||||
my $b=shift;
|
||||
if(!$beacon{$b}){
|
||||
return $b;
|
||||
}else{
|
||||
return "$b ($beacon{$b})";
|
||||
};
|
||||
};
|
||||
sub addbeacon{
|
||||
my($b,$n)=@_;
|
||||
if(!$beacon{$b}){
|
||||
$beacon{$b}=$n;
|
||||
};
|
||||
};
|
||||
sub writebeacon{
|
||||
open(B,">","BEACON") || die "write: $!";
|
||||
for(sort keys %beacon){
|
||||
print B "$_ $beacon{$_}\n";
|
||||
};
|
||||
close(B);
|
||||
};
|
||||
|
||||
### Packet mgmt
|
||||
|
||||
our $buffer;
|
||||
sub get_packet{
|
||||
my $dev=shift;
|
||||
|
||||
sub _get_bytes{
|
||||
my $rr;
|
||||
sysread($dev,$rr,1024);
|
||||
if(length($rr)<=1){
|
||||
select(undef,undef,undef,0.1);
|
||||
};
|
||||
$buffer.=$rr;
|
||||
};
|
||||
|
||||
my $cnt=0;
|
||||
while(++$cnt<10){
|
||||
if(length($buffer)<2){
|
||||
_get_bytes();
|
||||
}elsif($buffer !~ /^\\[12]/){
|
||||
$buffer=~s/^(.[^\\]*)//s;
|
||||
# print STDERR "Unparseable stuff: <",sprint($1),">\n";
|
||||
}elsif ($buffer =~ s/^\\([12])(.*?)\\0//s){
|
||||
my $str=$2;
|
||||
$str=~s/\\\\/\\/g; # dequote
|
||||
return $str;
|
||||
}else{
|
||||
_get_bytes();
|
||||
};
|
||||
};
|
||||
die "No packets for >1sec?\n";
|
||||
};
|
||||
|
||||
sub rest{
|
||||
if(length($buffer)>0){
|
||||
print "rest: <", sprint($buffer), ">\n";
|
||||
};
|
||||
};
|
||||
|
||||
### Pkt beautify
|
||||
sub nice_mesh{
|
||||
my $pkt=shift;
|
||||
my $out;
|
||||
my $type=substr($pkt,0,1);
|
||||
# next if(defined $arg && $arg ne $i);
|
||||
$out->{type}=$type;
|
||||
$out->{string}="[$type]";
|
||||
$out->{generation}=unpack("C",substr($pkt,1,1));
|
||||
$out->{string}.= " g=".$out->{generation};
|
||||
if($type eq "T"){
|
||||
$out->{time}= unpack("N",substr($pkt,2,4));
|
||||
$out->{release}=unpack("H*",substr($pkt,24,2));
|
||||
$out->{beacon}= unpack("H*",substr($pkt,26,4));
|
||||
|
||||
$out->{string}.=sprintf " t=%s (%+4d) rel=%s beacon=%s",
|
||||
strftime("%Y-%m-%d %H:%M:%S",gmtime $out->{time}),
|
||||
$out->{time}-(time+3600),
|
||||
$out->{release},
|
||||
resolvebeacon($out->{beacon});
|
||||
}elsif($type eq "i"){
|
||||
$out->{score}=unpack("N",substr($pkt,2,4));
|
||||
$out->{nick}= unpack("Z*",substr($pkt,6,length($pkt)-8));
|
||||
|
||||
$out->{string}.=sprintf " score=%d nick=%s",
|
||||
$out->{score},
|
||||
$out->{nick};
|
||||
}elsif($type eq "B"){
|
||||
$out->{time}=unpack("N",substr($pkt,2,4));
|
||||
$out->{id}= unpack("c",substr($pkt,6,1));
|
||||
$out->{hop}= unpack("n",substr($pkt,11,4));
|
||||
|
||||
$out->{string}.=sprintf " t=%d id=%s hop=%3d",
|
||||
$out->{time},
|
||||
$out->{id},
|
||||
$out->{hop};
|
||||
}else{
|
||||
$out->{string}.= " <??: ".unpack("H*",substr($pkt,2,length($pkt)-4)).">";
|
||||
};
|
||||
return $out;
|
||||
};
|
||||
|
||||
sub pkt_beauty{
|
||||
my $pkt=shift;
|
||||
my $out;
|
||||
|
||||
$out=nice_mesh($pkt);
|
||||
|
||||
my $pkt_crc= unpack("n",substr($pkt,length($pkt)-2,2));
|
||||
my $calc_crc= crcccitt(substr($pkt,0,length($pkt)-2));
|
||||
|
||||
if ($pkt_crc eq $calc_crc){
|
||||
$out->{crc}="ok";
|
||||
}else{
|
||||
$out->{crc}="fail";
|
||||
$out->{string}.= " CRCFAIL";
|
||||
};
|
||||
return $out;
|
||||
}
|
||||
|
||||
|
||||
1;
|
Loading…
Reference in a new issue