blob: cdf645b141774bcb77c2a03da08b9cae7549cf5c [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +01002/*
3 * Copyright (c) 2016 Toradex, Inc.
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +01004 */
5
6#include <common.h>
Simon Glass5e6201b2019-08-01 09:46:51 -06007#include <env.h>
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +01008#include <g_dnl.h>
Simon Glass97589732020-05-10 11:40:02 -06009#include <init.h>
Masahiro Yamada75f82d02018-03-05 01:20:11 +090010#include <linux/libfdt.h>
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010011
Simon Glass52cb5042022-10-18 07:46:31 -060012#ifdef CONFIG_VIDEO
Igor Opaniuk897822d2020-07-15 13:31:01 +030013#include <bmp_logo.h>
14#include <dm.h>
15#include <splash.h>
16#include <video.h>
17#endif
18
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010019#include "tdx-cfg-block.h"
Simon Glassd9a766f2017-05-17 08:23:00 -060020#include <asm/setup.h>
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010021#include "tdx-common.h"
22
Philippe Schenker1832bd32022-06-13 19:35:22 +020023#define SERIAL_STR_LEN 8
24#define MODULE_VER_STR_LEN 4 // V1.1
Philippe Schenker498f95a2022-06-13 19:35:23 +020025#define MODULE_REV_STR_LEN 3 // [A-Z] or #[26-99]
Philippe Schenker1832bd32022-06-13 19:35:22 +020026
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010027#ifdef CONFIG_TDX_CFG_BLOCK
Philippe Schenker1832bd32022-06-13 19:35:22 +020028static char tdx_serial_str[SERIAL_STR_LEN + 1];
29static char tdx_board_rev_str[MODULE_VER_STR_LEN + MODULE_REV_STR_LEN + 1];
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010030
Igor Opaniukecfe0f52020-07-15 13:30:57 +030031#ifdef CONFIG_TDX_CFG_BLOCK_EXTRA
Philippe Schenker1832bd32022-06-13 19:35:22 +020032static char tdx_car_serial_str[SERIAL_STR_LEN + 1];
33static char tdx_car_rev_str[MODULE_VER_STR_LEN + MODULE_REV_STR_LEN + 1];
Max Krummenacher2ded2d62023-07-18 11:07:33 +020034static const char *tdx_carrier_board_name;
Igor Opaniukecfe0f52020-07-15 13:30:57 +030035#endif
36
Tom Rini4cc38852021-08-30 09:16:30 -040037#if defined(CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG)
38u32 get_board_revision(void)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +010039{
40 /* Check validity */
41 if (!tdx_hw_tag.ver_major)
42 return 0;
43
44 return ((tdx_hw_tag.ver_major & 0xff) << 8) |
45 ((tdx_hw_tag.ver_minor & 0xf) << 4) |
46 ((tdx_hw_tag.ver_assembly & 0xf) + 0xa);
47}
48#endif /* CONFIG_TDX_CFG_BLOCK */
49
50#ifdef CONFIG_SERIAL_TAG
51void get_board_serial(struct tag_serialnr *serialnr)
52{
53 int array[8];
54 unsigned int serial = tdx_serial;
55 int i;
56
57 serialnr->low = 0;
58 serialnr->high = 0;
59
60 /* Check validity */
61 if (serial) {
62 /*
63 * Convert to Linux serial number format (hexadecimal coded
64 * decimal)
65 */
66 i = 7;
67 while (serial) {
68 array[i--] = serial % 10;
69 serial /= 10;
70 }
71 while (i >= 0)
72 array[i--] = 0;
73 serial = array[0];
74 for (i = 1; i < 8; i++) {
75 serial *= 16;
76 serial += array[i];
77 }
78
79 serialnr->low = serial;
80 }
81}
82#endif /* CONFIG_SERIAL_TAG */
83
Philippe Schenker498f95a2022-06-13 19:35:23 +020084static const char *get_board_assembly(u16 ver_assembly)
85{
86 static char ver_name[MODULE_REV_STR_LEN + 1];
87
88 if (ver_assembly < 26) {
89 ver_name[0] = (char)ver_assembly + 'A';
90 ver_name[1] = '\0';
91 } else {
92 snprintf(ver_name, sizeof(ver_name),
93 "#%u", ver_assembly);
94 }
95
96 return ver_name;
97}
98
Emanuele Ghidoli1e9459e2024-02-23 10:11:38 +010099__weak int print_bootinfo(void)
100{
101 return 0;
102}
103
104int checkboard(void)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100105{
106 unsigned char ethaddr[6];
107
108 if (read_tdx_cfg_block()) {
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100109 printf("MISSING TORADEX CONFIG BLOCK\n");
Philippe Schenkerd52a2572022-06-20 16:57:45 +0200110 get_mac_from_serial(tdx_serial, &tdx_eth_addr);
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100111 } else {
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200112 snprintf(tdx_serial_str, sizeof(tdx_serial_str),
113 "%08u", tdx_serial);
114 snprintf(tdx_board_rev_str, sizeof(tdx_board_rev_str),
Philippe Schenker498f95a2022-06-13 19:35:23 +0200115 "V%1d.%1d%s",
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200116 tdx_hw_tag.ver_major,
117 tdx_hw_tag.ver_minor,
Philippe Schenker498f95a2022-06-13 19:35:23 +0200118 get_board_assembly(tdx_hw_tag.ver_assembly));
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100119
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100120 env_set("serial#", tdx_serial_str);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100121
Francesco Dolcini6e9ba382022-07-21 15:17:38 +0200122 printf("Model: Toradex %04d %s %s\n",
123 tdx_hw_tag.prodid,
Francesco Dolcinifed3bc72022-07-21 15:17:34 +0200124 toradex_modules[tdx_hw_tag.prodid].name,
Francesco Dolcini6e9ba382022-07-21 15:17:38 +0200125 tdx_board_rev_str);
126 printf("Serial#: %s\n", tdx_serial_str);
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300127#ifdef CONFIG_TDX_CFG_BLOCK_EXTRA
128 if (read_tdx_cfg_block_carrier()) {
129 printf("MISSING TORADEX CARRIER CONFIG BLOCKS\n");
130 try_migrate_tdx_cfg_block_carrier();
131 } else {
Max Krummenacher2ded2d62023-07-18 11:07:33 +0200132 tdx_carrier_board_name =
133 get_toradex_carrier_boards(tdx_car_hw_tag.prodid);
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300134
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200135 snprintf(tdx_car_serial_str, sizeof(tdx_car_serial_str),
136 "%08u", tdx_car_serial);
137 snprintf(tdx_car_rev_str, sizeof(tdx_car_rev_str),
Philippe Schenker498f95a2022-06-13 19:35:23 +0200138 "V%1d.%1d%s",
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200139 tdx_car_hw_tag.ver_major,
140 tdx_car_hw_tag.ver_minor,
Philippe Schenker498f95a2022-06-13 19:35:23 +0200141 get_board_assembly(tdx_car_hw_tag.ver_assembly));
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300142
143 env_set("carrier_serial#", tdx_car_serial_str);
144 printf("Carrier: Toradex %s %s, Serial# %s\n",
145 tdx_carrier_board_name,
146 tdx_car_rev_str,
147 tdx_car_serial_str);
148 }
149#endif
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100150 }
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100151
Emanuele Ghidoli1e9459e2024-02-23 10:11:38 +0100152 print_bootinfo();
153
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100154 /*
155 * Check if environment contains a valid MAC address,
156 * set the one from config block if not
157 */
Simon Glass399a9ce2017-08-03 12:22:14 -0600158 if (!eth_env_get_enetaddr("ethaddr", ethaddr))
Simon Glass8551d552017-08-03 12:22:11 -0600159 eth_env_set_enetaddr("ethaddr", (u8 *)&tdx_eth_addr);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100160
Francesco Dolcini14163d62022-07-21 15:17:36 +0200161 if (IS_ENABLED(CONFIG_TDX_CFG_BLOCK_2ND_ETHADDR) &&
162 !eth_env_get_enetaddr("eth1addr", ethaddr)) {
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100163 /*
164 * Secondary MAC address is allocated from block
165 * 0x100000 higher then the first MAC address
166 */
167 memcpy(ethaddr, &tdx_eth_addr, 6);
168 ethaddr[3] += 0x10;
Simon Glass8551d552017-08-03 12:22:11 -0600169 eth_env_set_enetaddr("eth1addr", ethaddr);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100170 }
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100171
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100172 return 0;
173}
174
Stefan Agner57985bd2021-07-23 09:39:46 +0300175#ifdef CONFIG_TDX_CFG_BLOCK_USB_GADGET_PID
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100176int g_dnl_bind_fixup(struct usb_device_descriptor *dev, const char *name)
177{
178 unsigned short usb_pid;
179
180 usb_pid = TORADEX_USB_PRODUCT_NUM_OFFSET + tdx_hw_tag.prodid;
181 put_unaligned(usb_pid, &dev->idProduct);
182
183 return 0;
184}
Stefan Agnera1886522016-11-30 13:41:52 -0800185#endif
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100186
Stefan Agner98ffd0f2016-11-30 13:41:53 -0800187#if defined(CONFIG_OF_LIBFDT)
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900188int ft_common_board_setup(void *blob, struct bd_info *bd)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100189{
190 if (tdx_serial) {
191 fdt_setprop(blob, 0, "serial-number", tdx_serial_str,
192 strlen(tdx_serial_str) + 1);
193 }
194
195 if (tdx_hw_tag.ver_major) {
196 char prod_id[5];
197
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200198 snprintf(prod_id, sizeof(prod_id), "%04u", tdx_hw_tag.prodid);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100199 fdt_setprop(blob, 0, "toradex,product-id", prod_id, 5);
200
201 fdt_setprop(blob, 0, "toradex,board-rev", tdx_board_rev_str,
202 strlen(tdx_board_rev_str) + 1);
203 }
204
205 return 0;
206}
207#endif
208
209#else /* CONFIG_TDX_CFG_BLOCK */
210
Tom Rini4cc38852021-08-30 09:16:30 -0400211#if defined(CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG)
212u32 get_board_revision(void)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100213{
214 return 0;
215}
216#endif /* CONFIG_REVISION_TAG */
217
218#ifdef CONFIG_SERIAL_TAG
219u32 get_board_serial(void)
220{
221 return 0;
222}
223#endif /* CONFIG_SERIAL_TAG */
224
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900225int ft_common_board_setup(void *blob, struct bd_info *bd)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100226{
227 return 0;
228}
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100229
230#endif /* CONFIG_TDX_CFG_BLOCK */