blob: 8774a7e3acea56521fa9358f371da85c4df6a5d2 [file] [log] [blame]
John Rigby34d12a12011-07-21 09:10:30 -04001/*
2 * (C) Copyright 2010
3 * Linaro LTD, www.linaro.org
4 * Author: John Rigby <john.rigby@linaro.org>
5 * Based on TI's signGP.c
6 *
7 * (C) Copyright 2009
8 * Stefano Babic, DENX Software Engineering, sbabic@denx.de.
9 *
10 * (C) Copyright 2008
11 * Marvell Semiconductor <www.marvell.com>
12 * Written-by: Prafulla Wadaskar <prafulla@marvell.com>
13 *
Wolfgang Denkbd8ec7e2013-10-07 13:07:26 +020014 * SPDX-License-Identifier: GPL-2.0+
John Rigby34d12a12011-07-21 09:10:30 -040015 */
16
John Rigby34d12a12011-07-21 09:10:30 -040017#include "mkimage.h"
18#include <image.h>
19#include "omapimage.h"
20
21/* Header size is CH header rounded up to 512 bytes plus GP header */
22#define OMAP_CH_HDR_SIZE 512
23#define OMAP_GP_HDR_SIZE (sizeof(struct gp_header))
24#define OMAP_FILE_HDR_SIZE (OMAP_CH_HDR_SIZE+OMAP_GP_HDR_SIZE)
25
Tom Rinid505e902012-10-16 13:06:06 +000026static int do_swap32 = 0;
27
28static uint32_t omapimage_swap32(uint32_t data)
29{
30 uint32_t result = 0;
31 result = (data & 0xFF000000) >> 24;
32 result |= (data & 0x00FF0000) >> 8;
33 result |= (data & 0x0000FF00) << 8;
34 result |= (data & 0x000000FF) << 24;
35 return result;
36}
37
John Rigby34d12a12011-07-21 09:10:30 -040038static uint8_t omapimage_header[OMAP_FILE_HDR_SIZE];
39
40static int omapimage_check_image_types(uint8_t type)
41{
42 if (type == IH_TYPE_OMAPIMAGE)
43 return EXIT_SUCCESS;
44 else {
John Rigby34d12a12011-07-21 09:10:30 -040045 return EXIT_FAILURE;
46 }
47}
48
49/*
50 * Only the simplest image type is currently supported:
51 * TOC pointing to CHSETTINGS
52 * TOC terminator
53 * CHSETTINGS
54 *
55 * padding to OMAP_CH_HDR_SIZE bytes
56 *
57 * gp header
58 * size
59 * load_addr
60 */
61static int valid_gph_size(uint32_t size)
62{
63 return size;
64}
65
66static int valid_gph_load_addr(uint32_t load_addr)
67{
68 return load_addr;
69}
70
71static int omapimage_verify_header(unsigned char *ptr, int image_size,
72 struct mkimage_params *params)
73{
74 struct ch_toc *toc = (struct ch_toc *)ptr;
75 struct gp_header *gph = (struct gp_header *)(ptr+OMAP_CH_HDR_SIZE);
Tom Rinid505e902012-10-16 13:06:06 +000076 uint32_t offset, size, gph_size, gph_load_addr;
John Rigby34d12a12011-07-21 09:10:30 -040077
78 while (toc->section_offset != 0xffffffff
79 && toc->section_size != 0xffffffff) {
Tom Rinid505e902012-10-16 13:06:06 +000080 if (do_swap32) {
81 offset = omapimage_swap32(toc->section_offset);
82 size = omapimage_swap32(toc->section_size);
83 } else {
84 offset = toc->section_offset;
85 size = toc->section_size;
86 }
John Rigby34d12a12011-07-21 09:10:30 -040087 if (!offset || !size)
88 return -1;
89 if (offset >= OMAP_CH_HDR_SIZE ||
90 offset+size >= OMAP_CH_HDR_SIZE)
91 return -1;
92 toc++;
93 }
Tom Rinid505e902012-10-16 13:06:06 +000094
95 if (do_swap32) {
96 gph_size = omapimage_swap32(gph->size);
97 gph_load_addr = omapimage_swap32(gph->load_addr);
98 } else {
99 gph_size = gph->size;
100 gph_load_addr = gph->load_addr;
101 }
102
103 if (!valid_gph_size(gph_size))
John Rigby34d12a12011-07-21 09:10:30 -0400104 return -1;
Tom Rinid505e902012-10-16 13:06:06 +0000105 if (!valid_gph_load_addr(gph_load_addr))
John Rigby34d12a12011-07-21 09:10:30 -0400106 return -1;
107
108 return 0;
109}
110
111static void omapimage_print_section(struct ch_settings *chs)
112{
113 const char *section_name;
114
115 if (chs->section_key)
116 section_name = "CHSETTINGS";
117 else
118 section_name = "UNKNOWNKEY";
119
120 printf("%s (%x) "
121 "valid:%x "
122 "version:%x "
123 "reserved:%x "
124 "flags:%x\n",
125 section_name,
126 chs->section_key,
127 chs->valid,
128 chs->version,
129 chs->reserved,
130 chs->flags);
131}
132
133static void omapimage_print_header(const void *ptr)
134{
135 const struct ch_toc *toc = (struct ch_toc *)ptr;
136 const struct gp_header *gph =
137 (struct gp_header *)(ptr+OMAP_CH_HDR_SIZE);
Tom Rinid505e902012-10-16 13:06:06 +0000138 uint32_t offset, size, gph_size, gph_load_addr;
John Rigby34d12a12011-07-21 09:10:30 -0400139
140 while (toc->section_offset != 0xffffffff
141 && toc->section_size != 0xffffffff) {
Tom Rinid505e902012-10-16 13:06:06 +0000142 if (do_swap32) {
143 offset = omapimage_swap32(toc->section_offset);
144 size = omapimage_swap32(toc->section_size);
145 } else {
146 offset = toc->section_offset;
147 size = toc->section_size;
148 }
John Rigby34d12a12011-07-21 09:10:30 -0400149
150 if (offset >= OMAP_CH_HDR_SIZE ||
151 offset+size >= OMAP_CH_HDR_SIZE)
152 exit(EXIT_FAILURE);
153
154 printf("Section %s offset %x length %x\n",
155 toc->section_name,
156 toc->section_offset,
157 toc->section_size);
158
159 omapimage_print_section((struct ch_settings *)(ptr+offset));
160 toc++;
161 }
162
Tom Rinid505e902012-10-16 13:06:06 +0000163 if (do_swap32) {
164 gph_size = omapimage_swap32(gph->size);
165 gph_load_addr = omapimage_swap32(gph->load_addr);
166 } else {
167 gph_size = gph->size;
168 gph_load_addr = gph->load_addr;
169 }
170
171 if (!valid_gph_size(gph_size)) {
172 fprintf(stderr, "Error: invalid image size %x\n", gph_size);
John Rigby34d12a12011-07-21 09:10:30 -0400173 exit(EXIT_FAILURE);
174 }
175
Tom Rinid505e902012-10-16 13:06:06 +0000176 if (!valid_gph_load_addr(gph_load_addr)) {
177 fprintf(stderr, "Error: invalid image load address %x\n",
178 gph_load_addr);
John Rigby34d12a12011-07-21 09:10:30 -0400179 exit(EXIT_FAILURE);
180 }
181
Tom Rinid505e902012-10-16 13:06:06 +0000182 printf("GP Header: Size %x LoadAddr %x\n", gph_size, gph_load_addr);
John Rigby34d12a12011-07-21 09:10:30 -0400183}
184
185static int toc_offset(void *hdr, void *member)
186{
187 return member - hdr;
188}
189
190static void omapimage_set_header(void *ptr, struct stat *sbuf, int ifd,
191 struct mkimage_params *params)
192{
193 struct ch_toc *toc = (struct ch_toc *)ptr;
194 struct ch_settings *chs = (struct ch_settings *)
195 (ptr + 2 * sizeof(*toc));
196 struct gp_header *gph = (struct gp_header *)(ptr + OMAP_CH_HDR_SIZE);
197
198 toc->section_offset = toc_offset(ptr, chs);
199 toc->section_size = sizeof(struct ch_settings);
200 strcpy((char *)toc->section_name, "CHSETTINGS");
201
202 chs->section_key = KEY_CHSETTINGS;
203 chs->valid = 0;
204 chs->version = 1;
205 chs->reserved = 0;
206 chs->flags = 0;
207
208 toc++;
209 memset(toc, 0xff, sizeof(*toc));
210
211 gph->size = sbuf->st_size - OMAP_FILE_HDR_SIZE;
212 gph->load_addr = params->addr;
Tom Rinid505e902012-10-16 13:06:06 +0000213
214 if (strncmp(params->imagename, "byteswap", 8) == 0) {
215 do_swap32 = 1;
216 int swapped = 0;
217 uint32_t *data = (uint32_t *)ptr;
218
219 while (swapped <= (sbuf->st_size / sizeof(uint32_t))) {
220 *data = omapimage_swap32(*data);
221 swapped++;
222 data++;
223 }
224 }
John Rigby34d12a12011-07-21 09:10:30 -0400225}
226
227int omapimage_check_params(struct mkimage_params *params)
228{
229 return (params->dflag && (params->fflag || params->lflag)) ||
230 (params->fflag && (params->dflag || params->lflag)) ||
231 (params->lflag && (params->dflag || params->fflag));
232}
233
234/*
235 * omapimage parameters
236 */
237static struct image_type_params omapimage_params = {
238 .name = "TI OMAP CH/GP Boot Image support",
239 .header_size = OMAP_FILE_HDR_SIZE,
240 .hdr = (void *)&omapimage_header,
241 .check_image_type = omapimage_check_image_types,
242 .verify_header = omapimage_verify_header,
243 .print_header = omapimage_print_header,
244 .set_header = omapimage_set_header,
245 .check_params = omapimage_check_params,
246};
247
248void init_omap_image_type(void)
249{
250 mkimage_register(&omapimage_params);
251}