blob: c45ddb14aa336c0771ae7f9f70e05f8b8ae422a8 [file] [log] [blame]
Stephen Warren0e012c32012-08-05 16:07:22 +00001/*
Stephen Warrend6f47c82016-03-24 22:15:20 -06002 * (C) Copyright 2012-2016 Stephen Warren
Stephen Warren0e012c32012-08-05 16:07:22 +00003 *
Stephen Warren29e494c2015-02-16 12:16:13 -07004 * SPDX-License-Identifier: GPL-2.0
Stephen Warren0e012c32012-08-05 16:07:22 +00005 */
6
7#include <common.h>
Lubomir Rintel7d33bb62016-02-22 22:06:47 +01008#include <inttypes.h>
Stephen Warrenaa44d532013-05-27 18:31:18 +00009#include <config.h>
Simon Glass74807622014-09-22 17:30:56 -060010#include <dm.h>
Jeroen Hofsteed5b2fed2014-07-13 22:01:51 +020011#include <fdt_support.h>
Nikita Kiryanovb071e4c2015-02-03 13:32:31 +020012#include <fdt_simplefb.h>
Stephen Warrenaa44d532013-05-27 18:31:18 +000013#include <lcd.h>
Simon Glass2dd337a2015-09-02 17:24:58 -060014#include <memalign.h>
Jeroen Hofsteed5b2fed2014-07-13 22:01:51 +020015#include <mmc.h>
Simon Glass74807622014-09-22 17:30:56 -060016#include <asm/gpio.h>
Stephen Warrenacc8bf42013-01-29 16:37:37 +000017#include <asm/arch/mbox.h>
Stephen Warrenc4ab9712013-01-29 16:37:42 +000018#include <asm/arch/sdhci.h>
Stephen Warren0e012c32012-08-05 16:07:22 +000019#include <asm/global_data.h>
Simon Glassf52cada2014-11-24 21:36:34 -070020#include <dm/platform_data/serial_pl01x.h>
Stephen Warrend6f47c82016-03-24 22:15:20 -060021#include <dm/platform_data/serial_bcm283x_mu.h>
Stephen Warren93ea5262016-04-01 21:14:15 -060022#ifdef CONFIG_ARM64
23#include <asm/armv8/mmu.h>
24#endif
Stephen Warren0e012c32012-08-05 16:07:22 +000025
26DECLARE_GLOBAL_DATA_PTR;
27
Simon Glass74807622014-09-22 17:30:56 -060028static const struct bcm2835_gpio_platdata gpio_platdata = {
29 .base = BCM2835_GPIO_BASE,
30};
31
32U_BOOT_DEVICE(bcm2835_gpios) = {
33 .name = "gpio_bcm2835",
34 .platdata = &gpio_platdata,
35};
36
Stephen Warrend6f47c82016-03-24 22:15:20 -060037#ifdef CONFIG_PL01X_SERIAL
Simon Glassf52cada2014-11-24 21:36:34 -070038static const struct pl01x_serial_platdata serial_platdata = {
Stephen Warren9d5990b2016-03-16 21:40:56 -060039#ifndef CONFIG_BCM2835
Stephen Warrendc7ea682015-02-16 12:16:15 -070040 .base = 0x3f201000,
41#else
Simon Glassf52cada2014-11-24 21:36:34 -070042 .base = 0x20201000,
Stephen Warrendc7ea682015-02-16 12:16:15 -070043#endif
Simon Glassf52cada2014-11-24 21:36:34 -070044 .type = TYPE_PL011,
Eric Anholtbe5a7dd2016-03-13 18:16:54 -070045 .skip_init = true,
Simon Glassf52cada2014-11-24 21:36:34 -070046};
47
48U_BOOT_DEVICE(bcm2835_serials) = {
49 .name = "serial_pl01x",
50 .platdata = &serial_platdata,
51};
Stephen Warrend6f47c82016-03-24 22:15:20 -060052#else
53static const struct bcm283x_mu_serial_platdata serial_platdata = {
54 .base = 0x3f215040,
55 .clock = 250000000,
56 .skip_init = true,
57};
58
59U_BOOT_DEVICE(bcm2837_serials) = {
60 .name = "serial_bcm283x_mu",
61 .platdata = &serial_platdata,
62};
63#endif
Simon Glassf52cada2014-11-24 21:36:34 -070064
Stephen Warrenacc8bf42013-01-29 16:37:37 +000065struct msg_get_arm_mem {
66 struct bcm2835_mbox_hdr hdr;
67 struct bcm2835_mbox_tag_get_arm_mem get_arm_mem;
68 u32 end_tag;
69};
70
Stephen Warrencd210c12014-11-18 21:40:21 -070071struct msg_get_board_rev {
72 struct bcm2835_mbox_hdr hdr;
73 struct bcm2835_mbox_tag_get_board_rev get_board_rev;
74 u32 end_tag;
75};
76
Lubomir Rintel7d33bb62016-02-22 22:06:47 +010077struct msg_get_board_serial {
78 struct bcm2835_mbox_hdr hdr;
79 struct bcm2835_mbox_tag_get_board_serial get_board_serial;
80 u32 end_tag;
81};
82
Stephen Warrenaf6e20d2014-09-26 20:51:39 -060083struct msg_get_mac_address {
84 struct bcm2835_mbox_hdr hdr;
85 struct bcm2835_mbox_tag_get_mac_address get_mac_address;
86 u32 end_tag;
87};
88
Stephen Warren8672d202014-01-13 19:50:11 -070089struct msg_set_power_state {
90 struct bcm2835_mbox_hdr hdr;
91 struct bcm2835_mbox_tag_set_power_state set_power_state;
92 u32 end_tag;
93};
94
Stephen Warrenc4ab9712013-01-29 16:37:42 +000095struct msg_get_clock_rate {
96 struct bcm2835_mbox_hdr hdr;
97 struct bcm2835_mbox_tag_get_clock_rate get_clock_rate;
98 u32 end_tag;
99};
100
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700101/*
102 * http://raspberryalphaomega.org.uk/2013/02/06/automatic-raspberry-pi-board-revision-detection-model-a-b1-and-b2/
103 * http://www.raspberrypi.org/forums/viewtopic.php?f=63&t=32733
Stephen Warren26683852015-12-04 22:07:45 -0700104 * http://git.drogon.net/?p=wiringPi;a=blob;f=wiringPi/wiringPi.c;h=503151f61014418b9c42f4476a6086f75cd4e64b;hb=refs/heads/master#l922
Stephen Warrena33dc8b2016-01-28 22:24:44 -0700105 *
106 * In http://lists.denx.de/pipermail/u-boot/2016-January/243752.html
107 * ("[U-Boot] [PATCH] rpi: fix up Model B entries") Dom Cobley at the RPi
108 * Foundation stated that the following source was accurate:
109 * https://github.com/AndrewFromMelbourne/raspberry_pi_revision
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700110 */
Stephen Warren26683852015-12-04 22:07:45 -0700111struct rpi_model {
Stephen Warrencd210c12014-11-18 21:40:21 -0700112 const char *name;
113 const char *fdtfile;
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700114 bool has_onboard_eth;
Stephen Warren26683852015-12-04 22:07:45 -0700115};
116
117static const struct rpi_model rpi_model_unknown = {
118 "Unknown model",
Stephen Warren93970ce2016-03-24 22:15:17 -0600119 "bcm283x-rpi-other.dtb",
Stephen Warren26683852015-12-04 22:07:45 -0700120 false,
121};
122
123static const struct rpi_model rpi_models_new_scheme[] = {
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700124 [0x4] = {
Stephen Warrendc7ea682015-02-16 12:16:15 -0700125 "2 Model B",
126 "bcm2836-rpi-2-b.dtb",
127 true,
128 },
Stephen Warren303244a2016-03-24 22:15:18 -0600129 [0x8] = {
130 "3 Model B",
131 "bcm2837-rpi-3-b.dtb",
132 true,
133 },
Stephen Warren2f1f20b2015-12-04 22:07:46 -0700134 [0x9] = {
135 "Zero",
136 "bcm2835-rpi-zero.dtb",
137 false,
138 },
Stephen Warren26683852015-12-04 22:07:45 -0700139};
140
141static const struct rpi_model rpi_models_old_scheme[] = {
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700142 [0x2] = {
Lubomir Rintelc44f76e2016-01-29 09:35:52 +0100143 "Model B",
144 "bcm2835-rpi-b.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700145 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700146 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700147 [0x3] = {
Lubomir Rintelc44f76e2016-01-29 09:35:52 +0100148 "Model B",
149 "bcm2835-rpi-b.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700150 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700151 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700152 [0x4] = {
Lubomir Rintelc44f76e2016-01-29 09:35:52 +0100153 "Model B rev2",
154 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700155 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700156 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700157 [0x5] = {
Lubomir Rintelc44f76e2016-01-29 09:35:52 +0100158 "Model B rev2",
159 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700160 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700161 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700162 [0x6] = {
Lubomir Rintelc44f76e2016-01-29 09:35:52 +0100163 "Model B rev2",
164 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700165 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700166 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700167 [0x7] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700168 "Model A",
169 "bcm2835-rpi-a.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700170 false,
Stephen Warrencd210c12014-11-18 21:40:21 -0700171 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700172 [0x8] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700173 "Model A",
174 "bcm2835-rpi-a.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700175 false,
Stephen Warrencd210c12014-11-18 21:40:21 -0700176 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700177 [0x9] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700178 "Model A",
179 "bcm2835-rpi-a.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700180 false,
Stephen Warrencd210c12014-11-18 21:40:21 -0700181 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700182 [0xd] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700183 "Model B rev2",
184 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700185 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700186 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700187 [0xe] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700188 "Model B rev2",
189 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700190 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700191 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700192 [0xf] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700193 "Model B rev2",
194 "bcm2835-rpi-b-rev2.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700195 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700196 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700197 [0x10] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700198 "Model B+",
199 "bcm2835-rpi-b-plus.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700200 true,
Stephen Warrencd210c12014-11-18 21:40:21 -0700201 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700202 [0x11] = {
Stephen Warrencd210c12014-11-18 21:40:21 -0700203 "Compute Module",
204 "bcm2835-rpi-cm.dtb",
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700205 false,
Stephen Warrencd210c12014-11-18 21:40:21 -0700206 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700207 [0x12] = {
Stephen Warren18a5fc62014-12-23 20:01:43 -0700208 "Model A+",
209 "bcm2835-rpi-a-plus.dtb",
210 false,
211 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700212 [0x13] = {
Stephen Warren3648a852015-04-12 21:43:25 -0600213 "Model B+",
214 "bcm2835-rpi-b-plus.dtb",
215 true,
216 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700217 [0x14] = {
Stephen Warren3648a852015-04-12 21:43:25 -0600218 "Compute Module",
219 "bcm2835-rpi-cm.dtb",
220 false,
221 },
Stephen Warrenbe8efec2015-12-04 22:07:44 -0700222 [0x15] = {
Lubomir Rintelc87458c2015-10-14 17:17:54 +0200223 "Model A+",
224 "bcm2835-rpi-a-plus.dtb",
225 false,
226 },
Stephen Warrencd210c12014-11-18 21:40:21 -0700227};
228
Stephen Warren26683852015-12-04 22:07:45 -0700229static uint32_t revision;
230static uint32_t rev_scheme;
231static uint32_t rev_type;
232static const struct rpi_model *model;
Stephen Warrencd210c12014-11-18 21:40:21 -0700233
Stephen Warren93ea5262016-04-01 21:14:15 -0600234#ifdef CONFIG_ARM64
235static struct mm_region bcm2837_mem_map[] = {
236 {
237 .base = 0x00000000UL,
238 .size = 0x3f000000UL,
239 .attrs = PTE_BLOCK_MEMTYPE(MT_NORMAL) |
240 PTE_BLOCK_INNER_SHARE
241 }, {
242 .base = 0x3f000000UL,
243 .size = 0x01000000UL,
244 .attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
245 PTE_BLOCK_NON_SHARE |
246 PTE_BLOCK_PXN | PTE_BLOCK_UXN
247 }, {
248 /* List terminator */
249 0,
250 }
251};
252
253struct mm_region *mem_map = bcm2837_mem_map;
254#endif
255
Stephen Warren0e012c32012-08-05 16:07:22 +0000256int dram_init(void)
257{
Alexander Steinc56c9472015-07-24 09:22:12 +0200258 ALLOC_CACHE_ALIGN_BUFFER(struct msg_get_arm_mem, msg, 1);
Stephen Warrenacc8bf42013-01-29 16:37:37 +0000259 int ret;
260
261 BCM2835_MBOX_INIT_HDR(msg);
262 BCM2835_MBOX_INIT_TAG(&msg->get_arm_mem, GET_ARM_MEMORY);
263
264 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN, &msg->hdr);
265 if (ret) {
266 printf("bcm2835: Could not query ARM memory size\n");
267 return -1;
268 }
269
270 gd->ram_size = msg->get_arm_mem.body.resp.mem_size;
Stephen Warren0e012c32012-08-05 16:07:22 +0000271
272 return 0;
273}
274
Stephen Warrencd210c12014-11-18 21:40:21 -0700275static void set_fdtfile(void)
276{
277 const char *fdtfile;
278
279 if (getenv("fdtfile"))
280 return;
281
Stephen Warren26683852015-12-04 22:07:45 -0700282 fdtfile = model->fdtfile;
Stephen Warrencd210c12014-11-18 21:40:21 -0700283 setenv("fdtfile", fdtfile);
284}
285
286static void set_usbethaddr(void)
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600287{
Alexander Steinc56c9472015-07-24 09:22:12 +0200288 ALLOC_CACHE_ALIGN_BUFFER(struct msg_get_mac_address, msg, 1);
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600289 int ret;
290
Stephen Warren26683852015-12-04 22:07:45 -0700291 if (!model->has_onboard_eth)
Stephen Warrenb2fa38362014-12-05 20:56:46 -0700292 return;
293
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600294 if (getenv("usbethaddr"))
Stephen Warrencd210c12014-11-18 21:40:21 -0700295 return;
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600296
297 BCM2835_MBOX_INIT_HDR(msg);
298 BCM2835_MBOX_INIT_TAG(&msg->get_mac_address, GET_MAC_ADDRESS);
299
300 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN, &msg->hdr);
301 if (ret) {
302 printf("bcm2835: Could not query MAC address\n");
303 /* Ignore error; not critical */
Stephen Warrencd210c12014-11-18 21:40:21 -0700304 return;
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600305 }
306
307 eth_setenv_enetaddr("usbethaddr", msg->get_mac_address.body.resp.mac);
308
Lubomir Rintel3ac09f82016-02-03 16:08:09 +0100309 if (!getenv("ethaddr"))
310 setenv("ethaddr", getenv("usbethaddr"));
311
Stephen Warrencd210c12014-11-18 21:40:21 -0700312 return;
313}
314
Guillaume GARDET25fa8602015-08-25 15:10:26 +0200315#ifdef CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG
316static void set_board_info(void)
317{
Stephen Warren26683852015-12-04 22:07:45 -0700318 char s[11];
319
320 snprintf(s, sizeof(s), "0x%X", revision);
321 setenv("board_revision", s);
322 snprintf(s, sizeof(s), "%d", rev_scheme);
323 setenv("board_rev_scheme", s);
324 /* Can't rename this to board_rev_type since it's an ABI for scripts */
325 snprintf(s, sizeof(s), "0x%X", rev_type);
326 setenv("board_rev", s);
327 setenv("board_name", model->name);
Guillaume GARDET25fa8602015-08-25 15:10:26 +0200328}
329#endif /* CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG */
330
Lubomir Rintel7d33bb62016-02-22 22:06:47 +0100331static void set_serial_number(void)
332{
333 ALLOC_CACHE_ALIGN_BUFFER(struct msg_get_board_serial, msg, 1);
334 int ret;
335 char serial_string[17] = { 0 };
336
337 if (getenv("serial#"))
338 return;
339
340 BCM2835_MBOX_INIT_HDR(msg);
341 BCM2835_MBOX_INIT_TAG_NO_REQ(&msg->get_board_serial, GET_BOARD_SERIAL);
342
343 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN, &msg->hdr);
344 if (ret) {
345 printf("bcm2835: Could not query board serial\n");
346 /* Ignore error; not critical */
347 return;
348 }
349
350 snprintf(serial_string, sizeof(serial_string), "%016" PRIx64,
351 msg->get_board_serial.body.resp.serial);
352 setenv("serial#", serial_string);
353}
354
Stephen Warrencd210c12014-11-18 21:40:21 -0700355int misc_init_r(void)
356{
357 set_fdtfile();
358 set_usbethaddr();
Guillaume GARDET25fa8602015-08-25 15:10:26 +0200359#ifdef CONFIG_ENV_VARS_UBOOT_RUNTIME_CONFIG
360 set_board_info();
361#endif
Lubomir Rintel7d33bb62016-02-22 22:06:47 +0100362 set_serial_number();
363
Stephen Warrenaf6e20d2014-09-26 20:51:39 -0600364 return 0;
365}
366
Stephen Warren8672d202014-01-13 19:50:11 -0700367static int power_on_module(u32 module)
368{
Alexander Steinc56c9472015-07-24 09:22:12 +0200369 ALLOC_CACHE_ALIGN_BUFFER(struct msg_set_power_state, msg_pwr, 1);
Stephen Warren8672d202014-01-13 19:50:11 -0700370 int ret;
371
372 BCM2835_MBOX_INIT_HDR(msg_pwr);
373 BCM2835_MBOX_INIT_TAG(&msg_pwr->set_power_state,
374 SET_POWER_STATE);
375 msg_pwr->set_power_state.body.req.device_id = module;
376 msg_pwr->set_power_state.body.req.state =
377 BCM2835_MBOX_SET_POWER_STATE_REQ_ON |
378 BCM2835_MBOX_SET_POWER_STATE_REQ_WAIT;
379
380 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN,
381 &msg_pwr->hdr);
382 if (ret) {
383 printf("bcm2835: Could not set module %u power state\n",
384 module);
385 return -1;
386 }
387
388 return 0;
389}
390
Stephen Warrencd210c12014-11-18 21:40:21 -0700391static void get_board_rev(void)
392{
Alexander Steinc56c9472015-07-24 09:22:12 +0200393 ALLOC_CACHE_ALIGN_BUFFER(struct msg_get_board_rev, msg, 1);
Stephen Warrencd210c12014-11-18 21:40:21 -0700394 int ret;
Stephen Warren26683852015-12-04 22:07:45 -0700395 const struct rpi_model *models;
396 uint32_t models_count;
Stephen Warrencd210c12014-11-18 21:40:21 -0700397
398 BCM2835_MBOX_INIT_HDR(msg);
399 BCM2835_MBOX_INIT_TAG(&msg->get_board_rev, GET_BOARD_REV);
400
401 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN, &msg->hdr);
402 if (ret) {
403 printf("bcm2835: Could not query board revision\n");
404 /* Ignore error; not critical */
405 return;
406 }
407
Stephen Warrendc7ea682015-02-16 12:16:15 -0700408 /*
409 * For details of old-vs-new scheme, see:
410 * https://github.com/pimoroni/RPi.version/blob/master/RPi/version.py
411 * http://www.raspberrypi.org/forums/viewtopic.php?f=63&t=99293&p=690282
412 * (a few posts down)
Stephen Warrenb834af82015-03-23 23:00:25 -0600413 *
414 * For the RPi 1, bit 24 is the "warranty bit", so we mask off just the
415 * lower byte to use as the board rev:
416 * http://www.raspberrypi.org/forums/viewtopic.php?f=63&t=98367&start=250
417 * http://www.raspberrypi.org/forums/viewtopic.php?f=31&t=20594
Stephen Warrendc7ea682015-02-16 12:16:15 -0700418 */
Stephen Warren26683852015-12-04 22:07:45 -0700419 revision = msg->get_board_rev.body.resp.rev;
420 if (revision & 0x800000) {
421 rev_scheme = 1;
422 rev_type = (revision >> 4) & 0xff;
423 models = rpi_models_new_scheme;
424 models_count = ARRAY_SIZE(rpi_models_new_scheme);
425 } else {
426 rev_scheme = 0;
427 rev_type = revision & 0xff;
428 models = rpi_models_old_scheme;
429 models_count = ARRAY_SIZE(rpi_models_old_scheme);
Stephen Warren18a5fc62014-12-23 20:01:43 -0700430 }
Stephen Warren26683852015-12-04 22:07:45 -0700431 if (rev_type >= models_count) {
432 printf("RPI: Board rev 0x%x outside known range\n", rev_type);
433 model = &rpi_model_unknown;
434 } else if (!models[rev_type].name) {
435 printf("RPI: Board rev 0x%x unknown\n", rev_type);
436 model = &rpi_model_unknown;
437 } else {
438 model = &models[rev_type];
Stephen Warren18a5fc62014-12-23 20:01:43 -0700439 }
Stephen Warrenc61dfd12014-12-23 20:01:44 -0700440
Stephen Warren26683852015-12-04 22:07:45 -0700441 printf("RPI %s (0x%x)\n", model->name, revision);
Stephen Warrencd210c12014-11-18 21:40:21 -0700442}
443
Stephen Warren0e012c32012-08-05 16:07:22 +0000444int board_init(void)
445{
Stephen Warrencd210c12014-11-18 21:40:21 -0700446 get_board_rev();
447
Stephen Warren0e012c32012-08-05 16:07:22 +0000448 gd->bd->bi_boot_params = 0x100;
449
Stephen Warren8672d202014-01-13 19:50:11 -0700450 return power_on_module(BCM2835_MBOX_POWER_DEVID_USB_HCD);
Stephen Warren0e012c32012-08-05 16:07:22 +0000451}
Stephen Warrenc4ab9712013-01-29 16:37:42 +0000452
Jeroen Hofsteed5b2fed2014-07-13 22:01:51 +0200453int board_mmc_init(bd_t *bis)
Stephen Warrenc4ab9712013-01-29 16:37:42 +0000454{
Alexander Steinc56c9472015-07-24 09:22:12 +0200455 ALLOC_CACHE_ALIGN_BUFFER(struct msg_get_clock_rate, msg_clk, 1);
Stephen Warrenc4ab9712013-01-29 16:37:42 +0000456 int ret;
457
Stephen Warren8672d202014-01-13 19:50:11 -0700458 power_on_module(BCM2835_MBOX_POWER_DEVID_SDHCI);
459
Stephen Warrenc4ab9712013-01-29 16:37:42 +0000460 BCM2835_MBOX_INIT_HDR(msg_clk);
461 BCM2835_MBOX_INIT_TAG(&msg_clk->get_clock_rate, GET_CLOCK_RATE);
462 msg_clk->get_clock_rate.body.req.clock_id = BCM2835_MBOX_CLOCK_ID_EMMC;
463
464 ret = bcm2835_mbox_call_prop(BCM2835_MBOX_PROP_CHAN, &msg_clk->hdr);
465 if (ret) {
466 printf("bcm2835: Could not query eMMC clock rate\n");
467 return -1;
468 }
469
470 return bcm2835_sdhci_init(BCM2835_SDHCI_BASE,
471 msg_clk->get_clock_rate.body.resp.rate_hz);
472}
Stephen Warrenaa44d532013-05-27 18:31:18 +0000473
Simon Glass2aec3cc2014-10-23 18:58:47 -0600474int ft_board_setup(void *blob, bd_t *bd)
Stephen Warrenaa44d532013-05-27 18:31:18 +0000475{
476 /*
477 * For now, we simply always add the simplefb DT node. Later, we
478 * should be more intelligent, and e.g. only do this if no enabled DT
479 * node exists for the "real" graphics driver.
480 */
481 lcd_dt_simplefb_add_node(blob);
Simon Glass2aec3cc2014-10-23 18:58:47 -0600482
483 return 0;
Stephen Warrenaa44d532013-05-27 18:31:18 +0000484}