mirror of
https://github.com/fhem/fhem-mirror.git
synced 2025-04-06 12:18:46 +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:
parent
bbb88af887
commit
7f103d9a54
@ -102,6 +102,7 @@ use constant {
|
|||||||
HMUARTLGW_STATE_SEND => 100,
|
HMUARTLGW_STATE_SEND => 100,
|
||||||
HMUARTLGW_STATE_SEND_NOACK => 101,
|
HMUARTLGW_STATE_SEND_NOACK => 101,
|
||||||
HMUARTLGW_STATE_SEND_TIMED => 102,
|
HMUARTLGW_STATE_SEND_TIMED => 102,
|
||||||
|
HMUARTLGW_STATE_UPDATE_COPRO => 200,
|
||||||
|
|
||||||
HMUARTLGW_CMD_TIMEOUT => 10,
|
HMUARTLGW_CMD_TIMEOUT => 10,
|
||||||
HMUARTLGW_SEND_TIMEOUT => 10,
|
HMUARTLGW_SEND_TIMEOUT => 10,
|
||||||
@ -114,6 +115,7 @@ my %sets = (
|
|||||||
"open" => "",
|
"open" => "",
|
||||||
"close" => "",
|
"close" => "",
|
||||||
"restart" => "",
|
"restart" => "",
|
||||||
|
"updateCoPro" => "",
|
||||||
);
|
);
|
||||||
|
|
||||||
my %gets = (
|
my %gets = (
|
||||||
@ -162,6 +164,8 @@ sub HMUARTLGW_crc16($;$);
|
|||||||
sub HMUARTLGW_encrypt($$);
|
sub HMUARTLGW_encrypt($$);
|
||||||
sub HMUARTLGW_decrypt($$);
|
sub HMUARTLGW_decrypt($$);
|
||||||
sub HMUARTLGW_getVerbLvl($$$$);
|
sub HMUARTLGW_getVerbLvl($$$$);
|
||||||
|
sub HMUARTLGW_firmwareGetBlock($$$);
|
||||||
|
sub HMUARTLGW_updateCoPro($$);
|
||||||
|
|
||||||
sub HMUARTLGW_DoInit($)
|
sub HMUARTLGW_DoInit($)
|
||||||
{
|
{
|
||||||
@ -1134,13 +1138,19 @@ sub HMUARTLGW_Parse($$$$)
|
|||||||
Log3($hash, 1, "HMUARTLGW ${name} failed to enter App!");
|
Log3($hash, 1, "HMUARTLGW ${name} failed to enter App!");
|
||||||
}
|
}
|
||||||
} elsif ($hash->{DevState} > HMUARTLGW_STATE_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);
|
HMUARTLGW_Reopen($hash);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} elsif ($msg =~ m/^04(..)/) {
|
} elsif ($msg =~ m/^04(..)/) {
|
||||||
my $ack = $1;
|
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) {
|
if ($ack eq HMUARTLGW_ACK_INFO && $hash->{DevState} == HMUARTLGW_STATE_QUERY_APP) {
|
||||||
my $running = pack("H*", substr($msg, 4));
|
my $running = pack("H*", substr($msg, 4));
|
||||||
|
|
||||||
@ -1151,6 +1161,15 @@ sub HMUARTLGW_Parse($$$$)
|
|||||||
RemoveInternalTimer($hash);
|
RemoveInternalTimer($hash);
|
||||||
InternalTimer(gettimeofday()+1, "HMUARTLGW_GetSetParameters", $hash, 0);
|
InternalTimer(gettimeofday()+1, "HMUARTLGW_GetSetParameters", $hash, 0);
|
||||||
} else {
|
} 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;
|
$hash->{DevState} = HMUARTLGW_STATE_ENTER_APP;
|
||||||
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
|
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
|
||||||
RemoveInternalTimer($hash);
|
RemoveInternalTimer($hash);
|
||||||
@ -1669,13 +1688,23 @@ sub HMUARTLGW_Set($@)
|
|||||||
InternalTimer(gettimeofday()+20, "HMUARTLGW_RemoveHMPair", "hmPairForSec:".$name, 1);
|
InternalTimer(gettimeofday()+20, "HMUARTLGW_RemoveHMPair", "hmPairForSec:".$name, 1);
|
||||||
} elsif ($cmd eq "reopen") {
|
} elsif ($cmd eq "reopen") {
|
||||||
HMUARTLGW_Reopen($hash);
|
HMUARTLGW_Reopen($hash);
|
||||||
} elsif($cmd eq "close") {
|
} elsif ($cmd eq "close") {
|
||||||
HMUARTLGW_Undefine($hash, $name);
|
HMUARTLGW_Undefine($hash, $name);
|
||||||
readingsSingleUpdate($hash, "state", "closed", 1);
|
readingsSingleUpdate($hash, "state", "closed", 1);
|
||||||
$hash->{XmitOpen} = 0;
|
$hash->{XmitOpen} = 0;
|
||||||
} elsif($cmd eq "open") {
|
} elsif ($cmd eq "open") {
|
||||||
DevIo_OpenDev($hash, 0, "HMUARTLGW_DoInit", \&HMUARTLGW_Connect);
|
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);
|
HMUARTLGW_send($hash, HMUARTLGW_OS_CHANGE_APP, HMUARTLGW_DST_OS);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1858,6 +1887,9 @@ sub HMUARTLGW_updateCondition($)
|
|||||||
if ($hash->{DevState} == HMUARTLGW_STATE_NONE) {
|
if ($hash->{DevState} == HMUARTLGW_STATE_NONE) {
|
||||||
$cond = "disconnected";
|
$cond = "disconnected";
|
||||||
$loadLvl = "suspended";
|
$loadLvl = "suspended";
|
||||||
|
} elsif ($hash->{DevState} == HMUARTLGW_STATE_UPDATE_COPRO) {
|
||||||
|
$cond = "fwupdate";
|
||||||
|
$loadLvl = "suspended";
|
||||||
}
|
}
|
||||||
|
|
||||||
readingsBeginUpdate($hash);
|
readingsBeginUpdate($hash);
|
||||||
@ -2107,6 +2139,96 @@ sub HMUARTLGW_decrypt($$)
|
|||||||
$plaintext;
|
$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($$$$) {
|
sub HMUARTLGW_getVerbLvl($$$$) {
|
||||||
my ($hash, $src, $dst, $def) = @_;
|
my ($hash, $src, $dst, $def) = @_;
|
||||||
|
|
||||||
@ -2181,6 +2303,19 @@ sub HMUARTLGW_getVerbLvl($$$$) {
|
|||||||
<li>restart<br>
|
<li>restart<br>
|
||||||
Reboots the device.
|
Reboots the device.
|
||||||
</li>
|
</li>
|
||||||
|
<li>updateCoPro </path/to/firmware.eq3><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>
|
</ul>
|
||||||
<br>
|
<br>
|
||||||
<a name="HMUARTLGW_get"></a>
|
<a name="HMUARTLGW_get"></a>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user