/* Copyright (C) 2017-2021 Andreas Shimokawa This file is part of Gadgetbridge. Gadgetbridge is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Gadgetbridge is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package nodomain.freeyourgadget.gadgetbridge.service.devices.huami; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.Map; import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice; import nodomain.freeyourgadget.gadgetbridge.util.ArrayUtils; import nodomain.freeyourgadget.gadgetbridge.util.CheckSums; public abstract class HuamiFirmwareInfo extends AbstractHuamiFirmwareInfo { protected static final byte[] RES_HEADER = new byte[]{ // HMRES resources file (*.res) 0x48, 0x4d, 0x52, 0x45, 0x53 }; protected static final byte[] NEWRES_HEADER = new byte[]{ // NERES resources file (*.res) 0x4e, 0x45, 0x52, 0x45, 0x53 }; protected static final byte[] WATCHFACE_HEADER = new byte[]{ 0x48, 0x4d, 0x44, 0x49, 0x41, 0x4c }; public static final byte[] UIHH_HEADER = new byte[]{ 'U', 'I', 'H', 'H' }; public static final byte[] AGPS_UIHH_HEADER = new byte[]{ 'U', 'I', 'H', 'H', 0x04 }; protected static final byte[] FT_HEADER = new byte[]{ // HMZK font file (*.ft, *.ft.xx) 0x48, 0x4d, 0x5a, 0x4b }; protected static final byte[] NEWFT_HEADER = new byte[]{ // NEZK font file (*.ft, *.ft.xx) 0x4e, 0x45, 0x5a, 0x4b }; public static final byte[] GPS_ALMANAC_HEADER = new byte[]{ // probably wrong (byte) 0xa0, (byte) 0x80, 0x08, 0x00, (byte) 0x8b }; public static final byte[] GPS_CEP_HEADER = new byte[]{ // probably wrong 0x2a, 0x12, (byte) 0xa0, 0x02 }; // gps detection is totally bogus, just the first 16 bytes protected static final byte[][] GPS_HEADERS = { new byte[]{ (byte) 0xcb, 0x51, (byte) 0xc1, 0x30, 0x41, (byte) 0x9e, 0x5e, (byte) 0xd3, 0x51, 0x35, (byte) 0xdf, 0x66, (byte) 0xed, (byte) 0xd9, 0x5f, (byte) 0xa7 }, new byte[]{ 0x10, 0x50, 0x26, 0x76, (byte) 0x8f, 0x4a, (byte) 0xa1, 0x49, (byte) 0xa7, 0x26, (byte) 0xd0, (byte) 0xe6, 0x4a, 0x21, (byte) 0x88, (byte) 0xd4 }, new byte[]{ (byte) 0xeb, (byte) 0xfa, (byte) 0xc5, (byte) 0x89, (byte) 0xf0, 0x5c, 0x2e, (byte) 0xcc, (byte) 0xfa, (byte) 0xf3, 0x62, (byte) 0xeb, (byte) 0x92, (byte) 0xc6, (byte) 0xa1, (byte) 0xbb }, new byte[]{ 0x0b, 0x61, 0x53, (byte) 0xed, (byte) 0x83, (byte) 0xac, 0x07, 0x21, (byte) 0x8c, 0x36, 0x2e, (byte) 0x8c, (byte) 0x9c, 0x08, 0x54, (byte) 0xa6 }, new byte[]{ (byte) 0xec, 0x51, 0x73, 0x22, 0x60, 0x02, 0x14, (byte) 0xb7, (byte) 0xb5, (byte) 0xea, 0x4b, 0x22, 0x5d, 0x23, (byte) 0xe5, 0x4f }, new byte[]{ 0x73, 0x75, 0x68, (byte) 0xd0, 0x70, 0x73, (byte) 0xbb, 0x5a, 0x3e, (byte) 0xc3, (byte) 0xd3, 0x09, (byte) 0x9e, 0x1d, (byte) 0xd3, (byte) 0xc9 }, new byte[]{ 0x00, 0x00, 0x00, 0x01, 0x00, 0x09, 0x6F, (byte) 0xD0, 0x00, 0x01, 0x00, 0x02, 0x3D, (byte) 0xE0, 0x00, 0x69 }, new byte[]{ 0x00, 0x00, 0x00, 0x01, 0x00, 0x09, 0x32, 0x70, 0x00, 0x01, 0x00, 0x02, 0x3D, (byte) 0xE0, 0x00, (byte) 0xCC } }; protected static final int FONT_TYPE_OFFSET = 0x9; protected static final int COMPRESSED_RES_HEADER_OFFSET = 0x9; protected static final int COMPRESSED_RES_HEADER_OFFSET_NEW = 0xd; @Override public String toVersion(int crc16) { final byte[] bytes = getBytes(); String version = getCrcMap().get(crc16); if (version == null) { switch (firmwareType) { case FIRMWARE: version = searchFirmwareVersion(bytes); break; case RES: version = "RES " + bytes[5]; break; case RES_COMPRESSED: byte versionByte; // there are two possible locations of the version for compressed res, probe the format in a dirty way :P if (bytes[COMPRESSED_RES_HEADER_OFFSET + 2] == 0x52 && bytes[COMPRESSED_RES_HEADER_OFFSET + 3] == 0x45 && bytes[COMPRESSED_RES_HEADER_OFFSET + 4] == 0x53) { versionByte = bytes[14]; } else { versionByte = bytes[18]; } version = "RES " + (versionByte & 0xff); break; case FONT: version = "FONT " + bytes[4]; break; case FONT_LATIN: version = "FONT LATIN " + bytes[4]; break; } } if (version == null) { switch (firmwareType) { case FIRMWARE: version = "(unknown)"; break; case FONT: case FONT_LATIN: version = "(unknown font)"; break; case GPS: version = "(unknown GPS)"; break; case GPS_CEP: version = "(unknown CEP)"; break; case GPS_ALMANAC: version = "(unknown ALM)"; break; case AGPS_UIHH: version = "(unknown AGPS)"; break; case WATCHFACE: version = "(unknown watchface)"; break; } } return version; } public HuamiFirmwareInfo(byte[] bytes) { super(bytes); } protected String searchFirmwareVersion(byte[] fwbytes) { ByteBuffer buf = ByteBuffer.wrap(fwbytes); buf.order(ByteOrder.BIG_ENDIAN); while (buf.remaining() > 3) { int word = buf.getInt(); if (word == 0x5625642e) { word = buf.getInt(); if (word == 0x25642e25) { word = buf.getInt(); if (word == 0x642e2564) { word = buf.getInt(); if (word == 0x00000000) { byte[] version = new byte[8]; buf.get(version); return new String(version); } } } } } return null; } }