blob: 4af3cbe9fe2539bead0b36183c087a57a118e5ba [file] [log] [blame]
Marek Vasutf670cd72022-05-21 16:56:26 +02001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright 2022 Marek Vasut <marex@denx.de>
4 */
5
Marek Vasutf670cd72022-05-21 16:56:26 +02006#include <asm/arch/clock.h>
Marek Vasuteae88ad2023-12-16 06:42:29 +01007#include <asm/arch/ddr.h>
Marek Vasutf670cd72022-05-21 16:56:26 +02008#include <asm/arch/sys_proto.h>
9#include <asm/io.h>
10#include <dm.h>
11#include <env.h>
12#include <env_internal.h>
13#include <i2c_eeprom.h>
14#include <malloc.h>
15#include <net.h>
16#include <miiphy.h>
Marek Vasut891c5ef2024-09-25 00:08:23 +020017#include <power/regulator.h>
Marek Vasutf670cd72022-05-21 16:56:26 +020018
19#include "lpddr4_timing.h"
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020020#include "../common/dh_common.h"
21#include "../common/dh_imx.h"
Marek Vasutf670cd72022-05-21 16:56:26 +020022
23DECLARE_GLOBAL_DATA_PTR;
24
25int mach_cpu_init(void)
26{
27 icache_enable();
28 return 0;
29}
30
31int board_phys_sdram_size(phys_size_t *size)
32{
33 const u16 memsz[] = { 512, 1024, 1536, 2048, 3072, 4096, 6144, 8192 };
Marek Vasuteae88ad2023-12-16 06:42:29 +010034 const u8 ecc = readl(DDRC_ECCCFG0(0)) & DDRC_ECCCFG0_ECC_MODE_MASK;
Marek Vasutf670cd72022-05-21 16:56:26 +020035 u8 memcfg = dh_get_memcfg();
36
Marek Vasuteae88ad2023-12-16 06:42:29 +010037 /* 896 kiB, i.e. 1 MiB without 12.5% reserved for in-band ECC */
38 *size = (u64)memsz[memcfg] * (SZ_1M - (ecc ? (SZ_1M / 8) : 0));
Marek Vasutf670cd72022-05-21 16:56:26 +020039
40 return 0;
41}
42
Christoph Niedermaier276458b2024-12-07 00:04:18 +010043static int dh_imx8_setup_ethaddr(struct eeprom_id_page *eip)
Marek Vasutf670cd72022-05-21 16:56:26 +020044{
45 unsigned char enetaddr[6];
Marek Vasutf670cd72022-05-21 16:56:26 +020046
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020047 if (dh_mac_is_in_env("ethaddr"))
48 return 0;
Marek Vasutf670cd72022-05-21 16:56:26 +020049
Marek Vasut29ab1a92024-03-12 22:15:58 +010050 if (dh_get_mac_is_enabled("ethernet0"))
51 return 0;
52
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020053 if (!dh_imx_get_mac_from_fuse(enetaddr))
54 goto out;
Marek Vasutf670cd72022-05-21 16:56:26 +020055
Christoph Niedermaier276458b2024-12-07 00:04:18 +010056 if (!dh_get_value_from_eeprom_buffer(DH_MAC0, enetaddr, sizeof(enetaddr), eip))
57 goto out;
58
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020059 if (!dh_get_mac_from_eeprom(enetaddr, "eeprom0"))
60 goto out;
61
62 return -ENXIO;
63
64out:
65 return eth_env_set_enetaddr("ethaddr", enetaddr);
66}
67
Christoph Niedermaier276458b2024-12-07 00:04:18 +010068static int dh_imx8_setup_eth1addr(struct eeprom_id_page *eip)
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020069{
70 unsigned char enetaddr[6];
71
72 if (dh_mac_is_in_env("eth1addr"))
73 return 0;
74
Marek Vasut29ab1a92024-03-12 22:15:58 +010075 if (dh_get_mac_is_enabled("ethernet1"))
76 return 0;
77
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020078 if (!dh_imx_get_mac_from_fuse(enetaddr))
79 goto increment_out;
80
Christoph Niedermaier276458b2024-12-07 00:04:18 +010081 if (!dh_get_value_from_eeprom_buffer(DH_MAC1, enetaddr, sizeof(enetaddr), eip))
82 goto out;
83
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020084 if (!dh_get_mac_from_eeprom(enetaddr, "eeprom1"))
85 goto out;
Marek Vasutf670cd72022-05-21 16:56:26 +020086
87 /*
88 * Populate second ethernet MAC from first ethernet EEPROM with MAC
89 * address LSByte incremented by 1. This is only used on SoMs without
90 * second ethernet EEPROM, i.e. early prototypes.
91 */
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020092 if (!dh_get_mac_from_eeprom(enetaddr, "eeprom0"))
93 goto increment_out;
Marek Vasutf670cd72022-05-21 16:56:26 +020094
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020095 return -ENXIO;
Marek Vasutf670cd72022-05-21 16:56:26 +020096
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +020097increment_out:
98 enetaddr[5]++;
99
100out:
101 return eth_env_set_enetaddr("eth1addr", enetaddr);
Marek Vasutf670cd72022-05-21 16:56:26 +0200102}
103
Christoph Niedermaier276458b2024-12-07 00:04:18 +0100104int dh_setup_mac_address(struct eeprom_id_page *eip)
Marek Vasutf670cd72022-05-21 16:56:26 +0200105{
Marek Vasutf670cd72022-05-21 16:56:26 +0200106 int ret;
107
Christoph Niedermaier276458b2024-12-07 00:04:18 +0100108 ret = dh_imx8_setup_ethaddr(eip);
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +0200109 if (ret)
110 printf("%s: Unable to setup ethaddr! ret = %d\n", __func__, ret);
Marek Vasutf670cd72022-05-21 16:56:26 +0200111
Christoph Niedermaier276458b2024-12-07 00:04:18 +0100112 ret = dh_imx8_setup_eth1addr(eip);
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +0200113 if (ret)
114 printf("%s: Unable to setup eth1addr! ret = %d\n", __func__, ret);
Marek Vasutf670cd72022-05-21 16:56:26 +0200115
Philip Oberfichtnerd24f1de2022-07-26 15:04:52 +0200116 return ret;
Marek Vasutf670cd72022-05-21 16:56:26 +0200117}
118
Christoph Niedermaier276458b2024-12-07 00:04:18 +0100119void dh_add_item_number_and_serial_to_env(struct eeprom_id_page *eip)
120{
121 char *item_number_env;
122 char item_number[8]; /* String with 7 characters + string termination */
123 char *serial_env;
124 char serial[10]; /* String with 9 characters + string termination */
125 int ret;
126
127 ret = dh_get_value_from_eeprom_buffer(DH_ITEM_NUMBER, item_number, sizeof(item_number),
128 eip);
129 if (ret) {
130 printf("%s: Unable to get DHSOM item number from EEPROM ID page! ret = %d\n",
131 __func__, ret);
132 } else {
133 item_number_env = env_get("dh_som_item_number");
134 if (!item_number_env)
135 env_set("dh_som_item_number", item_number);
136 else if (strcmp(item_number_env, item_number))
137 printf("Warning: Environment dh_som_item_number differs from EEPROM ID page value (%s != %s)\n",
138 item_number_env, item_number);
139 }
140
141 ret = dh_get_value_from_eeprom_buffer(DH_SERIAL_NUMBER, serial, sizeof(serial),
142 eip);
143 if (ret) {
144 printf("%s: Unable to get DHSOM serial number from EEPROM ID page! ret = %d\n",
145 __func__, ret);
146 } else {
147 serial_env = env_get("dh_som_serial_number");
148 if (!serial_env)
149 env_set("dh_som_serial_number", serial);
150 else if (strcmp(serial_env, serial))
151 printf("Warning: Environment dh_som_serial_number differs from EEPROM ID page value (%s != %s)\n",
152 serial_env, serial);
153 }
154}
155
Marek Vasutf670cd72022-05-21 16:56:26 +0200156int board_init(void)
157{
Marek Vasutf670cd72022-05-21 16:56:26 +0200158 return 0;
159}
160
161int board_late_init(void)
162{
Christoph Niedermaier276458b2024-12-07 00:04:18 +0100163 u8 eeprom_buffer[DH_EEPROM_ID_PAGE_MAX_SIZE] = { 0 };
164 struct eeprom_id_page *eip = (struct eeprom_id_page *)eeprom_buffer;
165 int ret;
166
167 ret = dh_read_eeprom_id_page(eeprom_buffer, "eeprom0wl");
168 if (ret) {
169 /*
170 * The EEPROM ID page is available on SoM rev. 200 and greater.
171 * For SoM rev. 100 the return value will be -ENODEV. Suppress
172 * the error message for that, because the absence cannot be
173 * treated as an error.
174 */
175 if (ret != -ENODEV)
176 printf("%s: Cannot read valid data from EEPROM ID page! ret = %d\n",
177 __func__, ret);
178 dh_setup_mac_address(NULL);
179 } else {
180 dh_setup_mac_address(eip);
181 dh_add_item_number_and_serial_to_env(eip);
182 }
183
Marek Vasutf670cd72022-05-21 16:56:26 +0200184 return 0;
185}
186
187enum env_location env_get_location(enum env_operation op, int prio)
188{
189 return prio ? ENVL_UNKNOWN : ENVL_SPI_FLASH;
190}