2
0
mirror of https://github.com/fhem/fhem-mirror.git synced 2025-04-05 17:48:44 +00:00

00_HMUARTLGW.pm: Implement updating the coprocessor-firmware

git-svn-id: https://svn.fhem.de/fhem/trunk@12100 2b470e98-0d58-463d-a4d8-8e2adae1ed80
This commit is contained in:
mgernoth 2016-08-31 18:17:22 +00:00
parent bbb88af887
commit 7f103d9a54

View File

@ -102,6 +102,7 @@ use constant {
HMUARTLGW_STATE_SEND => 100,
HMUARTLGW_STATE_SEND_NOACK => 101,
HMUARTLGW_STATE_SEND_TIMED => 102,
HMUARTLGW_STATE_UPDATE_COPRO => 200,
HMUARTLGW_CMD_TIMEOUT => 10,
HMUARTLGW_SEND_TIMEOUT => 10,
@ -114,6 +115,7 @@ my %sets = (
"open" => "",
"close" => "",
"restart" => "",
"updateCoPro" => "",
);
my %gets = (
@ -162,6 +164,8 @@ sub HMUARTLGW_crc16($;$);
sub HMUARTLGW_encrypt($$);
sub HMUARTLGW_decrypt($$);
sub HMUARTLGW_getVerbLvl($$$$);
sub HMUARTLGW_firmwareGetBlock($$$);
sub HMUARTLGW_updateCoPro($$);
sub HMUARTLGW_DoInit($)
{
@ -1134,13 +1138,19 @@ sub HMUARTLGW_Parse($$$$)
Log3($hash, 1, "HMUARTLGW ${name} failed to enter App!");
}
} elsif ($hash->{DevState} > HMUARTLGW_STATE_ENTER_APP) {
Log3($hash, 1, "HMUARTLGW ${name} unexpected info about ${running} received (module crashed?), reopening");
Log3($hash, 1, "HMUARTLGW ${name} unexpected info about ${running} received (module crashed?), reopening")
if (!defined($hash->{FirmwareFile}));
HMUARTLGW_Reopen($hash);
return;
}
} elsif ($msg =~ m/^04(..)/) {
my $ack = $1;
if ($hash->{DevState} == HMUARTLGW_STATE_UPDATE_COPRO) {
HMUARTLGW_updateCoPro($hash, $msg);
return;
}
if ($ack eq HMUARTLGW_ACK_INFO && $hash->{DevState} == HMUARTLGW_STATE_QUERY_APP) {
my $running = pack("H*", substr($msg, 4));
@ -1151,6 +1161,15 @@ sub HMUARTLGW_Parse($$$$)
RemoveInternalTimer($hash);
InternalTimer(gettimeofday()+1, "HMUARTLGW_GetSetParameters", $hash, 0);
} else {
if (defined($hash->{FirmwareFile}) && $hash->{FirmwareFile} ne "") {
Log3($hash, 1, "HMUARTLGW ${name} starting firmware upgrade");
$hash->{FirmwareBlock} = 0;
$hash->{DevState} = HMUARTLGW_STATE_UPDATE_COPRO;
HMUARTLGW_updateCondition($hash);
HMUARTLGW_updateCoPro($hash, $msg);
return;
}
$hash->{DevState} = HMUARTLGW_STATE_ENTER_APP;
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
RemoveInternalTimer($hash);
@ -1669,13 +1688,23 @@ sub HMUARTLGW_Set($@)
InternalTimer(gettimeofday()+20, "HMUARTLGW_RemoveHMPair", "hmPairForSec:".$name, 1);
} elsif ($cmd eq "reopen") {
HMUARTLGW_Reopen($hash);
} elsif($cmd eq "close") {
} elsif ($cmd eq "close") {
HMUARTLGW_Undefine($hash, $name);
readingsSingleUpdate($hash, "state", "closed", 1);
$hash->{XmitOpen} = 0;
} elsif($cmd eq "open") {
} elsif ($cmd eq "open") {
DevIo_OpenDev($hash, 0, "HMUARTLGW_DoInit", \&HMUARTLGW_Connect);
} elsif($cmd eq "restart") {
} elsif ($cmd eq "restart") {
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
} elsif ($cmd eq "updateCoPro") {
return "Usage: set $name updateCoPro </path/to/firmware.eq3>"
if(!$arg);
my $block = HMUARTLGW_firmwareGetBlock($hash, $arg, 0);
return "${arg} is not a valid firmware file!"
if (!defined($block) || $block eq "");
$hash->{FirmwareFile} = $arg;
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
}
@ -1858,6 +1887,9 @@ sub HMUARTLGW_updateCondition($)
if ($hash->{DevState} == HMUARTLGW_STATE_NONE) {
$cond = "disconnected";
$loadLvl = "suspended";
} elsif ($hash->{DevState} == HMUARTLGW_STATE_UPDATE_COPRO) {
$cond = "fwupdate";
$loadLvl = "suspended";
}
readingsBeginUpdate($hash);
@ -2107,6 +2139,96 @@ sub HMUARTLGW_decrypt($$)
$plaintext;
}
sub HMUARTLGW_firmwareGetBlock($$$) {
my ($hash, $file, $id) = @_;
my $name = $hash->{NAME};
my $block = "";
my $ret = open(my $fd, "<", $file);
if (!$ret) {
Log3($hash, 1, "HMUARTLGW ${name} can't open firmware file ${file}: $!");
return undef;
}
my $fw = "";
while(<$fd>) {
$fw .= $_;
}
close($fd);
my $n = 0;
while(length($fw)) {
my $len = unpack('n', pack('H4', $fw));
if ($n eq $id) {
$block = substr($fw, 4, $len * 2);
last;
}
$fw = substr($fw, 4 + ($len * 2));
$n++;
}
if ($n != $id) {
Log3($hash, 1, "HMUARTLGW ${name} invalid block ${id} requested");
return undef;
}
$block;
}
sub HMUARTLGW_updateCoPro($$) {
my ($hash, $msg) = @_;
my $name = $hash->{NAME};
RemoveInternalTimer($hash);
if (($hash->{FirmwareBlock} > 0) && ($msg !~ /^0401$/)) {
Log3($hash, 1, "HMUARTLGW ${name} firmware flash failed on block " . ($hash->{FirmwareBlock} - 1));
delete($hash->{FirmwareFile});
delete($hash->{FirmwareBlock});
$hash->{DevState} = HMUARTLGW_STATE_QUERY_APP;
HMUARTLGW_send($hash, HMUARTLGW_OS_GET_APP, HMUARTLGW_DST_OS);
InternalTimer(gettimeofday()+HMUARTLGW_CMD_TIMEOUT, "HMUARTLGW_CheckCmdResp", $hash, 0);
return;
}
my $block = HMUARTLGW_firmwareGetBlock($hash, $hash->{FirmwareFile}, $hash->{FirmwareBlock});
if (!defined($block)) {
Log3($hash, 1, "HMUARTLGW ${name} firmware update aborted");
delete($hash->{FirmwareFile});
delete($hash->{FirmwareBlock});
$hash->{DevState} = HMUARTLGW_STATE_QUERY_APP;
HMUARTLGW_send($hash, HMUARTLGW_OS_GET_APP, HMUARTLGW_DST_OS);
InternalTimer(gettimeofday()+HMUARTLGW_CMD_TIMEOUT, "HMUARTLGW_CheckCmdResp", $hash, 0);
return;
} elsif ($block eq "") {
Log3($hash, 1, "HMUARTLGW ${name} firmware update successfull");
delete($hash->{FirmwareFile});
delete($hash->{FirmwareBlock});
$hash->{DevState} = HMUARTLGW_STATE_QUERY_APP;
HMUARTLGW_send($hash, HMUARTLGW_OS_GET_APP, HMUARTLGW_DST_OS);
InternalTimer(gettimeofday()+HMUARTLGW_CMD_TIMEOUT, "HMUARTLGW_CheckCmdResp", $hash, 0);
return;
}
#strip CRC from block
$block = substr($block, 0, -4);
HMUARTLGW_send($hash, HMUARTLGW_OS_UPDATE_FIRMWARE . ${block}, HMUARTLGW_DST_OS);
$hash->{FirmwareBlock}++;
InternalTimer(gettimeofday()+HMUARTLGW_CMD_TIMEOUT, "HMUARTLGW_CheckCmdResp", $hash, 0);
}
sub HMUARTLGW_getVerbLvl($$$$) {
my ($hash, $src, $dst, $def) = @_;
@ -2181,6 +2303,19 @@ sub HMUARTLGW_getVerbLvl($$$$) {
<li>restart<br>
Reboots the device.
</li>
<li>updateCoPro &lt;/path/to/firmware.eq3&gt;<br>
Update the coprocessor-firmware (reading D-firmware) from the
supplied file. Source for firmware-images (version 1.4.1, official
eQ-3 repository):<br>
<ul>
<li>HM-MOD-UART: <a href="https://raw.githubusercontent.com/eq-3/occu/28045df83480122f90ab92f7c6e625f9bf3b61aa/firmware/HM-MOD-UART/coprocessor_update.eq3">coprocessor_update.eq3</a></li>
<li>HM-LGW-O-TW-W-EU: <a href="https://raw.githubusercontent.com/eq-3/occu/28045df83480122f90ab92f7c6e625f9bf3b61aa/firmware/coprocessor_update_hm_only.eq3">coprocessor_update_hm_only.eq3</a><br>
Please also make sure that D-LANfirmware is at least at version
1.1.5. To update to this version, use the eQ-3 CLI tools (see wiki)
or use the eQ-3 netfinder with this firmware image: <a href="https://github.com/eq-3/occu/raw/28045df83480122f90ab92f7c6e625f9bf3b61aa/firmware/hm-lgw-o-tw-w-eu_update.eq3">hm-lgw-o-tw-w-eu_update.eq3</a><br>
<b>Do not flash hm-lgw-o-tw-w-eu_update.eq3 with updateCoPro!</b></li>
</ul>
</li>
</ul>
<br>
<a name="HMUARTLGW_get"></a>