162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * purgatory: Runs between two kernels 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2022 Huawei Technologies Co, Ltd. 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Author: Li Zhengyu (lizhengyu3@huawei.com) 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci */ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/purgatory.h> 1262306a36Sopenharmony_ci#include <linux/kernel.h> 1362306a36Sopenharmony_ci#include <linux/string.h> 1462306a36Sopenharmony_ci#include <asm/string.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ciu8 purgatory_sha256_digest[SHA256_DIGEST_SIZE] __section(".kexec-purgatory"); 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_cistruct kexec_sha_region purgatory_sha_regions[KEXEC_SEGMENT_MAX] __section(".kexec-purgatory"); 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_cistatic int verify_sha256_digest(void) 2162306a36Sopenharmony_ci{ 2262306a36Sopenharmony_ci struct kexec_sha_region *ptr, *end; 2362306a36Sopenharmony_ci struct sha256_state ss; 2462306a36Sopenharmony_ci u8 digest[SHA256_DIGEST_SIZE]; 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci sha256_init(&ss); 2762306a36Sopenharmony_ci end = purgatory_sha_regions + ARRAY_SIZE(purgatory_sha_regions); 2862306a36Sopenharmony_ci for (ptr = purgatory_sha_regions; ptr < end; ptr++) 2962306a36Sopenharmony_ci sha256_update(&ss, (uint8_t *)(ptr->start), ptr->len); 3062306a36Sopenharmony_ci sha256_final(&ss, digest); 3162306a36Sopenharmony_ci if (memcmp(digest, purgatory_sha256_digest, sizeof(digest)) != 0) 3262306a36Sopenharmony_ci return 1; 3362306a36Sopenharmony_ci return 0; 3462306a36Sopenharmony_ci} 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci/* workaround for a warning with -Wmissing-prototypes */ 3762306a36Sopenharmony_civoid purgatory(void); 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_civoid purgatory(void) 4062306a36Sopenharmony_ci{ 4162306a36Sopenharmony_ci if (verify_sha256_digest()) 4262306a36Sopenharmony_ci for (;;) 4362306a36Sopenharmony_ci /* loop forever */ 4462306a36Sopenharmony_ci ; 4562306a36Sopenharmony_ci} 46