blob: 6084436b48b49ae0bc6179118fdd4b76fa7dbfed [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{
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100106 if (valid_cfgblock) {
107 printf("Model: Toradex %04d %s %s\n",
108 tdx_hw_tag.prodid,
109 toradex_modules[tdx_hw_tag.prodid].name,
110 tdx_board_rev_str);
111 printf("Serial#: %s\n", tdx_serial_str);
112 }
113
114#ifdef CONFIG_TDX_CFG_BLOCK_EXTRA
115 if (tdx_carrier_board_name)
116 printf("Carrier: Toradex %s %s, Serial# %s\n",
117 tdx_carrier_board_name,
118 tdx_car_rev_str,
119 tdx_car_serial_str);
120#endif
121
122 print_bootinfo();
123
124 return 0;
125}
126
127static int settings_r(void)
128{
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100129 unsigned char ethaddr[6];
130
131 if (read_tdx_cfg_block()) {
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100132 printf("MISSING TORADEX CONFIG BLOCK\n");
Philippe Schenkerd52a2572022-06-20 16:57:45 +0200133 get_mac_from_serial(tdx_serial, &tdx_eth_addr);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100134
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100135 /* Board can run even if config block is not present */
136 return 0;
137 }
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100138
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100139 snprintf(tdx_serial_str, sizeof(tdx_serial_str),
140 "%08u", tdx_serial);
141 snprintf(tdx_board_rev_str, sizeof(tdx_board_rev_str),
142 "V%1d.%1d%s",
143 tdx_hw_tag.ver_major,
144 tdx_hw_tag.ver_minor,
145 get_board_assembly(tdx_hw_tag.ver_assembly));
146
147 env_set("serial#", tdx_serial_str);
148
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300149#ifdef CONFIG_TDX_CFG_BLOCK_EXTRA
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100150 if (read_tdx_cfg_block_carrier()) {
151 printf("MISSING TORADEX CARRIER CONFIG BLOCKS\n");
152 try_migrate_tdx_cfg_block_carrier();
153 } else {
154 tdx_carrier_board_name =
155 get_toradex_carrier_boards(tdx_car_hw_tag.prodid);
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300156
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100157 snprintf(tdx_car_serial_str, sizeof(tdx_car_serial_str),
158 "%08u", tdx_car_serial);
159 snprintf(tdx_car_rev_str, sizeof(tdx_car_rev_str),
160 "V%1d.%1d%s",
161 tdx_car_hw_tag.ver_major,
162 tdx_car_hw_tag.ver_minor,
163 get_board_assembly(tdx_car_hw_tag.ver_assembly));
Igor Opaniukecfe0f52020-07-15 13:30:57 +0300164
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100165 env_set("carrier_serial#", tdx_car_serial_str);
Bhuvanchandra DV2190ade2019-03-25 17:18:28 +0100166 }
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100167#endif
Emanuele Ghidoli1e9459e2024-02-23 10:11:38 +0100168
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100169 /*
170 * Check if environment contains a valid MAC address,
171 * set the one from config block if not
172 */
Simon Glass399a9ce2017-08-03 12:22:14 -0600173 if (!eth_env_get_enetaddr("ethaddr", ethaddr))
Simon Glass8551d552017-08-03 12:22:11 -0600174 eth_env_set_enetaddr("ethaddr", (u8 *)&tdx_eth_addr);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100175
Francesco Dolcini14163d62022-07-21 15:17:36 +0200176 if (IS_ENABLED(CONFIG_TDX_CFG_BLOCK_2ND_ETHADDR) &&
177 !eth_env_get_enetaddr("eth1addr", ethaddr)) {
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100178 /*
179 * Secondary MAC address is allocated from block
180 * 0x100000 higher then the first MAC address
181 */
182 memcpy(ethaddr, &tdx_eth_addr, 6);
183 ethaddr[3] += 0x10;
Simon Glass8551d552017-08-03 12:22:11 -0600184 eth_env_set_enetaddr("eth1addr", ethaddr);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100185 }
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100186
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100187 return 0;
188}
Emanuele Ghidoli1121f342024-02-23 10:11:39 +0100189EVENT_SPY_SIMPLE(EVT_SETTINGS_R, settings_r);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100190
Stefan Agner57985bd2021-07-23 09:39:46 +0300191#ifdef CONFIG_TDX_CFG_BLOCK_USB_GADGET_PID
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100192int g_dnl_bind_fixup(struct usb_device_descriptor *dev, const char *name)
193{
194 unsigned short usb_pid;
195
196 usb_pid = TORADEX_USB_PRODUCT_NUM_OFFSET + tdx_hw_tag.prodid;
197 put_unaligned(usb_pid, &dev->idProduct);
198
199 return 0;
200}
Stefan Agnera1886522016-11-30 13:41:52 -0800201#endif
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100202
Stefan Agner98ffd0f2016-11-30 13:41:53 -0800203#if defined(CONFIG_OF_LIBFDT)
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900204int ft_common_board_setup(void *blob, struct bd_info *bd)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100205{
206 if (tdx_serial) {
207 fdt_setprop(blob, 0, "serial-number", tdx_serial_str,
208 strlen(tdx_serial_str) + 1);
209 }
210
211 if (tdx_hw_tag.ver_major) {
212 char prod_id[5];
213
Philippe Schenker38cf79e2022-06-13 19:35:21 +0200214 snprintf(prod_id, sizeof(prod_id), "%04u", tdx_hw_tag.prodid);
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100215 fdt_setprop(blob, 0, "toradex,product-id", prod_id, 5);
216
217 fdt_setprop(blob, 0, "toradex,board-rev", tdx_board_rev_str,
218 strlen(tdx_board_rev_str) + 1);
219 }
220
221 return 0;
222}
223#endif
224
225#else /* CONFIG_TDX_CFG_BLOCK */
226
Tom Rini4cc38852021-08-30 09:16:30 -0400227#if defined(CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG)
228u32 get_board_revision(void)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100229{
230 return 0;
231}
232#endif /* CONFIG_REVISION_TAG */
233
234#ifdef CONFIG_SERIAL_TAG
235u32 get_board_serial(void)
236{
237 return 0;
238}
239#endif /* CONFIG_SERIAL_TAG */
240
Masahiro Yamadaf7ed78b2020-06-26 15:13:33 +0900241int ft_common_board_setup(void *blob, struct bd_info *bd)
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100242{
243 return 0;
244}
Marcel Ziswiler7a28dfc2016-11-16 17:49:22 +0100245
246#endif /* CONFIG_TDX_CFG_BLOCK */