2018-05-01 06:15:48 +01:00
|
|
|
/*
|
2018-08-05 12:40:32 +01:00
|
|
|
* Copyright (c) 2018 naehrwert
|
|
|
|
* Copyright (c) 2018 st4rk
|
|
|
|
* Copyright (c) 2018 CTCaer
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2018-05-01 06:15:48 +01:00
|
|
|
|
|
|
|
#include <string.h>
|
2018-08-05 12:40:32 +01:00
|
|
|
|
2018-05-01 06:15:48 +01:00
|
|
|
#include "pkg1.h"
|
2018-08-13 09:58:24 +01:00
|
|
|
#include "../utils/aarch64_util.h"
|
|
|
|
#include "../sec/se.h"
|
2018-05-01 06:15:48 +01:00
|
|
|
|
2018-06-01 17:28:22 +01:00
|
|
|
#define SM_100_ADR 0x4002B020
|
2018-05-01 06:15:48 +01:00
|
|
|
PATCHSET_DEF(_secmon_1_patchset,
|
2018-08-05 12:40:32 +01:00
|
|
|
// Patch the relocator to be able to run from SM_100_ADR.
|
2018-06-15 12:28:27 +01:00
|
|
|
{ 0x1E0, _ADRP(0, 0x7C013000 - _PAGEOFF(SM_100_ADR)) },
|
2018-05-01 06:15:48 +01:00
|
|
|
//Patch package2 decryption and signature/hash checks.
|
|
|
|
{ 0x9F0 + 0xADC, _NOP() }, //Header signature.
|
|
|
|
{ 0x9F0 + 0xB8C, _NOP() }, //Version.
|
2018-05-15 18:15:22 +01:00
|
|
|
{ 0x9F0 + 0xBB0, _NOP() } //Sections SHA2.
|
2018-05-01 06:15:48 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
PATCHSET_DEF(_secmon_2_patchset,
|
2018-08-05 12:40:32 +01:00
|
|
|
// Patch package2 decryption and signature/hash checks.
|
2018-05-01 06:15:48 +01:00
|
|
|
{ 0xAC8 + 0xAAC, _NOP() }, //Header signature.
|
|
|
|
{ 0xAC8 + 0xB3C, _NOP() }, //Version.
|
2018-05-15 18:15:22 +01:00
|
|
|
{ 0xAC8 + 0xB58, _NOP() } //Sections SHA2.
|
2018-05-01 06:15:48 +01:00
|
|
|
);
|
|
|
|
|
2018-05-04 08:15:29 +01:00
|
|
|
PATCHSET_DEF(_secmon_3_patchset,
|
2018-08-05 12:40:32 +01:00
|
|
|
// Patch package2 decryption and signature/hash checks.
|
2018-05-04 08:15:29 +01:00
|
|
|
{ 0xAC8 + 0xA30, _NOP() }, //Header signature.
|
2018-06-09 13:05:29 +01:00
|
|
|
{ 0xAC8 + 0xAB4, _NOP() }, //package2 structure.
|
2018-05-04 08:15:29 +01:00
|
|
|
{ 0xAC8 + 0xAC0, _NOP() }, //Version.
|
2018-05-15 18:15:22 +01:00
|
|
|
{ 0xAC8 + 0xADC, _NOP() } //Sections SHA2.
|
2018-05-04 08:15:29 +01:00
|
|
|
);
|
|
|
|
|
2018-06-01 17:22:11 +01:00
|
|
|
PATCHSET_DEF(_secmon_4_patchset,
|
2018-08-05 12:40:32 +01:00
|
|
|
// Patch package2 decryption and signature/hash checks.
|
2018-06-09 13:05:29 +01:00
|
|
|
{ 0x2300 + 0x5D80, _NOP() }, //package2 structure.
|
|
|
|
{ 0x2300 + 0x5D8C, _NOP() }, //Version.
|
|
|
|
{ 0x2300 + 0x5EFC, _NOP() }, //Header signature.
|
2018-07-04 23:51:36 +01:00
|
|
|
{ 0xAC8 + 0xA2C, _NOP() } //Sections SHA2.
|
2018-05-13 02:13:17 +01:00
|
|
|
);
|
|
|
|
|
2018-06-01 17:22:11 +01:00
|
|
|
PATCHSET_DEF(_secmon_5_patchset,
|
2018-08-05 12:40:32 +01:00
|
|
|
// Patch package2 decryption and signature/hash checks.
|
2018-06-09 13:05:29 +01:00
|
|
|
{ 0xDA8 + 0x9D8 , _NOP() }, //package2 structure.
|
|
|
|
{ 0xDA8 + 0x9E4 , _NOP() }, //Version.
|
|
|
|
{ 0xDA8 + 0xC9C , _NOP() }, //Header signature.
|
|
|
|
{ 0xDA8 + 0x1038 , _NOP() } //Sections SHA2.
|
2018-05-13 02:13:17 +01:00
|
|
|
);
|
|
|
|
|
2018-05-01 06:15:48 +01:00
|
|
|
/*
|
2018-08-05 12:40:32 +01:00
|
|
|
* package1.1 header: <wb, ldr, sm>
|
|
|
|
* package1.1 layout:
|
|
|
|
* 1.0: {sm, ldr, wb} { 2, 1, 0 }
|
|
|
|
* 2.0: {wb, ldr, sm} { 0, 1, 2 }
|
|
|
|
* 3.0: {wb, ldr, sm} { 0, 1, 2 }
|
|
|
|
* 3.1: {wb, ldr, sm} { 0, 1, 2 }
|
|
|
|
* 4.0: {ldr, sm, wb} { 1, 2, 0 }
|
|
|
|
* 5.0: {ldr, sm, wb} { 1, 2, 0 }
|
|
|
|
*/
|
2018-05-01 06:15:48 +01:00
|
|
|
|
|
|
|
static const pkg1_id_t _pkg1_ids[] = {
|
2018-08-13 10:12:53 +01:00
|
|
|
{ "20161121183008", 0, 0x1900, 0x3FE0, { 2, 1, 0 }, SM_100_ADR, 0x8000D000, true, _secmon_1_patchset }, //1.0.0 (Patched relocator)
|
|
|
|
{ "20170210155124", 0, 0x1900, 0x3FE0, { 0, 1, 2 }, 0x4002D000, 0x8000D000, true, _secmon_2_patchset }, //2.0.0 - 2.3.0
|
|
|
|
{ "20170519101410", 1, 0x1A00, 0x3FE0, { 0, 1, 2 }, 0x4002D000, 0x8000D000, true, _secmon_3_patchset }, //3.0.0
|
|
|
|
{ "20170710161758", 2, 0x1A00, 0x3FE0, { 0, 1, 2 }, 0x4002D000, 0x8000D000, true, _secmon_3_patchset }, //3.0.1 - 3.0.2
|
|
|
|
{ "20170921172629", 3, 0x1800, 0x3FE0, { 1, 2, 0 }, 0x4002B000, 0x4003B000, false, _secmon_4_patchset }, //4.0.0 - 4.1.0
|
|
|
|
{ "20180220163747", 4, 0x1900, 0x3FE0, { 1, 2, 0 }, 0x4002B000, 0x4003B000, false, _secmon_5_patchset }, //5.0.0 - 5.1.0
|
2018-06-08 10:42:24 +01:00
|
|
|
{ NULL } //End.
|
2018-05-01 06:15:48 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const pkg1_id_t *pkg1_identify(u8 *pkg1)
|
|
|
|
{
|
|
|
|
for (u32 i = 0; _pkg1_ids[i].id; i++)
|
|
|
|
if (!memcmp(pkg1 + 0x10, _pkg1_ids[i].id, 12))
|
|
|
|
return &_pkg1_ids[i];
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pkg1_decrypt(const pkg1_id_t *id, u8 *pkg1)
|
|
|
|
{
|
2018-08-05 12:40:32 +01:00
|
|
|
// Decrypt package1.
|
2018-05-01 06:15:48 +01:00
|
|
|
u8 *pkg11 = pkg1 + id->pkg11_off;
|
|
|
|
u32 pkg11_size = *(u32 *)pkg11;
|
|
|
|
se_aes_crypt_ctr(11, pkg11 + 0x20, pkg11_size, pkg11 + 0x20, pkg11_size, pkg11 + 0x10);
|
|
|
|
}
|
|
|
|
|
2018-06-04 23:55:32 +01:00
|
|
|
void pkg1_unpack(void *warmboot_dst, void *secmon_dst, void *ldr_dst, const pkg1_id_t *id, u8 *pkg1)
|
2018-05-01 06:15:48 +01:00
|
|
|
{
|
|
|
|
pk11_hdr_t *hdr = (pk11_hdr_t *)(pkg1 + id->pkg11_off + 0x20);
|
|
|
|
|
|
|
|
u32 sec_size[3] = { hdr->wb_size, hdr->ldr_size, hdr->sm_size };
|
|
|
|
//u32 sec_off[3] = { hdr->wb_off, hdr->ldr_off, hdr->sm_off };
|
|
|
|
|
|
|
|
u8 *pdata = (u8 *)hdr + sizeof(pk11_hdr_t);
|
|
|
|
for (u32 i = 0; i < 3; i++)
|
|
|
|
{
|
|
|
|
if (id->sec_map[i] == 0 && warmboot_dst)
|
|
|
|
memcpy(warmboot_dst, pdata, sec_size[id->sec_map[i]]);
|
2018-06-04 23:55:32 +01:00
|
|
|
else if (id->sec_map[i] == 1 && ldr_dst)
|
|
|
|
memcpy(ldr_dst, pdata, sec_size[id->sec_map[i]]);
|
2018-05-01 06:15:48 +01:00
|
|
|
else if (id->sec_map[i] == 2 && secmon_dst)
|
|
|
|
memcpy(secmon_dst, pdata, sec_size[id->sec_map[i]]);
|
|
|
|
pdata += sec_size[id->sec_map[i]];
|
|
|
|
}
|
|
|
|
}
|