Prepare v2023.10
[platform/kernel/u-boot.git] / common / spl / spl_legacy.c
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Copyright (C) 2020 Stefan Roese <sr@denx.de>
4  */
5
6 #include <common.h>
7 #include <image.h>
8 #include <log.h>
9 #include <malloc.h>
10 #include <asm/sections.h>
11 #include <spl.h>
12
13 #include <lzma/LzmaTypes.h>
14 #include <lzma/LzmaDec.h>
15 #include <lzma/LzmaTools.h>
16
17 #define LZMA_LEN        (1 << 20)
18
19 static void spl_parse_legacy_validate(uintptr_t start, uintptr_t size)
20 {
21         uintptr_t spl_start = (uintptr_t)_start;
22         uintptr_t spl_end = (uintptr_t)_image_binary_end;
23         uintptr_t end = start + size;
24
25         if ((start >= spl_start && start < spl_end) ||
26             (end > spl_start && end <= spl_end) ||
27             (start < spl_start && end >= spl_end) ||
28             (start > end && end > spl_start))
29                 panic("SPL: Image overlaps SPL\n");
30
31         if (size > CONFIG_SYS_BOOTM_LEN)
32                 panic("SPL: Image too large\n");
33 }
34
35 int spl_parse_legacy_header(struct spl_image_info *spl_image,
36                             const struct legacy_img_hdr *header)
37 {
38         u32 header_size = sizeof(struct legacy_img_hdr);
39
40         /* check uImage header CRC */
41         if (IS_ENABLED(CONFIG_SPL_LEGACY_IMAGE_CRC_CHECK) &&
42             !image_check_hcrc(header)) {
43                 puts("SPL: Image header CRC check failed!\n");
44                 return -EINVAL;
45         }
46
47         if (spl_image->flags & SPL_COPY_PAYLOAD_ONLY) {
48                 /*
49                  * On some system (e.g. powerpc), the load-address and
50                  * entry-point is located at address 0. We can't load
51                  * to 0-0x40. So skip header in this case.
52                  */
53                 spl_image->load_addr = image_get_load(header);
54                 spl_image->entry_point = image_get_ep(header);
55                 spl_image->size = image_get_data_size(header);
56         } else {
57                 spl_image->entry_point = image_get_ep(header);
58                 /* Load including the header */
59                 spl_image->load_addr = image_get_load(header) -
60                         header_size;
61                 spl_image->size = image_get_data_size(header) +
62                         header_size;
63         }
64
65 #ifdef CONFIG_SPL_LEGACY_IMAGE_CRC_CHECK
66         /* store uImage data length and CRC to check later */
67         spl_image->dcrc_data = image_get_load(header);
68         spl_image->dcrc_length = image_get_data_size(header);
69         spl_image->dcrc = image_get_dcrc(header);
70 #endif
71
72         spl_image->os = image_get_os(header);
73         spl_image->name = image_get_name(header);
74         debug(SPL_TPL_PROMPT
75               "payload image: %32s load addr: 0x%lx size: %d\n",
76               spl_image->name, spl_image->load_addr, spl_image->size);
77
78         spl_parse_legacy_validate(spl_image->load_addr, spl_image->size);
79         spl_parse_legacy_validate(spl_image->entry_point, 0);
80
81         return 0;
82 }
83
84 /*
85  * This function is added explicitly to avoid code size increase, when
86  * no compression method is enabled. The compiler will optimize the
87  * following switch/case statement in spl_load_legacy_img() away due to
88  * Dead Code Elimination.
89  */
90 static inline int spl_image_get_comp(const struct legacy_img_hdr *hdr)
91 {
92         if (IS_ENABLED(CONFIG_SPL_LZMA))
93                 return image_get_comp(hdr);
94
95         return IH_COMP_NONE;
96 }
97
98 int spl_load_legacy_img(struct spl_image_info *spl_image,
99                         struct spl_boot_device *bootdev,
100                         struct spl_load_info *load, ulong offset,
101                         struct legacy_img_hdr *hdr)
102 {
103         __maybe_unused SizeT lzma_len;
104         __maybe_unused void *src;
105         ulong dataptr;
106         int ret;
107
108         /*
109          * If the payload is compressed, the decompressed data should be
110          * directly write to its load address.
111          */
112         if (spl_image_get_comp(hdr) != IH_COMP_NONE)
113                 spl_image->flags |= SPL_COPY_PAYLOAD_ONLY;
114
115         ret = spl_parse_image_header(spl_image, bootdev, hdr);
116         if (ret)
117                 return ret;
118
119         /* Read image */
120         switch (spl_image_get_comp(hdr)) {
121         case IH_COMP_NONE:
122                 dataptr = offset;
123
124                 /*
125                  * Image header will be skipped only if SPL_COPY_PAYLOAD_ONLY
126                  * is set
127                  */
128                 if (spl_image->flags & SPL_COPY_PAYLOAD_ONLY)
129                         dataptr += sizeof(*hdr);
130
131                 load->read(load, dataptr, spl_image->size,
132                            (void *)(unsigned long)spl_image->load_addr);
133                 break;
134
135         case IH_COMP_LZMA:
136                 lzma_len = LZMA_LEN;
137
138                 /* dataptr points to compressed payload  */
139                 dataptr = offset + sizeof(*hdr);
140
141                 debug("LZMA: Decompressing %08lx to %08lx\n",
142                       dataptr, spl_image->load_addr);
143                 src = malloc(spl_image->size);
144                 if (!src) {
145                         printf("Unable to allocate %d bytes for LZMA\n",
146                                spl_image->size);
147                         return -ENOMEM;
148                 }
149
150                 load->read(load, dataptr, spl_image->size, src);
151                 ret = lzmaBuffToBuffDecompress((void *)spl_image->load_addr,
152                                                &lzma_len, src, spl_image->size);
153                 if (ret) {
154                         printf("LZMA decompression error: %d\n", ret);
155                         return ret;
156                 }
157
158                 spl_image->size = lzma_len;
159                 break;
160
161         default:
162                 debug("Compression method %s is not supported\n",
163                       genimg_get_comp_short_name(image_get_comp(hdr)));
164                 return -EINVAL;
165         }
166
167         return 0;
168 }