blob: a5bf355ce1d632f032b855ff4089365d1d5b3ee1 [file] [log] [blame]
Thomas Gleixner2874c5f2019-05-27 08:55:01 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002/*
3 * Procedures for interfacing to Open Firmware.
4 *
5 * Paul Mackerras August 1996.
6 * Copyright (C) 1996-2005 Paul Mackerras.
7 *
8 * Adapted for 64bit PowerPC by Dave Engebretsen and Peter Bergner.
9 * {engebret|bergner}@us.ibm.com
Paul Mackerras9b6b5632005-10-06 12:06:20 +100010 */
11
12#undef DEBUG_PROM
13
Daniel Axtens054f3672017-07-12 14:36:04 -070014/* we cannot use FORTIFY as it brings in new symbols */
15#define __NO_FORTIFY
16
Paul Mackerras9b6b5632005-10-06 12:06:20 +100017#include <stdarg.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100018#include <linux/kernel.h>
19#include <linux/string.h>
20#include <linux/init.h>
21#include <linux/threads.h>
22#include <linux/spinlock.h>
23#include <linux/types.h>
24#include <linux/pci.h>
25#include <linux/proc_fs.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100026#include <linux/delay.h>
27#include <linux/initrd.h>
28#include <linux/bitops.h>
Mike Rapoport65fddcf2020-06-08 21:32:42 -070029#include <linux/pgtable.h>
Michael Ellermanffaacd92021-06-21 16:49:38 +100030#include <linux/printk.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100031#include <asm/prom.h>
32#include <asm/rtas.h>
33#include <asm/page.h>
34#include <asm/processor.h>
Nicholas Piggin59dc5bf2021-06-18 01:51:03 +100035#include <asm/interrupt.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100036#include <asm/irq.h>
37#include <asm/io.h>
38#include <asm/smp.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100039#include <asm/mmu.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100040#include <asm/iommu.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100041#include <asm/btext.h>
42#include <asm/sections.h>
43#include <asm/machdep.h>
Daniel Axtens0545d542016-09-06 15:32:43 +100044#include <asm/asm-prototypes.h>
Ram Pai6a9c9302019-08-19 23:13:14 -030045#include <asm/ultravisor-api.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100046
Paul Mackerras9b6b5632005-10-06 12:06:20 +100047#include <linux/linux_logo.h>
Paul Mackerras9b6b5632005-10-06 12:06:20 +100048
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +110049/* All of prom_init bss lives here */
Joe Perches33def842020-10-21 19:36:07 -070050#define __prombss __section(".bss.prominit")
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +110051
Paul Mackerras9b6b5632005-10-06 12:06:20 +100052/*
Paul Mackerras9b6b5632005-10-06 12:06:20 +100053 * Eventually bump that one up
54 */
55#define DEVTREE_CHUNK_SIZE 0x100000
56
57/*
58 * This is the size of the local memory reserve map that gets copied
59 * into the boot params passed to the kernel. That size is totally
60 * flexible as the kernel just reads the list until it encounters an
61 * entry with size 0, so it can be changed without breaking binary
62 * compatibility
63 */
64#define MEM_RESERVE_MAP_SIZE 8
65
66/*
67 * prom_init() is called very early on, before the kernel text
68 * and data have been mapped to KERNELBASE. At this point the code
69 * is running at whatever address it has been loaded at.
70 * On ppc32 we compile with -mrelocatable, which means that references
71 * to extern and static variables get relocated automatically.
Anton Blanchard5ac47f72012-11-26 17:39:03 +000072 * ppc64 objects are always relocatable, we just need to relocate the
73 * TOC.
Paul Mackerras9b6b5632005-10-06 12:06:20 +100074 *
75 * Because OF may have mapped I/O devices into the area starting at
76 * KERNELBASE, particularly on CHRP machines, we can't safely call
77 * OF once the kernel has been mapped to KERNELBASE. Therefore all
78 * OF calls must be done within prom_init().
79 *
80 * ADDR is used in calls to call_prom. The 4th and following
81 * arguments to call_prom should be 32-bit values.
82 * On ppc64, 64 bit values are truncated to 32 bits (and
83 * fortunately don't get interpreted as two arguments).
84 */
Anton Blanchard5ac47f72012-11-26 17:39:03 +000085#define ADDR(x) (u32)(unsigned long)(x)
86
Paul Mackerras9b6b5632005-10-06 12:06:20 +100087#ifdef CONFIG_PPC64
Paul Mackerrasa23414b2005-11-10 12:00:55 +110088#define OF_WORKAROUNDS 0
Paul Mackerras9b6b5632005-10-06 12:06:20 +100089#else
Paul Mackerrasa23414b2005-11-10 12:00:55 +110090#define OF_WORKAROUNDS of_workarounds
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +110091static int of_workarounds __prombss;
Paul Mackerras9b6b5632005-10-06 12:06:20 +100092#endif
93
Paul Mackerrasa23414b2005-11-10 12:00:55 +110094#define OF_WA_CLAIM 1 /* do phys/virt claim separately, then map */
95#define OF_WA_LONGTRAIL 2 /* work around longtrail bugs */
96
Paul Mackerras9b6b5632005-10-06 12:06:20 +100097#define PROM_BUG() do { \
98 prom_printf("kernel BUG at %s line 0x%x!\n", \
Anton Blanchard5827d412012-11-26 17:40:03 +000099 __FILE__, __LINE__); \
Christophe Leroy63ce2712019-08-26 11:10:23 +0000100 __builtin_trap(); \
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000101} while (0)
102
103#ifdef DEBUG_PROM
104#define prom_debug(x...) prom_printf(x)
105#else
Mathieu Malaterre85aa4b92018-04-23 21:36:38 +0200106#define prom_debug(x...) do { } while (0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000107#endif
108
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000109
110typedef u32 prom_arg_t;
111
112struct prom_args {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000113 __be32 service;
114 __be32 nargs;
115 __be32 nret;
116 __be32 args[10];
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000117};
118
119struct prom_t {
120 ihandle root;
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100121 phandle chosen;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000122 int cpu;
123 ihandle stdout;
Paul Mackerrasa575b802005-10-23 17:23:21 +1000124 ihandle mmumap;
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100125 ihandle memory;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000126};
127
128struct mem_map_entry {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000129 __be64 base;
130 __be64 size;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000131};
132
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000133typedef __be32 cell_t;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000134
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +0000135extern void __start(unsigned long r3, unsigned long r4, unsigned long r5,
136 unsigned long r6, unsigned long r7, unsigned long r8,
137 unsigned long r9);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000138
139#ifdef CONFIG_PPC64
Paul Mackerrasc49888202005-10-26 21:52:53 +1000140extern int enter_prom(struct prom_args *args, unsigned long entry);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000141#else
Paul Mackerrasc49888202005-10-26 21:52:53 +1000142static inline int enter_prom(struct prom_args *args, unsigned long entry)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000143{
Paul Mackerrasc49888202005-10-26 21:52:53 +1000144 return ((int (*)(struct prom_args *))entry)(args);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000145}
146#endif
147
148extern void copy_and_flush(unsigned long dest, unsigned long src,
149 unsigned long size, unsigned long offset);
150
151/* prom structure */
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100152static struct prom_t __prombss prom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000153
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100154static unsigned long __prombss prom_entry;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000155
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100156static char __prombss of_stdout_device[256];
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +0000157static char __prombss prom_scratch[256];
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000158
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100159static unsigned long __prombss dt_header_start;
160static unsigned long __prombss dt_struct_start, dt_struct_end;
161static unsigned long __prombss dt_string_start, dt_string_end;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000162
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100163static unsigned long __prombss prom_initrd_start, prom_initrd_end;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000164
165#ifdef CONFIG_PPC64
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100166static int __prombss prom_iommu_force_on;
167static int __prombss prom_iommu_off;
168static unsigned long __prombss prom_tce_alloc_start;
169static unsigned long __prombss prom_tce_alloc_end;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000170#endif
171
Benjamin Herrenschmidtc8860872018-10-15 13:49:54 +1100172#ifdef CONFIG_PPC_PSERIES
Benjamin Herrenschmidt8ca2d512018-10-15 13:49:57 +1100173static bool __prombss prom_radix_disable;
Aneesh Kumar K.Vbf6b7662020-07-27 14:29:08 +0530174static bool __prombss prom_radix_gtse_disable;
Greg Kurza3bf9fb2019-05-15 12:05:01 +0200175static bool __prombss prom_xive_disable;
Benjamin Herrenschmidtc8860872018-10-15 13:49:54 +1100176#endif
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100177
Ram Pai6a9c9302019-08-19 23:13:14 -0300178#ifdef CONFIG_PPC_SVM
179static bool __prombss prom_svm_enable;
180#endif
181
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100182struct platform_support {
183 bool hash_mmu;
184 bool radix_mmu;
185 bool radix_gtse;
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +0200186 bool xive;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100187};
188
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +1100189/* Platforms codes are now obsolete in the kernel. Now only used within this
190 * file and ultimately gone too. Feel free to change them if you need, they
191 * are not shared with anything outside of this file anymore
192 */
193#define PLATFORM_PSERIES 0x0100
194#define PLATFORM_PSERIES_LPAR 0x0101
195#define PLATFORM_LPAR 0x0001
196#define PLATFORM_POWERMAC 0x0400
197#define PLATFORM_GENERIC 0x0500
198
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100199static int __prombss of_platform;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000200
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100201static char __prombss prom_cmd_line[COMMAND_LINE_SIZE];
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000202
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100203static unsigned long __prombss prom_memory_limit;
Benjamin Krillcf687872009-07-27 22:02:39 +0000204
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100205static unsigned long __prombss alloc_top;
206static unsigned long __prombss alloc_top_high;
207static unsigned long __prombss alloc_bottom;
208static unsigned long __prombss rmo_top;
209static unsigned long __prombss ram_top;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000210
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100211static struct mem_map_entry __prombss mem_reserve_map[MEM_RESERVE_MAP_SIZE];
212static int __prombss mem_reserve_cnt;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000213
Benjamin Herrenschmidte63334e2018-10-15 13:49:52 +1100214static cell_t __prombss regbuf[1024];
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000215
Benjamin Herrenschmidt8ca2d512018-10-15 13:49:57 +1100216static bool __prombss rtas_has_query_cpu_stopped;
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +1000217
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000218
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000219/*
220 * Error results ... some OF calls will return "-1" on error, some
221 * will return 0, some will return either. To simplify, here are
222 * macros to use with any ihandle or phandle return value to check if
223 * it is valid
224 */
225
226#define PROM_ERROR (-1u)
227#define PHANDLE_VALID(p) ((p) != 0 && (p) != PROM_ERROR)
228#define IHANDLE_VALID(i) ((i) != 0 && (i) != PROM_ERROR)
229
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000230/* Copied from lib/string.c and lib/kstrtox.c */
231
232static int __init prom_strcmp(const char *cs, const char *ct)
233{
234 unsigned char c1, c2;
235
236 while (1) {
237 c1 = *cs++;
238 c2 = *ct++;
239 if (c1 != c2)
240 return c1 < c2 ? -1 : 1;
241 if (!c1)
242 break;
243 }
244 return 0;
245}
246
Michael Ellermanf47d5a42021-06-21 16:49:37 +1000247static ssize_t __init prom_strscpy_pad(char *dest, const char *src, size_t n)
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000248{
Michael Ellermanf47d5a42021-06-21 16:49:37 +1000249 ssize_t rc;
250 size_t i;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000251
Michael Ellermanf47d5a42021-06-21 16:49:37 +1000252 if (n == 0 || n > INT_MAX)
253 return -E2BIG;
254
255 // Copy up to n bytes
256 for (i = 0; i < n && src[i] != '\0'; i++)
257 dest[i] = src[i];
258
259 rc = i;
260
261 // If we copied all n then we have run out of space for the nul
262 if (rc == n) {
263 // Rewind by one character to ensure nul termination
264 i--;
265 rc = -E2BIG;
266 }
267
268 for (; i < n; i++)
269 dest[i] = '\0';
270
271 return rc;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000272}
273
274static int __init prom_strncmp(const char *cs, const char *ct, size_t count)
275{
276 unsigned char c1, c2;
277
278 while (count) {
279 c1 = *cs++;
280 c2 = *ct++;
281 if (c1 != c2)
282 return c1 < c2 ? -1 : 1;
283 if (!c1)
284 break;
285 count--;
286 }
287 return 0;
288}
289
290static size_t __init prom_strlen(const char *s)
291{
292 const char *sc;
293
294 for (sc = s; *sc != '\0'; ++sc)
295 /* nothing */;
296 return sc - s;
297}
298
299static int __init prom_memcmp(const void *cs, const void *ct, size_t count)
300{
301 const unsigned char *su1, *su2;
302 int res = 0;
303
304 for (su1 = cs, su2 = ct; 0 < count; ++su1, ++su2, count--)
305 if ((res = *su1 - *su2) != 0)
306 break;
307 return res;
308}
309
310static char __init *prom_strstr(const char *s1, const char *s2)
311{
312 size_t l1, l2;
313
314 l2 = prom_strlen(s2);
315 if (!l2)
316 return (char *)s1;
317 l1 = prom_strlen(s1);
318 while (l1 >= l2) {
319 l1--;
320 if (!prom_memcmp(s1, s2, l2))
321 return (char *)s1;
322 s1++;
323 }
324 return NULL;
325}
326
Chris Packhamd79fbb32019-08-02 10:50:06 +1200327static size_t __init prom_strlcat(char *dest, const char *src, size_t count)
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000328{
Chris Packhamd79fbb32019-08-02 10:50:06 +1200329 size_t dsize = prom_strlen(dest);
330 size_t len = prom_strlen(src);
331 size_t res = dsize + len;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000332
Chris Packhamd79fbb32019-08-02 10:50:06 +1200333 /* This would be a bug */
334 if (dsize >= count)
335 return count;
336
337 dest += dsize;
338 count -= dsize;
339 if (len >= count)
340 len = count-1;
341 memcpy(dest, src, len);
342 dest[len] = 0;
343 return res;
344
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000345}
346
347#ifdef CONFIG_PPC_PSERIES
348static int __init prom_strtobool(const char *s, bool *res)
349{
350 if (!s)
351 return -EINVAL;
352
353 switch (s[0]) {
354 case 'y':
355 case 'Y':
356 case '1':
357 *res = true;
358 return 0;
359 case 'n':
360 case 'N':
361 case '0':
362 *res = false;
363 return 0;
364 case 'o':
365 case 'O':
366 switch (s[1]) {
367 case 'n':
368 case 'N':
369 *res = true;
370 return 0;
371 case 'f':
372 case 'F':
373 *res = false;
374 return 0;
375 default:
376 break;
377 }
Nick Desaulniers49a41362020-11-15 20:35:32 -0800378 break;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000379 default:
380 break;
381 }
382
383 return -EINVAL;
384}
385#endif
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000386
387/* This is the one and *ONLY* place where we actually call open
388 * firmware.
389 */
390
391static int __init call_prom(const char *service, int nargs, int nret, ...)
392{
393 int i;
394 struct prom_args args;
395 va_list list;
396
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000397 args.service = cpu_to_be32(ADDR(service));
398 args.nargs = cpu_to_be32(nargs);
399 args.nret = cpu_to_be32(nret);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000400
401 va_start(list, nret);
402 for (i = 0; i < nargs; i++)
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000403 args.args[i] = cpu_to_be32(va_arg(list, prom_arg_t));
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000404 va_end(list);
405
406 for (i = 0; i < nret; i++)
407 args.args[nargs+i] = 0;
408
Anton Blanchard5827d412012-11-26 17:40:03 +0000409 if (enter_prom(&args, prom_entry) < 0)
Paul Mackerrasc49888202005-10-26 21:52:53 +1000410 return PROM_ERROR;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000411
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000412 return (nret > 0) ? be32_to_cpu(args.args[nargs]) : 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000413}
414
415static int __init call_prom_ret(const char *service, int nargs, int nret,
416 prom_arg_t *rets, ...)
417{
418 int i;
419 struct prom_args args;
420 va_list list;
421
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000422 args.service = cpu_to_be32(ADDR(service));
423 args.nargs = cpu_to_be32(nargs);
424 args.nret = cpu_to_be32(nret);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000425
426 va_start(list, rets);
427 for (i = 0; i < nargs; i++)
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000428 args.args[i] = cpu_to_be32(va_arg(list, prom_arg_t));
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000429 va_end(list);
430
431 for (i = 0; i < nret; i++)
Olaf Heringed1189b72005-11-29 14:04:05 +0100432 args.args[nargs+i] = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000433
Anton Blanchard5827d412012-11-26 17:40:03 +0000434 if (enter_prom(&args, prom_entry) < 0)
Paul Mackerrasc49888202005-10-26 21:52:53 +1000435 return PROM_ERROR;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000436
437 if (rets != NULL)
438 for (i = 1; i < nret; ++i)
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000439 rets[i-1] = be32_to_cpu(args.args[nargs+i]);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000440
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000441 return (nret > 0) ? be32_to_cpu(args.args[nargs]) : 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000442}
443
444
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000445static void __init prom_print(const char *msg)
446{
447 const char *p, *q;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000448
Anton Blanchard5827d412012-11-26 17:40:03 +0000449 if (prom.stdout == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000450 return;
451
452 for (p = msg; *p != 0; p = q) {
453 for (q = p; *q != 0 && *q != '\n'; ++q)
454 ;
455 if (q > p)
Anton Blanchard5827d412012-11-26 17:40:03 +0000456 call_prom("write", 3, 1, prom.stdout, p, q - p);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000457 if (*q == 0)
458 break;
459 ++q;
Anton Blanchard5827d412012-11-26 17:40:03 +0000460 call_prom("write", 3, 1, prom.stdout, ADDR("\r\n"), 2);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000461 }
462}
463
464
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200465/*
466 * Both prom_print_hex & prom_print_dec takes an unsigned long as input so that
467 * we do not need __udivdi3 or __umoddi3 on 32bits.
468 */
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000469static void __init prom_print_hex(unsigned long val)
470{
471 int i, nibbles = sizeof(val)*2;
472 char buf[sizeof(val)*2+1];
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000473
474 for (i = nibbles-1; i >= 0; i--) {
475 buf[i] = (val & 0xf) + '0';
476 if (buf[i] > '9')
477 buf[i] += ('a'-'0'-10);
478 val >>= 4;
479 }
480 buf[nibbles] = '\0';
Anton Blanchard5827d412012-11-26 17:40:03 +0000481 call_prom("write", 3, 1, prom.stdout, buf, nibbles);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000482}
483
Michael Neuling2c48a7d2010-07-27 18:26:21 +0000484/* max number of decimal digits in an unsigned long */
485#define UL_DIGITS 21
486static void __init prom_print_dec(unsigned long val)
487{
488 int i, size;
489 char buf[UL_DIGITS+1];
Michael Neuling2c48a7d2010-07-27 18:26:21 +0000490
491 for (i = UL_DIGITS-1; i >= 0; i--) {
492 buf[i] = (val % 10) + '0';
493 val = val/10;
494 if (val == 0)
495 break;
496 }
497 /* shift stuff down */
498 size = UL_DIGITS - i;
Anton Blanchard5827d412012-11-26 17:40:03 +0000499 call_prom("write", 3, 1, prom.stdout, buf+i, size);
Michael Neuling2c48a7d2010-07-27 18:26:21 +0000500}
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000501
Mathieu Malaterreeae5f702018-04-06 22:12:19 +0200502__printf(1, 2)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000503static void __init prom_printf(const char *format, ...)
504{
505 const char *p, *q, *s;
506 va_list args;
507 unsigned long v;
Benjamin Herrenschmidtaf277142011-04-06 10:51:17 +1000508 long vs;
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200509 int n = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000510
511 va_start(args, format);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000512 for (p = format; *p != 0; p = q) {
513 for (q = p; *q != 0 && *q != '\n' && *q != '%'; ++q)
514 ;
515 if (q > p)
Anton Blanchard5827d412012-11-26 17:40:03 +0000516 call_prom("write", 3, 1, prom.stdout, p, q - p);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000517 if (*q == 0)
518 break;
519 if (*q == '\n') {
520 ++q;
Anton Blanchard5827d412012-11-26 17:40:03 +0000521 call_prom("write", 3, 1, prom.stdout,
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000522 ADDR("\r\n"), 2);
523 continue;
524 }
525 ++q;
526 if (*q == 0)
527 break;
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200528 while (*q == 'l') {
529 ++q;
530 ++n;
531 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000532 switch (*q) {
533 case 's':
534 ++q;
535 s = va_arg(args, const char *);
536 prom_print(s);
537 break;
538 case 'x':
539 ++q;
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200540 switch (n) {
541 case 0:
542 v = va_arg(args, unsigned int);
543 break;
544 case 1:
545 v = va_arg(args, unsigned long);
546 break;
547 case 2:
548 default:
549 v = va_arg(args, unsigned long long);
550 break;
551 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000552 prom_print_hex(v);
553 break;
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200554 case 'u':
555 ++q;
556 switch (n) {
557 case 0:
558 v = va_arg(args, unsigned int);
559 break;
560 case 1:
561 v = va_arg(args, unsigned long);
562 break;
563 case 2:
564 default:
565 v = va_arg(args, unsigned long long);
566 break;
567 }
568 prom_print_dec(v);
569 break;
Benjamin Herrenschmidtaf277142011-04-06 10:51:17 +1000570 case 'd':
571 ++q;
Mathieu Malaterre8af1da42018-05-29 21:20:01 +0200572 switch (n) {
573 case 0:
574 vs = va_arg(args, int);
575 break;
576 case 1:
577 vs = va_arg(args, long);
578 break;
579 case 2:
580 default:
581 vs = va_arg(args, long long);
582 break;
583 }
Benjamin Herrenschmidtaf277142011-04-06 10:51:17 +1000584 if (vs < 0) {
Anton Blanchard5827d412012-11-26 17:40:03 +0000585 prom_print("-");
Benjamin Herrenschmidtaf277142011-04-06 10:51:17 +1000586 vs = -vs;
587 }
588 prom_print_dec(vs);
589 break;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000590 }
591 }
Daniel Axtens1b855e12015-12-17 19:41:00 +1100592 va_end(args);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000593}
594
595
Paul Mackerrasa575b802005-10-23 17:23:21 +1000596static unsigned int __init prom_claim(unsigned long virt, unsigned long size,
597 unsigned long align)
598{
Paul Mackerrasa575b802005-10-23 17:23:21 +1000599
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100600 if (align == 0 && (OF_WORKAROUNDS & OF_WA_CLAIM)) {
601 /*
602 * Old OF requires we claim physical and virtual separately
603 * and then map explicitly (assuming virtual mode)
604 */
605 int ret;
606 prom_arg_t result;
607
608 ret = call_prom_ret("call-method", 5, 2, &result,
Anton Blanchard5827d412012-11-26 17:40:03 +0000609 ADDR("claim"), prom.memory,
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100610 align, size, virt);
611 if (ret != 0 || result == -1)
612 return -1;
613 ret = call_prom_ret("call-method", 5, 2, &result,
Anton Blanchard5827d412012-11-26 17:40:03 +0000614 ADDR("claim"), prom.mmumap,
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100615 align, size, virt);
616 if (ret != 0) {
617 call_prom("call-method", 4, 1, ADDR("release"),
Anton Blanchard5827d412012-11-26 17:40:03 +0000618 prom.memory, size, virt);
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100619 return -1;
620 }
621 /* the 0x12 is M (coherence) + PP == read/write */
Paul Mackerrasa575b802005-10-23 17:23:21 +1000622 call_prom("call-method", 6, 1,
Anton Blanchard5827d412012-11-26 17:40:03 +0000623 ADDR("map"), prom.mmumap, 0x12, size, virt, virt);
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100624 return virt;
625 }
626 return call_prom("claim", 3, 1, (prom_arg_t)virt, (prom_arg_t)size,
627 (prom_arg_t)align);
Paul Mackerrasa575b802005-10-23 17:23:21 +1000628}
629
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000630static void __init __attribute__((noreturn)) prom_panic(const char *reason)
631{
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000632 prom_print(reason);
Olaf Heringadd60ef2006-03-23 22:03:57 +0100633 /* Do not call exit because it clears the screen on pmac
634 * it also causes some sort of double-fault on early pmacs */
Anton Blanchard5827d412012-11-26 17:40:03 +0000635 if (of_platform == PLATFORM_POWERMAC)
Olaf Heringadd60ef2006-03-23 22:03:57 +0100636 asm("trap\n");
637
Stephen Rothwell1d9a4732012-03-21 18:23:27 +0000638 /* ToDo: should put up an SRC here on pSeries */
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000639 call_prom("exit", 0, 0);
640
641 for (;;) /* should never get here */
642 ;
643}
644
645
646static int __init prom_next_node(phandle *nodep)
647{
648 phandle node;
649
650 if ((node = *nodep) != 0
651 && (*nodep = call_prom("child", 1, 1, node)) != 0)
652 return 1;
653 if ((*nodep = call_prom("peer", 1, 1, node)) != 0)
654 return 1;
655 for (;;) {
656 if ((node = call_prom("parent", 1, 1, node)) == 0)
657 return 0;
658 if ((*nodep = call_prom("peer", 1, 1, node)) != 0)
659 return 1;
660 }
661}
662
Masahiro Yamada480795a2019-05-14 15:42:14 -0700663static inline int __init prom_getprop(phandle node, const char *pname,
664 void *value, size_t valuelen)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000665{
666 return call_prom("getprop", 4, 1, node, ADDR(pname),
667 (u32)(unsigned long) value, (u32) valuelen);
668}
669
Masahiro Yamada480795a2019-05-14 15:42:14 -0700670static inline int __init prom_getproplen(phandle node, const char *pname)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000671{
672 return call_prom("getproplen", 2, 1, node, ADDR(pname));
673}
674
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100675static void add_string(char **str, const char *q)
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000676{
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100677 char *p = *str;
678
679 while (*q)
680 *p++ = *q++;
681 *p++ = ' ';
682 *str = p;
683}
684
685static char *tohex(unsigned int x)
686{
Benjamin Herrenschmidt8ca2d512018-10-15 13:49:57 +1100687 static const char digits[] __initconst = "0123456789abcdef";
688 static char result[9] __prombss;
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100689 int i;
690
691 result[8] = 0;
692 i = 8;
693 do {
694 --i;
695 result[i] = digits[x & 0xf];
696 x >>= 4;
697 } while (x != 0 && i > 0);
698 return &result[i];
699}
700
701static int __init prom_setprop(phandle node, const char *nodename,
702 const char *pname, void *value, size_t valuelen)
703{
704 char cmd[256], *p;
705
706 if (!(OF_WORKAROUNDS & OF_WA_LONGTRAIL))
707 return call_prom("setprop", 4, 1, node, ADDR(pname),
708 (u32)(unsigned long) value, (u32) valuelen);
709
710 /* gah... setprop doesn't work on longtrail, have to use interpret */
711 p = cmd;
712 add_string(&p, "dev");
713 add_string(&p, nodename);
714 add_string(&p, tohex((u32)(unsigned long) value));
715 add_string(&p, tohex(valuelen));
716 add_string(&p, tohex(ADDR(pname)));
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000717 add_string(&p, tohex(prom_strlen(pname)));
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100718 add_string(&p, "property");
719 *p = 0;
720 return call_prom("interpret", 1, 1, (u32)(unsigned long) cmd);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000721}
722
Anton Blanchard5827d412012-11-26 17:40:03 +0000723/* We can't use the standard versions because of relocation headaches. */
Andy Shevchenko4cfdd922021-05-10 17:49:25 +0300724#define prom_isxdigit(c) \
725 (('0' <= (c) && (c) <= '9') || ('a' <= (c) && (c) <= 'f') || ('A' <= (c) && (c) <= 'F'))
Benjamin Krillcf687872009-07-27 22:02:39 +0000726
Andy Shevchenko4cfdd922021-05-10 17:49:25 +0300727#define prom_isdigit(c) ('0' <= (c) && (c) <= '9')
728#define prom_islower(c) ('a' <= (c) && (c) <= 'z')
729#define prom_toupper(c) (prom_islower(c) ? ((c) - 'a' + 'A') : (c))
Benjamin Krillcf687872009-07-27 22:02:39 +0000730
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000731static unsigned long prom_strtoul(const char *cp, const char **endp)
Benjamin Krillcf687872009-07-27 22:02:39 +0000732{
733 unsigned long result = 0, base = 10, value;
734
735 if (*cp == '0') {
736 base = 8;
737 cp++;
Andy Shevchenko4cfdd922021-05-10 17:49:25 +0300738 if (prom_toupper(*cp) == 'X') {
Benjamin Krillcf687872009-07-27 22:02:39 +0000739 cp++;
740 base = 16;
741 }
742 }
743
Andy Shevchenko4cfdd922021-05-10 17:49:25 +0300744 while (prom_isxdigit(*cp) &&
745 (value = prom_isdigit(*cp) ? *cp - '0' : prom_toupper(*cp) - 'A' + 10) < base) {
Benjamin Krillcf687872009-07-27 22:02:39 +0000746 result = result * base + value;
747 cp++;
748 }
749
750 if (endp)
751 *endp = cp;
752
753 return result;
754}
755
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +1000756static unsigned long prom_memparse(const char *ptr, const char **retptr)
Benjamin Krillcf687872009-07-27 22:02:39 +0000757{
758 unsigned long ret = prom_strtoul(ptr, retptr);
759 int shift = 0;
760
761 /*
762 * We can't use a switch here because GCC *may* generate a
763 * jump table which won't work, because we're not running at
764 * the address we're linked at.
765 */
766 if ('G' == **retptr || 'g' == **retptr)
767 shift = 30;
768
769 if ('M' == **retptr || 'm' == **retptr)
770 shift = 20;
771
772 if ('K' == **retptr || 'k' == **retptr)
773 shift = 10;
774
775 if (shift) {
776 ret <<= shift;
777 (*retptr)++;
778 }
779
780 return ret;
781}
782
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000783/*
784 * Early parsing of the command line passed to the kernel, used for
785 * "mem=x" and the options that affect the iommu
786 */
787static void __init early_cmdline_parse(void)
788{
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100789 const char *opt;
Benjamin Krillcf687872009-07-27 22:02:39 +0000790
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100791 char *p;
Christophe Leroycbe46bd2019-04-26 16:23:27 +0000792 int l = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000793
Anton Blanchard5827d412012-11-26 17:40:03 +0000794 prom_cmd_line[0] = 0;
795 p = prom_cmd_line;
Chris Packhamd79fbb32019-08-02 10:50:06 +1200796
797 if (!IS_ENABLED(CONFIG_CMDLINE_FORCE) && (long)prom.chosen > 0)
Anton Blanchard5827d412012-11-26 17:40:03 +0000798 l = prom_getprop(prom.chosen, "bootargs", p, COMMAND_LINE_SIZE-1);
Chris Packhamd79fbb32019-08-02 10:50:06 +1200799
800 if (IS_ENABLED(CONFIG_CMDLINE_EXTEND) || l <= 0 || p[0] == '\0')
801 prom_strlcat(prom_cmd_line, " " CONFIG_CMDLINE,
802 sizeof(prom_cmd_line));
803
Anton Blanchard5827d412012-11-26 17:40:03 +0000804 prom_printf("command line: %s\n", prom_cmd_line);
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000805
806#ifdef CONFIG_PPC64
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000807 opt = prom_strstr(prom_cmd_line, "iommu=");
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000808 if (opt) {
809 prom_printf("iommu opt is: %s\n", opt);
810 opt += 6;
811 while (*opt && *opt == ' ')
812 opt++;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000813 if (!prom_strncmp(opt, "off", 3))
Anton Blanchard5827d412012-11-26 17:40:03 +0000814 prom_iommu_off = 1;
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000815 else if (!prom_strncmp(opt, "force", 5))
Anton Blanchard5827d412012-11-26 17:40:03 +0000816 prom_iommu_force_on = 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000817 }
818#endif
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000819 opt = prom_strstr(prom_cmd_line, "mem=");
Benjamin Krillcf687872009-07-27 22:02:39 +0000820 if (opt) {
821 opt += 4;
Anton Blanchard5827d412012-11-26 17:40:03 +0000822 prom_memory_limit = prom_memparse(opt, (const char **)&opt);
Benjamin Krillcf687872009-07-27 22:02:39 +0000823#ifdef CONFIG_PPC64
824 /* Align to 16 MB == size of ppc64 large page */
Anton Blanchard5827d412012-11-26 17:40:03 +0000825 prom_memory_limit = ALIGN(prom_memory_limit, 0x1000000);
Benjamin Krillcf687872009-07-27 22:02:39 +0000826#endif
827 }
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100828
Benjamin Herrenschmidtc8860872018-10-15 13:49:54 +1100829#ifdef CONFIG_PPC_PSERIES
830 prom_radix_disable = !IS_ENABLED(CONFIG_PPC_RADIX_MMU_DEFAULT);
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000831 opt = prom_strstr(prom_cmd_line, "disable_radix");
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100832 if (opt) {
Aneesh Kumar K.V7a22d632018-03-30 17:39:02 +0530833 opt += 13;
834 if (*opt && *opt == '=') {
835 bool val;
836
Christophe Leroy450e7dd2019-04-26 16:23:28 +0000837 if (prom_strtobool(++opt, &val))
Aneesh Kumar K.V7a22d632018-03-30 17:39:02 +0530838 prom_radix_disable = false;
839 else
840 prom_radix_disable = val;
841 } else
842 prom_radix_disable = true;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100843 }
Aneesh Kumar K.V7a22d632018-03-30 17:39:02 +0530844 if (prom_radix_disable)
845 prom_debug("Radix disabled from cmdline\n");
Greg Kurza3bf9fb2019-05-15 12:05:01 +0200846
Aneesh Kumar K.Vbf6b7662020-07-27 14:29:08 +0530847 opt = prom_strstr(prom_cmd_line, "radix_hcall_invalidate=on");
848 if (opt) {
849 prom_radix_gtse_disable = true;
850 prom_debug("Radix GTSE disabled from cmdline\n");
851 }
852
Greg Kurza3bf9fb2019-05-15 12:05:01 +0200853 opt = prom_strstr(prom_cmd_line, "xive=off");
854 if (opt) {
855 prom_xive_disable = true;
856 prom_debug("XIVE disabled from cmdline\n");
857 }
Benjamin Herrenschmidtc8860872018-10-15 13:49:54 +1100858#endif /* CONFIG_PPC_PSERIES */
Ram Pai6a9c9302019-08-19 23:13:14 -0300859
860#ifdef CONFIG_PPC_SVM
861 opt = prom_strstr(prom_cmd_line, "svm=");
862 if (opt) {
863 bool val;
864
865 opt += sizeof("svm=") - 1;
866 if (!prom_strtobool(opt, &val))
867 prom_svm_enable = val;
868 }
869#endif /* CONFIG_PPC_SVM */
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000870}
871
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +1100872#ifdef CONFIG_PPC_PSERIES
Paul Mackerras9b6b5632005-10-06 12:06:20 +1000873/*
Nathan Fontenot530b5e12013-04-24 05:53:10 +0000874 * The architecture vector has an array of PVR mask/value pairs,
875 * followed by # option vectors - 1, followed by the option vectors.
876 *
877 * See prom.h for the definition of the bits specified in the
878 * architecture vector.
Paul Mackerrasf709bfa2006-04-28 16:28:35 +1000879 */
Paul Mackerrasf709bfa2006-04-28 16:28:35 +1000880
Michael Ellermane8a4fd02014-08-29 17:01:43 +1000881/* Firmware expects the value to be n - 1, where n is the # of vectors */
882#define NUM_VECTORS(n) ((n) - 1)
883
884/*
885 * Firmware expects 1 + n - 2, where n is the length of the option vector in
886 * bytes. The 1 accounts for the length byte itself, the - 2 .. ?
887 */
888#define VECTOR_LENGTH(n) (1 + (n) - 2)
889
Michael Ellermand03d1d62016-11-18 23:15:41 +1100890struct option_vector1 {
891 u8 byte1;
892 u8 arch_versions;
Paul Mackerrascc3d2942017-01-30 21:21:36 +1100893 u8 arch_versions3;
Michael Ellermand03d1d62016-11-18 23:15:41 +1100894} __packed;
895
896struct option_vector2 {
897 u8 byte1;
898 __be16 reserved;
899 __be32 real_base;
900 __be32 real_size;
901 __be32 virt_base;
902 __be32 virt_size;
903 __be32 load_base;
904 __be32 min_rma;
905 __be32 min_load;
906 u8 min_rma_percent;
907 u8 max_pft_size;
908} __packed;
909
910struct option_vector3 {
911 u8 byte1;
912 u8 byte2;
913} __packed;
914
915struct option_vector4 {
916 u8 byte1;
917 u8 min_vp_cap;
918} __packed;
919
920struct option_vector5 {
921 u8 byte1;
922 u8 byte2;
923 u8 byte3;
924 u8 cmo;
925 u8 associativity;
926 u8 bin_opts;
927 u8 micro_checkpoint;
928 u8 reserved0;
929 __be32 max_cpus;
930 __be16 papr_level;
931 __be16 reserved1;
932 u8 platform_facilities;
933 u8 reserved2;
934 __be16 reserved3;
935 u8 subprocessors;
Paul Mackerrascc3d2942017-01-30 21:21:36 +1100936 u8 byte22;
937 u8 intarch;
938 u8 mmu;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +1100939 u8 hash_ext;
940 u8 radix_ext;
Michael Ellermand03d1d62016-11-18 23:15:41 +1100941} __packed;
942
943struct option_vector6 {
944 u8 reserved;
945 u8 secondary_pteg;
946 u8 os_name;
947} __packed;
948
Michael Ellermanffaacd92021-06-21 16:49:38 +1000949struct option_vector7 {
950 u8 os_id[256];
951} __packed;
952
Michael Ellerman76ffb572016-11-18 23:15:42 +1100953struct ibm_arch_vec {
Alistair Popplea3ea40d2020-05-21 11:43:41 +1000954 struct { u32 mask, val; } pvrs[14];
Michael Ellerman76ffb572016-11-18 23:15:42 +1100955
956 u8 num_vectors;
957
958 u8 vec1_len;
959 struct option_vector1 vec1;
960
961 u8 vec2_len;
962 struct option_vector2 vec2;
963
964 u8 vec3_len;
965 struct option_vector3 vec3;
966
967 u8 vec4_len;
968 struct option_vector4 vec4;
969
970 u8 vec5_len;
971 struct option_vector5 vec5;
972
973 u8 vec6_len;
974 struct option_vector6 vec6;
Michael Ellermanffaacd92021-06-21 16:49:38 +1000975
976 u8 vec7_len;
977 struct option_vector7 vec7;
Michael Ellerman76ffb572016-11-18 23:15:42 +1100978} __packed;
979
Benjamin Herrenschmidtd00e34b2018-10-15 13:49:56 +1100980static const struct ibm_arch_vec ibm_architecture_vec_template __initconst = {
Michael Ellerman76ffb572016-11-18 23:15:42 +1100981 .pvrs = {
982 {
983 .mask = cpu_to_be32(0xfffe0000), /* POWER5/POWER5+ */
984 .val = cpu_to_be32(0x003a0000),
985 },
986 {
987 .mask = cpu_to_be32(0xffff0000), /* POWER6 */
988 .val = cpu_to_be32(0x003e0000),
989 },
990 {
991 .mask = cpu_to_be32(0xffff0000), /* POWER7 */
992 .val = cpu_to_be32(0x003f0000),
993 },
994 {
995 .mask = cpu_to_be32(0xffff0000), /* POWER8E */
996 .val = cpu_to_be32(0x004b0000),
997 },
998 {
999 .mask = cpu_to_be32(0xffff0000), /* POWER8NVL */
1000 .val = cpu_to_be32(0x004c0000),
1001 },
1002 {
1003 .mask = cpu_to_be32(0xffff0000), /* POWER8 */
1004 .val = cpu_to_be32(0x004d0000),
1005 },
1006 {
Paul Mackerrascc3d2942017-01-30 21:21:36 +11001007 .mask = cpu_to_be32(0xffff0000), /* POWER9 */
1008 .val = cpu_to_be32(0x004e0000),
1009 },
1010 {
Alistair Popplea3ea40d2020-05-21 11:43:41 +10001011 .mask = cpu_to_be32(0xffff0000), /* POWER10 */
1012 .val = cpu_to_be32(0x00800000),
1013 },
1014 {
1015 .mask = cpu_to_be32(0xffffffff), /* all 3.1-compliant */
1016 .val = cpu_to_be32(0x0f000006),
1017 },
1018 {
Paul Mackerrascc3d2942017-01-30 21:21:36 +11001019 .mask = cpu_to_be32(0xffffffff), /* all 3.00-compliant */
1020 .val = cpu_to_be32(0x0f000005),
1021 },
1022 {
Michael Ellerman76ffb572016-11-18 23:15:42 +11001023 .mask = cpu_to_be32(0xffffffff), /* all 2.07-compliant */
1024 .val = cpu_to_be32(0x0f000004),
1025 },
1026 {
1027 .mask = cpu_to_be32(0xffffffff), /* all 2.06-compliant */
1028 .val = cpu_to_be32(0x0f000003),
1029 },
1030 {
1031 .mask = cpu_to_be32(0xffffffff), /* all 2.05-compliant */
1032 .val = cpu_to_be32(0x0f000002),
1033 },
1034 {
1035 .mask = cpu_to_be32(0xfffffffe), /* all 2.04-compliant and earlier */
1036 .val = cpu_to_be32(0x0f000001),
1037 },
1038 },
1039
1040 .num_vectors = NUM_VECTORS(6),
1041
1042 .vec1_len = VECTOR_LENGTH(sizeof(struct option_vector1)),
1043 .vec1 = {
1044 .byte1 = 0,
1045 .arch_versions = OV1_PPC_2_00 | OV1_PPC_2_01 | OV1_PPC_2_02 | OV1_PPC_2_03 |
1046 OV1_PPC_2_04 | OV1_PPC_2_05 | OV1_PPC_2_06 | OV1_PPC_2_07,
Alistair Popplea3ea40d2020-05-21 11:43:41 +10001047 .arch_versions3 = OV1_PPC_3_00 | OV1_PPC_3_1,
Michael Ellerman76ffb572016-11-18 23:15:42 +11001048 },
1049
1050 .vec2_len = VECTOR_LENGTH(sizeof(struct option_vector2)),
1051 /* option vector 2: Open Firmware options supported */
1052 .vec2 = {
1053 .byte1 = OV2_REAL_MODE,
1054 .reserved = 0,
1055 .real_base = cpu_to_be32(0xffffffff),
1056 .real_size = cpu_to_be32(0xffffffff),
1057 .virt_base = cpu_to_be32(0xffffffff),
1058 .virt_size = cpu_to_be32(0xffffffff),
1059 .load_base = cpu_to_be32(0xffffffff),
Sukadev Bhattiprolu687da8f2017-03-27 19:43:14 -04001060 .min_rma = cpu_to_be32(512), /* 512MB min RMA */
Michael Ellerman76ffb572016-11-18 23:15:42 +11001061 .min_load = cpu_to_be32(0xffffffff), /* full client load */
1062 .min_rma_percent = 0, /* min RMA percentage of total RAM */
1063 .max_pft_size = 48, /* max log_2(hash table size) */
1064 },
1065
1066 .vec3_len = VECTOR_LENGTH(sizeof(struct option_vector3)),
1067 /* option vector 3: processor options supported */
1068 .vec3 = {
1069 .byte1 = 0, /* don't ignore, don't halt */
1070 .byte2 = OV3_FP | OV3_VMX | OV3_DFP,
1071 },
1072
1073 .vec4_len = VECTOR_LENGTH(sizeof(struct option_vector4)),
1074 /* option vector 4: IBM PAPR implementation */
1075 .vec4 = {
1076 .byte1 = 0, /* don't halt */
1077 .min_vp_cap = OV4_MIN_ENT_CAP, /* minimum VP entitled capacity */
1078 },
1079
1080 .vec5_len = VECTOR_LENGTH(sizeof(struct option_vector5)),
1081 /* option vector 5: PAPR/OF options */
1082 .vec5 = {
1083 .byte1 = 0, /* don't ignore, don't halt */
1084 .byte2 = OV5_FEAT(OV5_LPAR) | OV5_FEAT(OV5_SPLPAR) | OV5_FEAT(OV5_LARGE_PAGES) |
1085 OV5_FEAT(OV5_DRCONF_MEMORY) | OV5_FEAT(OV5_DONATE_DEDICATE_CPU) |
1086#ifdef CONFIG_PCI_MSI
1087 /* PCIe/MSI support. Without MSI full PCIe is not supported */
1088 OV5_FEAT(OV5_MSI),
1089#else
1090 0,
1091#endif
1092 .byte3 = 0,
1093 .cmo =
1094#ifdef CONFIG_PPC_SMLPAR
1095 OV5_FEAT(OV5_CMO) | OV5_FEAT(OV5_XCMO),
1096#else
1097 0,
1098#endif
1099 .associativity = OV5_FEAT(OV5_TYPE1_AFFINITY) | OV5_FEAT(OV5_PRRN),
Michael Roth3dbbaf22017-02-20 19:12:18 -06001100 .bin_opts = OV5_FEAT(OV5_RESIZE_HPT) | OV5_FEAT(OV5_HP_EVT),
Michael Ellerman76ffb572016-11-18 23:15:42 +11001101 .micro_checkpoint = 0,
1102 .reserved0 = 0,
1103 .max_cpus = cpu_to_be32(NR_CPUS), /* number of cores supported */
1104 .papr_level = 0,
1105 .reserved1 = 0,
1106 .platform_facilities = OV5_FEAT(OV5_PFO_HW_RNG) | OV5_FEAT(OV5_PFO_HW_ENCR) | OV5_FEAT(OV5_PFO_HW_842),
1107 .reserved2 = 0,
1108 .reserved3 = 0,
1109 .subprocessors = 1,
Tyrel Datwyler0a87ccd2019-11-10 23:21:37 -06001110 .byte22 = OV5_FEAT(OV5_DRMEM_V2) | OV5_FEAT(OV5_DRC_INFO),
Paul Mackerrascc3d2942017-01-30 21:21:36 +11001111 .intarch = 0,
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001112 .mmu = 0,
1113 .hash_ext = 0,
1114 .radix_ext = 0,
Michael Ellerman76ffb572016-11-18 23:15:42 +11001115 },
1116
1117 /* option vector 6: IBM PAPR hints */
1118 .vec6_len = VECTOR_LENGTH(sizeof(struct option_vector6)),
1119 .vec6 = {
1120 .reserved = 0,
1121 .secondary_pteg = 0,
1122 .os_name = OV6_LINUX,
1123 },
Michael Ellermanffaacd92021-06-21 16:49:38 +10001124
1125 /* option vector 7: OS Identification */
1126 .vec7_len = VECTOR_LENGTH(sizeof(struct option_vector7)),
Michael Ellerman76ffb572016-11-18 23:15:42 +11001127};
1128
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001129static struct ibm_arch_vec __prombss ibm_architecture_vec ____cacheline_aligned;
1130
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001131/* Old method - ELF header with PT_NOTE sections only works on BE */
1132#ifdef __BIG_ENDIAN__
Benjamin Herrenschmidt30c69ca2018-05-31 14:33:40 +10001133static const struct fake_elf {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001134 Elf32_Ehdr elfhdr;
1135 Elf32_Phdr phdr[2];
1136 struct chrpnote {
1137 u32 namesz;
1138 u32 descsz;
1139 u32 type;
1140 char name[8]; /* "PowerPC" */
1141 struct chrpdesc {
1142 u32 real_mode;
1143 u32 real_base;
1144 u32 real_size;
1145 u32 virt_base;
1146 u32 virt_size;
1147 u32 load_base;
1148 } chrpdesc;
1149 } chrpnote;
1150 struct rpanote {
1151 u32 namesz;
1152 u32 descsz;
1153 u32 type;
1154 char name[24]; /* "IBM,RPA-Client-Config" */
1155 struct rpadesc {
1156 u32 lpar_affinity;
1157 u32 min_rmo_size;
1158 u32 min_rmo_percent;
1159 u32 max_pft_size;
1160 u32 splpar;
1161 u32 min_load;
1162 u32 new_mem_def;
1163 u32 ignore_me;
1164 } rpadesc;
1165 } rpanote;
Benjamin Herrenschmidtd00e34b2018-10-15 13:49:56 +11001166} fake_elf __initconst = {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001167 .elfhdr = {
1168 .e_ident = { 0x7f, 'E', 'L', 'F',
1169 ELFCLASS32, ELFDATA2MSB, EV_CURRENT },
1170 .e_type = ET_EXEC, /* yeah right */
1171 .e_machine = EM_PPC,
1172 .e_version = EV_CURRENT,
1173 .e_phoff = offsetof(struct fake_elf, phdr),
1174 .e_phentsize = sizeof(Elf32_Phdr),
1175 .e_phnum = 2
1176 },
1177 .phdr = {
1178 [0] = {
1179 .p_type = PT_NOTE,
1180 .p_offset = offsetof(struct fake_elf, chrpnote),
1181 .p_filesz = sizeof(struct chrpnote)
1182 }, [1] = {
1183 .p_type = PT_NOTE,
1184 .p_offset = offsetof(struct fake_elf, rpanote),
1185 .p_filesz = sizeof(struct rpanote)
1186 }
1187 },
1188 .chrpnote = {
1189 .namesz = sizeof("PowerPC"),
1190 .descsz = sizeof(struct chrpdesc),
1191 .type = 0x1275,
1192 .name = "PowerPC",
1193 .chrpdesc = {
1194 .real_mode = ~0U, /* ~0 means "don't care" */
1195 .real_base = ~0U,
1196 .real_size = ~0U,
1197 .virt_base = ~0U,
1198 .virt_size = ~0U,
1199 .load_base = ~0U
1200 },
1201 },
1202 .rpanote = {
1203 .namesz = sizeof("IBM,RPA-Client-Config"),
1204 .descsz = sizeof(struct rpadesc),
1205 .type = 0x12759999,
1206 .name = "IBM,RPA-Client-Config",
1207 .rpadesc = {
Paul Mackerras5663a122008-10-31 22:27:17 +11001208 .lpar_affinity = 0,
1209 .min_rmo_size = 64, /* in megabytes */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001210 .min_rmo_percent = 0,
Paul Mackerras5663a122008-10-31 22:27:17 +11001211 .max_pft_size = 48, /* 2^48 bytes max PFT size */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001212 .splpar = 1,
1213 .min_load = ~0U,
Paul Mackerras5663a122008-10-31 22:27:17 +11001214 .new_mem_def = 0
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001215 }
1216 }
1217};
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001218#endif /* __BIG_ENDIAN__ */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001219
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001220static int __init prom_count_smt_threads(void)
1221{
1222 phandle node;
1223 char type[64];
1224 unsigned int plen;
1225
1226 /* Pick up th first CPU node we can find */
1227 for (node = 0; prom_next_node(&node); ) {
1228 type[0] = 0;
1229 prom_getprop(node, "device_type", type, sizeof(type));
1230
Christophe Leroy450e7dd2019-04-26 16:23:28 +00001231 if (prom_strcmp(type, "cpu"))
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001232 continue;
1233 /*
1234 * There is an entry for each smt thread, each entry being
1235 * 4 bytes long. All cpus should have the same number of
1236 * smt threads, so return after finding the first.
1237 */
1238 plen = prom_getproplen(node, "ibm,ppc-interrupt-server#s");
1239 if (plen == PROM_ERROR)
1240 break;
1241 plen >>= 2;
Michael Neuling2c48a7d2010-07-27 18:26:21 +00001242 prom_debug("Found %lu smt threads per core\n", (unsigned long)plen);
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001243
1244 /* Sanity check */
1245 if (plen < 1 || plen > 64) {
Michael Neuling2c48a7d2010-07-27 18:26:21 +00001246 prom_printf("Threads per core %lu out of bounds, assuming 1\n",
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001247 (unsigned long)plen);
1248 return 1;
1249 }
1250 return plen;
1251 }
1252 prom_debug("No threads found, assuming 1 per core\n");
1253
1254 return 1;
1255
1256}
1257
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001258static void __init prom_parse_mmu_model(u8 val,
1259 struct platform_support *support)
1260{
1261 switch (val) {
1262 case OV5_FEAT(OV5_MMU_DYNAMIC):
1263 case OV5_FEAT(OV5_MMU_EITHER): /* Either Available */
1264 prom_debug("MMU - either supported\n");
1265 support->radix_mmu = !prom_radix_disable;
1266 support->hash_mmu = true;
1267 break;
1268 case OV5_FEAT(OV5_MMU_RADIX): /* Only Radix */
1269 prom_debug("MMU - radix only\n");
1270 if (prom_radix_disable) {
1271 /*
1272 * If we __have__ to do radix, we're better off ignoring
1273 * the command line rather than not booting.
1274 */
1275 prom_printf("WARNING: Ignoring cmdline option disable_radix\n");
1276 }
1277 support->radix_mmu = true;
1278 break;
1279 case OV5_FEAT(OV5_MMU_HASH):
1280 prom_debug("MMU - hash only\n");
1281 support->hash_mmu = true;
1282 break;
1283 default:
1284 prom_debug("Unknown mmu support option: 0x%x\n", val);
1285 break;
1286 }
1287}
1288
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001289static void __init prom_parse_xive_model(u8 val,
1290 struct platform_support *support)
1291{
1292 switch (val) {
1293 case OV5_FEAT(OV5_XIVE_EITHER): /* Either Available */
1294 prom_debug("XIVE - either mode supported\n");
Greg Kurza3bf9fb2019-05-15 12:05:01 +02001295 support->xive = !prom_xive_disable;
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001296 break;
1297 case OV5_FEAT(OV5_XIVE_EXPLOIT): /* Only Exploitation mode */
1298 prom_debug("XIVE - exploitation mode supported\n");
Greg Kurza3bf9fb2019-05-15 12:05:01 +02001299 if (prom_xive_disable) {
1300 /*
1301 * If we __have__ to do XIVE, we're better off ignoring
1302 * the command line rather than not booting.
1303 */
1304 prom_printf("WARNING: Ignoring cmdline option xive=off\n");
1305 }
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001306 support->xive = true;
1307 break;
1308 case OV5_FEAT(OV5_XIVE_LEGACY): /* Only Legacy mode */
1309 prom_debug("XIVE - legacy mode supported\n");
1310 break;
1311 default:
1312 prom_debug("Unknown xive support option: 0x%x\n", val);
1313 break;
1314 }
1315}
1316
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001317static void __init prom_parse_platform_support(u8 index, u8 val,
1318 struct platform_support *support)
1319{
1320 switch (index) {
1321 case OV5_INDX(OV5_MMU_SUPPORT): /* MMU Model */
1322 prom_parse_mmu_model(val & OV5_FEAT(OV5_MMU_SUPPORT), support);
1323 break;
1324 case OV5_INDX(OV5_RADIX_GTSE): /* Radix Extensions */
Aneesh Kumar K.Vbf6b7662020-07-27 14:29:08 +05301325 if (val & OV5_FEAT(OV5_RADIX_GTSE))
1326 support->radix_gtse = !prom_radix_gtse_disable;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001327 break;
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001328 case OV5_INDX(OV5_XIVE_SUPPORT): /* Interrupt mode */
1329 prom_parse_xive_model(val & OV5_FEAT(OV5_XIVE_SUPPORT),
1330 support);
1331 break;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001332 }
1333}
1334
1335static void __init prom_check_platform_support(void)
1336{
1337 struct platform_support supported = {
1338 .hash_mmu = false,
1339 .radix_mmu = false,
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001340 .radix_gtse = false,
1341 .xive = false
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001342 };
1343 int prop_len = prom_getproplen(prom.chosen,
1344 "ibm,arch-vec-5-platform-support");
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001345
Christophe Leroyadcf5912019-04-26 16:23:29 +00001346 /*
1347 * First copy the architecture vec template
1348 *
1349 * use memcpy() instead of *vec = *vec_template so that GCC replaces it
1350 * by __memcpy() when KASAN is active
1351 */
1352 memcpy(&ibm_architecture_vec, &ibm_architecture_vec_template,
1353 sizeof(ibm_architecture_vec));
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001354
Michael Ellermanffaacd92021-06-21 16:49:38 +10001355 prom_strscpy_pad(ibm_architecture_vec.vec7.os_id, linux_banner, 256);
1356
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001357 if (prop_len > 1) {
1358 int i;
Suraj Jitindar Singhab912392018-09-05 12:09:50 +10001359 u8 vec[8];
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001360 prom_debug("Found ibm,arch-vec-5-platform-support, len: %d\n",
1361 prop_len);
Suraj Jitindar Singhab912392018-09-05 12:09:50 +10001362 if (prop_len > sizeof(vec))
1363 prom_printf("WARNING: ibm,arch-vec-5-platform-support longer than expected (len: %d)\n",
1364 prop_len);
Cédric Le Goatered5b00a2021-01-22 08:50:29 +01001365 prom_getprop(prom.chosen, "ibm,arch-vec-5-platform-support", &vec, sizeof(vec));
1366 for (i = 0; i < prop_len; i += 2) {
1367 prom_debug("%d: index = 0x%x val = 0x%x\n", i / 2, vec[i], vec[i + 1]);
1368 prom_parse_platform_support(vec[i], vec[i + 1], &supported);
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001369 }
1370 }
1371
Bharata B Rao029ab302020-07-03 11:06:06 +05301372 if (supported.radix_mmu && IS_ENABLED(CONFIG_PPC_RADIX_MMU)) {
1373 /* Radix preferred - Check if GTSE is also supported */
1374 prom_debug("Asking for radix\n");
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001375 ibm_architecture_vec.vec5.mmu = OV5_FEAT(OV5_MMU_RADIX);
Bharata B Rao029ab302020-07-03 11:06:06 +05301376 if (supported.radix_gtse)
1377 ibm_architecture_vec.vec5.radix_ext =
1378 OV5_FEAT(OV5_RADIX_GTSE);
1379 else
1380 prom_debug("Radix GTSE isn't supported\n");
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001381 } else if (supported.hash_mmu) {
1382 /* Default to hash mmu (if we can) */
1383 prom_debug("Asking for hash\n");
1384 ibm_architecture_vec.vec5.mmu = OV5_FEAT(OV5_MMU_HASH);
1385 } else {
1386 /* We're probably on a legacy hypervisor */
1387 prom_debug("Assuming legacy hash support\n");
1388 }
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001389
1390 if (supported.xive) {
1391 prom_debug("Asking for XIVE\n");
1392 ibm_architecture_vec.vec5.intarch = OV5_FEAT(OV5_XIVE_EXPLOIT);
1393 }
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001394}
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001395
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001396static void __init prom_send_capabilities(void)
1397{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001398 ihandle root;
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001399 prom_arg_t ret;
Laurent Dufourdbd0c5d2013-09-17 11:52:48 +02001400 u32 cores;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001401
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001402 /* Check ibm,arch-vec-5-platform-support and fixup vec5 if required */
1403 prom_check_platform_support();
1404
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001405 root = call_prom("open", 1, 1, ADDR("/"));
1406 if (root != 0) {
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001407 /* We need to tell the FW about the number of cores we support.
1408 *
1409 * To do that, we count the number of threads on the first core
1410 * (we assume this is the same for all cores) and use it to
1411 * divide NR_CPUS.
1412 */
Laurent Dufourdbd0c5d2013-09-17 11:52:48 +02001413
Michael Ellerman76ffb572016-11-18 23:15:42 +11001414 cores = DIV_ROUND_UP(NR_CPUS, prom_count_smt_threads());
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001415 prom_printf("Max number of cores passed to firmware: %u (NR_CPUS = %d)\n",
Michael Ellerman76ffb572016-11-18 23:15:42 +11001416 cores, NR_CPUS);
1417
1418 ibm_architecture_vec.vec5.max_cpus = cpu_to_be32(cores);
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001419
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001420 /* try calling the ibm,client-architecture-support method */
Anton Blanchard049d0492009-09-21 20:47:39 +00001421 prom_printf("Calling ibm,client-architecture-support...");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001422 if (call_prom_ret("call-method", 3, 2, &ret,
1423 ADDR("ibm,client-architecture-support"),
Benjamin Herrenschmidt33b74972006-06-07 12:01:32 +10001424 root,
Michael Ellerman76ffb572016-11-18 23:15:42 +11001425 ADDR(&ibm_architecture_vec)) == 0) {
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001426 /* the call exists... */
1427 if (ret)
Anton Blanchard4da727a2009-03-31 20:06:14 +00001428 prom_printf("\nWARNING: ibm,client-architecture"
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001429 "-support call FAILED!\n");
1430 call_prom("close", 1, 0, root);
Anton Blanchard4da727a2009-03-31 20:06:14 +00001431 prom_printf(" done\n");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001432 return;
1433 }
1434 call_prom("close", 1, 0, root);
Anton Blanchard049d0492009-09-21 20:47:39 +00001435 prom_printf(" not implemented\n");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001436 }
1437
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001438#ifdef __BIG_ENDIAN__
1439 {
1440 ihandle elfloader;
1441
1442 /* no ibm,client-architecture-support call, try the old way */
1443 elfloader = call_prom("open", 1, 1,
1444 ADDR("/packages/elf-loader"));
1445 if (elfloader == 0) {
1446 prom_printf("couldn't open /packages/elf-loader\n");
1447 return;
1448 }
1449 call_prom("call-method", 3, 1, ADDR("process-elf-header"),
1450 elfloader, ADDR(&fake_elf));
1451 call_prom("close", 1, 0, elfloader);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001452 }
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001453#endif /* __BIG_ENDIAN__ */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001454}
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11001455#endif /* CONFIG_PPC_PSERIES */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001456
1457/*
1458 * Memory allocation strategy... our layout is normally:
1459 *
1460 * at 14Mb or more we have vmlinux, then a gap and initrd. In some
1461 * rare cases, initrd might end up being before the kernel though.
1462 * We assume this won't override the final kernel at 0, we have no
1463 * provision to handle that in this version, but it should hopefully
1464 * never happen.
1465 *
1466 * alloc_top is set to the top of RMO, eventually shrink down if the
1467 * TCEs overlap
1468 *
1469 * alloc_bottom is set to the top of kernel/initrd
1470 *
1471 * from there, allocations are done this way : rtas is allocated
1472 * topmost, and the device-tree is allocated from the bottom. We try
1473 * to grow the device-tree allocation as we progress. If we can't,
1474 * then we fail, we don't currently have a facility to restart
1475 * elsewhere, but that shouldn't be necessary.
1476 *
1477 * Note that calls to reserve_mem have to be done explicitly, memory
1478 * allocated with either alloc_up or alloc_down isn't automatically
1479 * reserved.
1480 */
1481
1482
1483/*
1484 * Allocates memory in the RMO upward from the kernel/initrd
1485 *
1486 * When align is 0, this is a special case, it means to allocate in place
1487 * at the current location of alloc_bottom or fail (that is basically
1488 * extending the previous allocation). Used for the device-tree flattening
1489 */
1490static unsigned long __init alloc_up(unsigned long size, unsigned long align)
1491{
Anton Blanchard5827d412012-11-26 17:40:03 +00001492 unsigned long base = alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001493 unsigned long addr = 0;
1494
Paul Mackerrasc49888202005-10-26 21:52:53 +10001495 if (align)
Christophe Leroyb7115312020-04-20 18:36:36 +00001496 base = ALIGN(base, align);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001497 prom_debug("%s(%lx, %lx)\n", __func__, size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001498 if (ram_top == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001499 prom_panic("alloc_up() called with mem not initialized\n");
1500
1501 if (align)
Christophe Leroyb7115312020-04-20 18:36:36 +00001502 base = ALIGN(alloc_bottom, align);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001503 else
Anton Blanchard5827d412012-11-26 17:40:03 +00001504 base = alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001505
Anton Blanchard5827d412012-11-26 17:40:03 +00001506 for(; (base + size) <= alloc_top;
Christophe Leroyb7115312020-04-20 18:36:36 +00001507 base = ALIGN(base + 0x100000, align)) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001508 prom_debug(" trying: 0x%lx\n\r", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001509 addr = (unsigned long)prom_claim(base, size, 0);
Paul Mackerrasc49888202005-10-26 21:52:53 +10001510 if (addr != PROM_ERROR && addr != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001511 break;
1512 addr = 0;
1513 if (align == 0)
1514 break;
1515 }
1516 if (addr == 0)
1517 return 0;
Anton Blanchard5827d412012-11-26 17:40:03 +00001518 alloc_bottom = addr + size;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001519
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001520 prom_debug(" -> %lx\n", addr);
1521 prom_debug(" alloc_bottom : %lx\n", alloc_bottom);
1522 prom_debug(" alloc_top : %lx\n", alloc_top);
1523 prom_debug(" alloc_top_hi : %lx\n", alloc_top_high);
1524 prom_debug(" rmo_top : %lx\n", rmo_top);
1525 prom_debug(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001526
1527 return addr;
1528}
1529
1530/*
1531 * Allocates memory downward, either from top of RMO, or if highmem
1532 * is set, from the top of RAM. Note that this one doesn't handle
1533 * failures. It does claim memory if highmem is not set.
1534 */
1535static unsigned long __init alloc_down(unsigned long size, unsigned long align,
1536 int highmem)
1537{
1538 unsigned long base, addr = 0;
1539
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001540 prom_debug("%s(%lx, %lx, %s)\n", __func__, size, align,
Anton Blanchard5827d412012-11-26 17:40:03 +00001541 highmem ? "(high)" : "(low)");
1542 if (ram_top == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001543 prom_panic("alloc_down() called with mem not initialized\n");
1544
1545 if (highmem) {
1546 /* Carve out storage for the TCE table. */
Christophe Leroye96d9042020-04-20 18:36:35 +00001547 addr = ALIGN_DOWN(alloc_top_high - size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001548 if (addr <= alloc_bottom)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001549 return 0;
1550 /* Will we bump into the RMO ? If yes, check out that we
1551 * didn't overlap existing allocations there, if we did,
1552 * we are dead, we must be the first in town !
1553 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001554 if (addr < rmo_top) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001555 /* Good, we are first */
Anton Blanchard5827d412012-11-26 17:40:03 +00001556 if (alloc_top == rmo_top)
1557 alloc_top = rmo_top = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001558 else
1559 return 0;
1560 }
Anton Blanchard5827d412012-11-26 17:40:03 +00001561 alloc_top_high = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001562 goto bail;
1563 }
1564
Christophe Leroye96d9042020-04-20 18:36:35 +00001565 base = ALIGN_DOWN(alloc_top - size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001566 for (; base > alloc_bottom;
Christophe Leroye96d9042020-04-20 18:36:35 +00001567 base = ALIGN_DOWN(base - 0x100000, align)) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001568 prom_debug(" trying: 0x%lx\n\r", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001569 addr = (unsigned long)prom_claim(base, size, 0);
Paul Mackerrasc49888202005-10-26 21:52:53 +10001570 if (addr != PROM_ERROR && addr != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001571 break;
1572 addr = 0;
1573 }
1574 if (addr == 0)
1575 return 0;
Anton Blanchard5827d412012-11-26 17:40:03 +00001576 alloc_top = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001577
1578 bail:
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001579 prom_debug(" -> %lx\n", addr);
1580 prom_debug(" alloc_bottom : %lx\n", alloc_bottom);
1581 prom_debug(" alloc_top : %lx\n", alloc_top);
1582 prom_debug(" alloc_top_hi : %lx\n", alloc_top_high);
1583 prom_debug(" rmo_top : %lx\n", rmo_top);
1584 prom_debug(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001585
1586 return addr;
1587}
1588
1589/*
1590 * Parse a "reg" cell
1591 */
1592static unsigned long __init prom_next_cell(int s, cell_t **cellp)
1593{
1594 cell_t *p = *cellp;
1595 unsigned long r = 0;
1596
1597 /* Ignore more than 2 cells */
1598 while (s > sizeof(unsigned long) / 4) {
1599 p++;
1600 s--;
1601 }
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001602 r = be32_to_cpu(*p++);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001603#ifdef CONFIG_PPC64
Paul Mackerras35499c02005-10-22 16:02:39 +10001604 if (s > 1) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001605 r <<= 32;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001606 r |= be32_to_cpu(*(p++));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001607 }
1608#endif
1609 *cellp = p;
1610 return r;
1611}
1612
1613/*
1614 * Very dumb function for adding to the memory reserve list, but
1615 * we don't need anything smarter at this point
1616 *
1617 * XXX Eventually check for collisions. They should NEVER happen.
1618 * If problems seem to show up, it would be a good start to track
1619 * them down.
1620 */
Michael Ellerman0108d3f2007-05-07 15:58:28 +10001621static void __init reserve_mem(u64 base, u64 size)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001622{
Kumar Galacbbcf342006-01-11 17:57:13 -06001623 u64 top = base + size;
Anton Blanchard5827d412012-11-26 17:40:03 +00001624 unsigned long cnt = mem_reserve_cnt;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001625
1626 if (size == 0)
1627 return;
1628
1629 /* We need to always keep one empty entry so that we
1630 * have our terminator with "size" set to 0 since we are
1631 * dumb and just copy this entire array to the boot params
1632 */
Christophe Leroye96d9042020-04-20 18:36:35 +00001633 base = ALIGN_DOWN(base, PAGE_SIZE);
Christophe Leroyb7115312020-04-20 18:36:36 +00001634 top = ALIGN(top, PAGE_SIZE);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001635 size = top - base;
1636
1637 if (cnt >= (MEM_RESERVE_MAP_SIZE - 1))
1638 prom_panic("Memory reserve map exhausted !\n");
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001639 mem_reserve_map[cnt].base = cpu_to_be64(base);
1640 mem_reserve_map[cnt].size = cpu_to_be64(size);
Anton Blanchard5827d412012-11-26 17:40:03 +00001641 mem_reserve_cnt = cnt + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001642}
1643
1644/*
Adrian Bunkb3c2ffd2006-06-30 18:20:44 +02001645 * Initialize memory allocation mechanism, parse "memory" nodes and
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001646 * obtain that way the top of memory and RMO to setup out local allocator
1647 */
1648static void __init prom_init_mem(void)
1649{
1650 phandle node;
Mathieu Malaterreeab00a22018-04-04 22:08:35 +02001651 char type[64];
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001652 unsigned int plen;
1653 cell_t *p, *endp;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001654 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001655 u32 rac, rsc;
1656
1657 /*
1658 * We iterate the memory nodes to find
1659 * 1) top of RMO (first node)
1660 * 2) top of memory
1661 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001662 val = cpu_to_be32(2);
1663 prom_getprop(prom.root, "#address-cells", &val, sizeof(val));
1664 rac = be32_to_cpu(val);
1665 val = cpu_to_be32(1);
1666 prom_getprop(prom.root, "#size-cells", &val, sizeof(rsc));
1667 rsc = be32_to_cpu(val);
1668 prom_debug("root_addr_cells: %x\n", rac);
1669 prom_debug("root_size_cells: %x\n", rsc);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001670
1671 prom_debug("scanning memory:\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001672
1673 for (node = 0; prom_next_node(&node); ) {
1674 type[0] = 0;
1675 prom_getprop(node, "device_type", type, sizeof(type));
1676
Paul Mackerrasc49888202005-10-26 21:52:53 +10001677 if (type[0] == 0) {
1678 /*
1679 * CHRP Longtrail machines have no device_type
1680 * on the memory node, so check the name instead...
1681 */
1682 prom_getprop(node, "name", type, sizeof(type));
1683 }
Christophe Leroy450e7dd2019-04-26 16:23:28 +00001684 if (prom_strcmp(type, "memory"))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001685 continue;
Paul Mackerrasc49888202005-10-26 21:52:53 +10001686
Anton Blanchard5827d412012-11-26 17:40:03 +00001687 plen = prom_getprop(node, "reg", regbuf, sizeof(regbuf));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001688 if (plen > sizeof(regbuf)) {
1689 prom_printf("memory node too large for buffer !\n");
1690 plen = sizeof(regbuf);
1691 }
Anton Blanchard5827d412012-11-26 17:40:03 +00001692 p = regbuf;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001693 endp = p + (plen / sizeof(cell_t));
1694
1695#ifdef DEBUG_PROM
Mathieu Malaterrec806a6f2019-05-23 12:25:20 +02001696 memset(prom_scratch, 0, sizeof(prom_scratch));
1697 call_prom("package-to-path", 3, 1, node, prom_scratch,
1698 sizeof(prom_scratch) - 1);
1699 prom_debug(" node %s :\n", prom_scratch);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001700#endif /* DEBUG_PROM */
1701
1702 while ((endp - p) >= (rac + rsc)) {
1703 unsigned long base, size;
1704
1705 base = prom_next_cell(rac, &p);
1706 size = prom_next_cell(rsc, &p);
1707
1708 if (size == 0)
1709 continue;
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001710 prom_debug(" %lx %lx\n", base, size);
Anton Blanchard5827d412012-11-26 17:40:03 +00001711 if (base == 0 && (of_platform & PLATFORM_LPAR))
1712 rmo_top = size;
1713 if ((base + size) > ram_top)
1714 ram_top = base + size;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001715 }
1716 }
1717
Anton Blanchard5827d412012-11-26 17:40:03 +00001718 alloc_bottom = PAGE_ALIGN((unsigned long)&_end + 0x4000);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001719
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001720 /*
Benjamin Krillcf687872009-07-27 22:02:39 +00001721 * If prom_memory_limit is set we reduce the upper limits *except* for
1722 * alloc_top_high. This must be the real top of RAM so we can put
1723 * TCE's up there.
1724 */
1725
Anton Blanchard5827d412012-11-26 17:40:03 +00001726 alloc_top_high = ram_top;
Benjamin Krillcf687872009-07-27 22:02:39 +00001727
Anton Blanchard5827d412012-11-26 17:40:03 +00001728 if (prom_memory_limit) {
1729 if (prom_memory_limit <= alloc_bottom) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001730 prom_printf("Ignoring mem=%lx <= alloc_bottom.\n",
1731 prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00001732 prom_memory_limit = 0;
1733 } else if (prom_memory_limit >= ram_top) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001734 prom_printf("Ignoring mem=%lx >= ram_top.\n",
1735 prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00001736 prom_memory_limit = 0;
Benjamin Krillcf687872009-07-27 22:02:39 +00001737 } else {
Anton Blanchard5827d412012-11-26 17:40:03 +00001738 ram_top = prom_memory_limit;
1739 rmo_top = min(rmo_top, prom_memory_limit);
Benjamin Krillcf687872009-07-27 22:02:39 +00001740 }
1741 }
1742
1743 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001744 * Setup our top alloc point, that is top of RMO or top of
1745 * segment 0 when running non-LPAR.
1746 * Some RS64 machines have buggy firmware where claims up at
1747 * 1GB fail. Cap at 768MB as a workaround.
1748 * Since 768MB is plenty of room, and we need to cap to something
1749 * reasonable on 32-bit, cap at 768MB on all machines.
1750 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001751 if (!rmo_top)
1752 rmo_top = ram_top;
1753 rmo_top = min(0x30000000ul, rmo_top);
1754 alloc_top = rmo_top;
1755 alloc_top_high = ram_top;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001756
Paul Mackerras64968f62011-12-13 17:54:13 +00001757 /*
1758 * Check if we have an initrd after the kernel but still inside
1759 * the RMO. If we do move our bottom point to after it.
1760 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001761 if (prom_initrd_start &&
1762 prom_initrd_start < rmo_top &&
1763 prom_initrd_end > alloc_bottom)
1764 alloc_bottom = PAGE_ALIGN(prom_initrd_end);
Paul Mackerras64968f62011-12-13 17:54:13 +00001765
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001766 prom_printf("memory layout at init:\n");
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001767 prom_printf(" memory_limit : %lx (16 MB aligned)\n",
1768 prom_memory_limit);
1769 prom_printf(" alloc_bottom : %lx\n", alloc_bottom);
1770 prom_printf(" alloc_top : %lx\n", alloc_top);
1771 prom_printf(" alloc_top_hi : %lx\n", alloc_top_high);
1772 prom_printf(" rmo_top : %lx\n", rmo_top);
1773 prom_printf(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001774}
1775
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001776static void __init prom_close_stdin(void)
1777{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001778 __be32 val;
1779 ihandle stdin;
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001780
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001781 if (prom_getprop(prom.chosen, "stdin", &val, sizeof(val)) > 0) {
1782 stdin = be32_to_cpu(val);
1783 call_prom("close", 1, 0, stdin);
1784 }
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001785}
1786
Ram Pai6a9c9302019-08-19 23:13:14 -03001787#ifdef CONFIG_PPC_SVM
1788static int prom_rtas_hcall(uint64_t args)
1789{
1790 register uint64_t arg1 asm("r3") = H_RTAS;
1791 register uint64_t arg2 asm("r4") = args;
1792
1793 asm volatile("sc 1\n" : "=r" (arg1) :
1794 "r" (arg1),
1795 "r" (arg2) :);
Nicholas Piggin59dc5bf2021-06-18 01:51:03 +10001796 srr_regs_clobbered();
1797
Ram Pai6a9c9302019-08-19 23:13:14 -03001798 return arg1;
1799}
1800
1801static struct rtas_args __prombss os_term_args;
1802
1803static void __init prom_rtas_os_term(char *str)
1804{
1805 phandle rtas_node;
1806 __be32 val;
1807 u32 token;
1808
1809 prom_debug("%s: start...\n", __func__);
1810 rtas_node = call_prom("finddevice", 1, 1, ADDR("/rtas"));
1811 prom_debug("rtas_node: %x\n", rtas_node);
1812 if (!PHANDLE_VALID(rtas_node))
1813 return;
1814
1815 val = 0;
1816 prom_getprop(rtas_node, "ibm,os-term", &val, sizeof(val));
1817 token = be32_to_cpu(val);
1818 prom_debug("ibm,os-term: %x\n", token);
1819 if (token == 0)
1820 prom_panic("Could not get token for ibm,os-term\n");
1821 os_term_args.token = cpu_to_be32(token);
Alexey Kardashevskiy74bb84e2020-03-12 18:44:04 +11001822 os_term_args.nargs = cpu_to_be32(1);
1823 os_term_args.nret = cpu_to_be32(1);
1824 os_term_args.args[0] = cpu_to_be32(__pa(str));
Ram Pai6a9c9302019-08-19 23:13:14 -03001825 prom_rtas_hcall((uint64_t)&os_term_args);
1826}
1827#endif /* CONFIG_PPC_SVM */
1828
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001829/*
1830 * Allocate room for and instantiate RTAS
1831 */
1832static void __init prom_instantiate_rtas(void)
1833{
1834 phandle rtas_node;
1835 ihandle rtas_inst;
1836 u32 base, entry = 0;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001837 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001838 u32 size = 0;
1839
1840 prom_debug("prom_instantiate_rtas: start...\n");
1841
1842 rtas_node = call_prom("finddevice", 1, 1, ADDR("/rtas"));
1843 prom_debug("rtas_node: %x\n", rtas_node);
1844 if (!PHANDLE_VALID(rtas_node))
1845 return;
1846
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001847 val = 0;
1848 prom_getprop(rtas_node, "rtas-size", &val, sizeof(size));
1849 size = be32_to_cpu(val);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001850 if (size == 0)
1851 return;
1852
1853 base = alloc_down(size, PAGE_SIZE, 0);
Anton Blanchard6d1e2c62011-11-14 12:55:47 +00001854 if (base == 0)
1855 prom_panic("Could not allocate memory for RTAS\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001856
1857 rtas_inst = call_prom("open", 1, 1, ADDR("/rtas"));
1858 if (!IHANDLE_VALID(rtas_inst)) {
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001859 prom_printf("opening rtas package failed (%x)\n", rtas_inst);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001860 return;
1861 }
1862
Anton Blanchard1f8737a2009-03-31 20:06:15 +00001863 prom_printf("instantiating rtas at 0x%x...", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001864
1865 if (call_prom_ret("call-method", 3, 2, &entry,
1866 ADDR("instantiate-rtas"),
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001867 rtas_inst, base) != 0
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001868 || entry == 0) {
1869 prom_printf(" failed\n");
1870 return;
1871 }
1872 prom_printf(" done\n");
1873
1874 reserve_mem(base, size);
1875
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001876 val = cpu_to_be32(base);
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001877 prom_setprop(rtas_node, "/rtas", "linux,rtas-base",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001878 &val, sizeof(val));
1879 val = cpu_to_be32(entry);
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001880 prom_setprop(rtas_node, "/rtas", "linux,rtas-entry",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001881 &val, sizeof(val));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001882
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10001883 /* Check if it supports "query-cpu-stopped-state" */
1884 if (prom_getprop(rtas_node, "query-cpu-stopped-state",
1885 &val, sizeof(val)) != PROM_ERROR)
1886 rtas_has_query_cpu_stopped = true;
1887
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001888 prom_debug("rtas base = 0x%x\n", base);
1889 prom_debug("rtas entry = 0x%x\n", entry);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001890 prom_debug("rtas size = 0x%x\n", size);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001891
1892 prom_debug("prom_instantiate_rtas: end...\n");
1893}
1894
1895#ifdef CONFIG_PPC64
1896/*
Ashley Lai4a727422012-08-14 18:34:57 -05001897 * Allocate room for and instantiate Stored Measurement Log (SML)
1898 */
1899static void __init prom_instantiate_sml(void)
1900{
1901 phandle ibmvtpm_node;
1902 ihandle ibmvtpm_inst;
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001903 u32 entry = 0, size = 0, succ = 0;
Ashley Lai4a727422012-08-14 18:34:57 -05001904 u64 base;
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001905 __be32 val;
Ashley Lai4a727422012-08-14 18:34:57 -05001906
1907 prom_debug("prom_instantiate_sml: start...\n");
1908
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001909 ibmvtpm_node = call_prom("finddevice", 1, 1, ADDR("/vdevice/vtpm"));
Ashley Lai4a727422012-08-14 18:34:57 -05001910 prom_debug("ibmvtpm_node: %x\n", ibmvtpm_node);
1911 if (!PHANDLE_VALID(ibmvtpm_node))
1912 return;
1913
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001914 ibmvtpm_inst = call_prom("open", 1, 1, ADDR("/vdevice/vtpm"));
Ashley Lai4a727422012-08-14 18:34:57 -05001915 if (!IHANDLE_VALID(ibmvtpm_inst)) {
1916 prom_printf("opening vtpm package failed (%x)\n", ibmvtpm_inst);
1917 return;
1918 }
1919
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001920 if (prom_getprop(ibmvtpm_node, "ibm,sml-efi-reformat-supported",
1921 &val, sizeof(val)) != PROM_ERROR) {
1922 if (call_prom_ret("call-method", 2, 2, &succ,
1923 ADDR("reformat-sml-to-efi-alignment"),
1924 ibmvtpm_inst) != 0 || succ == 0) {
1925 prom_printf("Reformat SML to EFI alignment failed\n");
1926 return;
1927 }
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001928
Hon Ching \(Vicky\) Lo9e5d4af2015-10-07 20:11:54 -04001929 if (call_prom_ret("call-method", 2, 2, &size,
1930 ADDR("sml-get-allocated-size"),
1931 ibmvtpm_inst) != 0 || size == 0) {
1932 prom_printf("SML get allocated size failed\n");
1933 return;
1934 }
1935 } else {
1936 if (call_prom_ret("call-method", 2, 2, &size,
1937 ADDR("sml-get-handover-size"),
1938 ibmvtpm_inst) != 0 || size == 0) {
1939 prom_printf("SML get handover size failed\n");
1940 return;
1941 }
Ashley Lai4a727422012-08-14 18:34:57 -05001942 }
1943
1944 base = alloc_down(size, PAGE_SIZE, 0);
1945 if (base == 0)
1946 prom_panic("Could not allocate memory for sml\n");
1947
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001948 prom_printf("instantiating sml at 0x%llx...", base);
Ashley Lai4a727422012-08-14 18:34:57 -05001949
Hon Ching \(Vicky\) Lo9e5d4af2015-10-07 20:11:54 -04001950 memset((void *)base, 0, size);
1951
Ashley Lai4a727422012-08-14 18:34:57 -05001952 if (call_prom_ret("call-method", 4, 2, &entry,
1953 ADDR("sml-handover"),
1954 ibmvtpm_inst, size, base) != 0 || entry == 0) {
1955 prom_printf("SML handover failed\n");
1956 return;
1957 }
1958 prom_printf(" done\n");
1959
1960 reserve_mem(base, size);
1961
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001962 prom_setprop(ibmvtpm_node, "/vdevice/vtpm", "linux,sml-base",
Ashley Lai4a727422012-08-14 18:34:57 -05001963 &base, sizeof(base));
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001964 prom_setprop(ibmvtpm_node, "/vdevice/vtpm", "linux,sml-size",
Ashley Lai4a727422012-08-14 18:34:57 -05001965 &size, sizeof(size));
1966
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001967 prom_debug("sml base = 0x%llx\n", base);
1968 prom_debug("sml size = 0x%x\n", size);
Ashley Lai4a727422012-08-14 18:34:57 -05001969
1970 prom_debug("prom_instantiate_sml: end...\n");
1971}
1972
1973/*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001974 * Allocate room for and initialize TCE tables
1975 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001976#ifdef __BIG_ENDIAN__
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001977static void __init prom_initialize_tce_table(void)
1978{
1979 phandle node;
1980 ihandle phb_node;
1981 char compatible[64], type[64], model[64];
Anton Blanchard5827d412012-11-26 17:40:03 +00001982 char *path = prom_scratch;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001983 u64 base, align;
1984 u32 minalign, minsize;
1985 u64 tce_entry, *tce_entryp;
1986 u64 local_alloc_top, local_alloc_bottom;
1987 u64 i;
1988
Anton Blanchard5827d412012-11-26 17:40:03 +00001989 if (prom_iommu_off)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001990 return;
1991
1992 prom_debug("starting prom_initialize_tce_table\n");
1993
1994 /* Cache current top of allocs so we reserve a single block */
Anton Blanchard5827d412012-11-26 17:40:03 +00001995 local_alloc_top = alloc_top_high;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001996 local_alloc_bottom = local_alloc_top;
1997
1998 /* Search all nodes looking for PHBs. */
1999 for (node = 0; prom_next_node(&node); ) {
2000 compatible[0] = 0;
2001 type[0] = 0;
2002 model[0] = 0;
2003 prom_getprop(node, "compatible",
2004 compatible, sizeof(compatible));
2005 prom_getprop(node, "device_type", type, sizeof(type));
2006 prom_getprop(node, "model", model, sizeof(model));
2007
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002008 if ((type[0] == 0) || (prom_strstr(type, "pci") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002009 continue;
2010
Linas Vepstase788ff12007-09-07 03:45:21 +10002011 /* Keep the old logic intact to avoid regression. */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002012 if (compatible[0] != 0) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002013 if ((prom_strstr(compatible, "python") == NULL) &&
2014 (prom_strstr(compatible, "Speedwagon") == NULL) &&
2015 (prom_strstr(compatible, "Winnipeg") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002016 continue;
2017 } else if (model[0] != 0) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002018 if ((prom_strstr(model, "ython") == NULL) &&
2019 (prom_strstr(model, "peedwagon") == NULL) &&
2020 (prom_strstr(model, "innipeg") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002021 continue;
2022 }
2023
2024 if (prom_getprop(node, "tce-table-minalign", &minalign,
2025 sizeof(minalign)) == PROM_ERROR)
2026 minalign = 0;
2027 if (prom_getprop(node, "tce-table-minsize", &minsize,
2028 sizeof(minsize)) == PROM_ERROR)
2029 minsize = 4UL << 20;
2030
2031 /*
2032 * Even though we read what OF wants, we just set the table
2033 * size to 4 MB. This is enough to map 2GB of PCI DMA space.
2034 * By doing this, we avoid the pitfalls of trying to DMA to
2035 * MMIO space and the DMA alias hole.
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002036 */
Nicholas Piggin471d7ff2018-02-21 05:08:29 +10002037 minsize = 4UL << 20;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002038
2039 /* Align to the greater of the align or size */
2040 align = max(minalign, minsize);
2041 base = alloc_down(minsize, align, 1);
2042 if (base == 0)
2043 prom_panic("ERROR, cannot find space for TCE table.\n");
2044 if (base < local_alloc_bottom)
2045 local_alloc_bottom = base;
2046
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002047 /* It seems OF doesn't null-terminate the path :-( */
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002048 memset(path, 0, sizeof(prom_scratch));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002049 /* Call OF to setup the TCE hardware */
2050 if (call_prom("package-to-path", 3, 1, node,
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002051 path, sizeof(prom_scratch) - 1) == PROM_ERROR) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002052 prom_printf("package-to-path failed\n");
2053 }
2054
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002055 /* Save away the TCE table attributes for later use. */
2056 prom_setprop(node, path, "linux,tce-base", &base, sizeof(base));
2057 prom_setprop(node, path, "linux,tce-size", &minsize, sizeof(minsize));
2058
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002059 prom_debug("TCE table: %s\n", path);
2060 prom_debug("\tnode = 0x%x\n", node);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002061 prom_debug("\tbase = 0x%llx\n", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002062 prom_debug("\tsize = 0x%x\n", minsize);
2063
2064 /* Initialize the table to have a one-to-one mapping
2065 * over the allocated size.
2066 */
Ingo Molnar2b931fb2009-01-06 13:56:52 +00002067 tce_entryp = (u64 *)base;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002068 for (i = 0; i < (minsize >> 3) ;tce_entryp++, i++) {
2069 tce_entry = (i << PAGE_SHIFT);
2070 tce_entry |= 0x3;
2071 *tce_entryp = tce_entry;
2072 }
2073
2074 prom_printf("opening PHB %s", path);
2075 phb_node = call_prom("open", 1, 1, path);
2076 if (phb_node == 0)
2077 prom_printf("... failed\n");
2078 else
2079 prom_printf("... done\n");
2080
2081 call_prom("call-method", 6, 0, ADDR("set-64-bit-addressing"),
2082 phb_node, -1, minsize,
2083 (u32) base, (u32) (base >> 32));
2084 call_prom("close", 1, 0, phb_node);
2085 }
2086
2087 reserve_mem(local_alloc_bottom, local_alloc_top - local_alloc_bottom);
2088
Michael Ellerman2babf5c2006-05-17 18:00:46 +10002089 /* These are only really needed if there is a memory limit in
2090 * effect, but we don't know so export them always. */
Anton Blanchard5827d412012-11-26 17:40:03 +00002091 prom_tce_alloc_start = local_alloc_bottom;
2092 prom_tce_alloc_end = local_alloc_top;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002093
2094 /* Flag the first invalid entry */
2095 prom_debug("ending prom_initialize_tce_table\n");
2096}
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002097#endif /* __BIG_ENDIAN__ */
2098#endif /* CONFIG_PPC64 */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002099
2100/*
2101 * With CHRP SMP we need to use the OF to start the other processors.
2102 * We can't wait until smp_boot_cpus (the OF is trashed by then)
2103 * so we have to put the processors into a holding pattern controlled
2104 * by the kernel (not OF) before we destroy the OF.
2105 *
2106 * This uses a chunk of low memory, puts some holding pattern
2107 * code there and sends the other processors off to there until
2108 * smp_boot_cpus tells them to do something. The holding pattern
2109 * checks that address until its cpu # is there, when it is that
2110 * cpu jumps to __secondary_start(). smp_boot_cpus() takes care
2111 * of setting those values.
2112 *
2113 * We also use physical address 0x4 here to tell when a cpu
2114 * is in its holding pattern code.
2115 *
2116 * -- Cort
2117 */
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002118/*
2119 * We want to reference the copy of __secondary_hold_* in the
2120 * 0 - 0x100 address range
2121 */
2122#define LOW_ADDR(x) (((unsigned long) &(x)) & 0xff)
2123
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002124static void __init prom_hold_cpus(void)
2125{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002126 unsigned long i;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002127 phandle node;
2128 char type[64];
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002129 unsigned long *spinloop
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002130 = (void *) LOW_ADDR(__secondary_hold_spinloop);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002131 unsigned long *acknowledge
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002132 = (void *) LOW_ADDR(__secondary_hold_acknowledge);
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002133 unsigned long secondary_hold = LOW_ADDR(__secondary_hold);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002134
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10002135 /*
2136 * On pseries, if RTAS supports "query-cpu-stopped-state",
2137 * we skip this stage, the CPUs will be started by the
2138 * kernel using RTAS.
2139 */
2140 if ((of_platform == PLATFORM_PSERIES ||
2141 of_platform == PLATFORM_PSERIES_LPAR) &&
2142 rtas_has_query_cpu_stopped) {
2143 prom_printf("prom_hold_cpus: skipped\n");
2144 return;
2145 }
2146
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002147 prom_debug("prom_hold_cpus: start...\n");
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002148 prom_debug(" 1) spinloop = 0x%lx\n", (unsigned long)spinloop);
2149 prom_debug(" 1) *spinloop = 0x%lx\n", *spinloop);
2150 prom_debug(" 1) acknowledge = 0x%lx\n",
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002151 (unsigned long)acknowledge);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002152 prom_debug(" 1) *acknowledge = 0x%lx\n", *acknowledge);
2153 prom_debug(" 1) secondary_hold = 0x%lx\n", secondary_hold);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002154
2155 /* Set the common spinloop variable, so all of the secondary cpus
2156 * will block when they are awakened from their OF spinloop.
2157 * This must occur for both SMP and non SMP kernels, since OF will
2158 * be trashed when we move the kernel.
2159 */
2160 *spinloop = 0;
2161
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002162 /* look for cpus */
2163 for (node = 0; prom_next_node(&node); ) {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002164 unsigned int cpu_no;
2165 __be32 reg;
2166
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002167 type[0] = 0;
2168 prom_getprop(node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002169 if (prom_strcmp(type, "cpu") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002170 continue;
2171
2172 /* Skip non-configured cpus. */
2173 if (prom_getprop(node, "status", type, sizeof(type)) > 0)
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002174 if (prom_strcmp(type, "okay") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002175 continue;
2176
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002177 reg = cpu_to_be32(-1); /* make sparse happy */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002178 prom_getprop(node, "reg", &reg, sizeof(reg));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002179 cpu_no = be32_to_cpu(reg);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002180
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002181 prom_debug("cpu hw idx = %u\n", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002182
2183 /* Init the acknowledge var which will be reset by
2184 * the secondary cpu when it awakens from its OF
2185 * spinloop.
2186 */
2187 *acknowledge = (unsigned long)-1;
2188
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002189 if (cpu_no != prom.cpu) {
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00002190 /* Primary Thread of non-boot cpu or any thread */
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002191 prom_printf("starting cpu hw idx %u... ", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002192 call_prom("start-cpu", 3, 0, node,
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002193 secondary_hold, cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002194
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002195 for (i = 0; (i < 100000000) &&
2196 (*acknowledge == ((unsigned long)-1)); i++ )
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002197 mb();
2198
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002199 if (*acknowledge == cpu_no)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002200 prom_printf("done\n");
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002201 else
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002202 prom_printf("failed: %lx\n", *acknowledge);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002203 }
2204#ifdef CONFIG_SMP
2205 else
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002206 prom_printf("boot cpu hw idx %u\n", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002207#endif /* CONFIG_SMP */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002208 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002209
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002210 prom_debug("prom_hold_cpus: end...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002211}
2212
2213
2214static void __init prom_init_client_services(unsigned long pp)
2215{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002216 /* Get a handle to the prom entry point before anything else */
Anton Blanchard5827d412012-11-26 17:40:03 +00002217 prom_entry = pp;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002218
2219 /* get a handle for the stdout device */
Anton Blanchard5827d412012-11-26 17:40:03 +00002220 prom.chosen = call_prom("finddevice", 1, 1, ADDR("/chosen"));
2221 if (!PHANDLE_VALID(prom.chosen))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002222 prom_panic("cannot find chosen"); /* msg won't be printed :( */
2223
2224 /* get device tree root */
Anton Blanchard5827d412012-11-26 17:40:03 +00002225 prom.root = call_prom("finddevice", 1, 1, ADDR("/"));
2226 if (!PHANDLE_VALID(prom.root))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002227 prom_panic("cannot find device tree root"); /* msg won't be printed :( */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002228
Anton Blanchard5827d412012-11-26 17:40:03 +00002229 prom.mmumap = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002230}
2231
Paul Mackerrasa575b802005-10-23 17:23:21 +10002232#ifdef CONFIG_PPC32
2233/*
2234 * For really old powermacs, we need to map things we claim.
2235 * For that, we need the ihandle of the mmu.
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002236 * Also, on the longtrail, we need to work around other bugs.
Paul Mackerrasa575b802005-10-23 17:23:21 +10002237 */
2238static void __init prom_find_mmu(void)
2239{
Paul Mackerrasa575b802005-10-23 17:23:21 +10002240 phandle oprom;
2241 char version[64];
2242
2243 oprom = call_prom("finddevice", 1, 1, ADDR("/openprom"));
2244 if (!PHANDLE_VALID(oprom))
2245 return;
2246 if (prom_getprop(oprom, "model", version, sizeof(version)) <= 0)
2247 return;
2248 version[sizeof(version) - 1] = 0;
Paul Mackerrasa575b802005-10-23 17:23:21 +10002249 /* XXX might need to add other versions here */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002250 if (prom_strcmp(version, "Open Firmware, 1.0.5") == 0)
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002251 of_workarounds = OF_WA_CLAIM;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002252 else if (prom_strncmp(version, "FirmWorks,3.", 12) == 0) {
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002253 of_workarounds = OF_WA_CLAIM | OF_WA_LONGTRAIL;
2254 call_prom("interpret", 1, 1, "dev /memory 0 to allow-reclaim");
2255 } else
Paul Mackerrasa575b802005-10-23 17:23:21 +10002256 return;
Anton Blanchard5827d412012-11-26 17:40:03 +00002257 prom.memory = call_prom("open", 1, 1, ADDR("/memory"));
2258 prom_getprop(prom.chosen, "mmu", &prom.mmumap,
2259 sizeof(prom.mmumap));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002260 prom.mmumap = be32_to_cpu(prom.mmumap);
Anton Blanchard5827d412012-11-26 17:40:03 +00002261 if (!IHANDLE_VALID(prom.memory) || !IHANDLE_VALID(prom.mmumap))
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002262 of_workarounds &= ~OF_WA_CLAIM; /* hmmm */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002263}
2264#else
2265#define prom_find_mmu()
2266#endif
2267
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002268static void __init prom_init_stdout(void)
2269{
Anton Blanchard5827d412012-11-26 17:40:03 +00002270 char *path = of_stdout_device;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002271 char type[16];
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002272 phandle stdout_node;
2273 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002274
Anton Blanchard5827d412012-11-26 17:40:03 +00002275 if (prom_getprop(prom.chosen, "stdout", &val, sizeof(val)) <= 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002276 prom_panic("cannot find stdout");
2277
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002278 prom.stdout = be32_to_cpu(val);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002279
2280 /* Get the full OF pathname of the stdout device */
2281 memset(path, 0, 256);
Anton Blanchard5827d412012-11-26 17:40:03 +00002282 call_prom("instance-to-path", 3, 1, prom.stdout, path, 255);
Anton Blanchard5827d412012-11-26 17:40:03 +00002283 prom_printf("OF stdout device is: %s\n", of_stdout_device);
2284 prom_setprop(prom.chosen, "/chosen", "linux,stdout-path",
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002285 path, prom_strlen(path) + 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002286
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002287 /* instance-to-package fails on PA-Semi */
2288 stdout_node = call_prom("instance-to-package", 1, 1, prom.stdout);
2289 if (stdout_node != PROM_ERROR) {
2290 val = cpu_to_be32(stdout_node);
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002291
2292 /* If it's a display, note it */
2293 memset(type, 0, sizeof(type));
2294 prom_getprop(stdout_node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002295 if (prom_strcmp(type, "display") == 0)
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002296 prom_setprop(stdout_node, path, "linux,boot-display", NULL, 0);
2297 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002298}
2299
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002300static int __init prom_find_machine_type(void)
2301{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002302 char compat[256];
2303 int len, i = 0;
Benjamin Herrenschmidt21fe3302005-11-07 16:41:59 +11002304#ifdef CONFIG_PPC64
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002305 phandle rtas;
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002306 int x;
Benjamin Herrenschmidt21fe3302005-11-07 16:41:59 +11002307#endif
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002308
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002309 /* Look for a PowerMac or a Cell */
Anton Blanchard5827d412012-11-26 17:40:03 +00002310 len = prom_getprop(prom.root, "compatible",
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002311 compat, sizeof(compat)-1);
2312 if (len > 0) {
2313 compat[len] = 0;
2314 while (i < len) {
2315 char *p = &compat[i];
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002316 int sl = prom_strlen(p);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002317 if (sl == 0)
2318 break;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002319 if (prom_strstr(p, "Power Macintosh") ||
2320 prom_strstr(p, "MacRISC"))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002321 return PLATFORM_POWERMAC;
Arnd Bergmann133dda12006-06-07 12:04:18 +10002322#ifdef CONFIG_PPC64
2323 /* We must make sure we don't detect the IBM Cell
2324 * blades as pSeries due to some firmware issues,
2325 * so we do it here.
2326 */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002327 if (prom_strstr(p, "IBM,CBEA") ||
2328 prom_strstr(p, "IBM,CPBW-1.0"))
Arnd Bergmann133dda12006-06-07 12:04:18 +10002329 return PLATFORM_GENERIC;
2330#endif /* CONFIG_PPC64 */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002331 i += sl + 1;
2332 }
2333 }
2334#ifdef CONFIG_PPC64
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002335 /* Try to figure out if it's an IBM pSeries or any other
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002336 * PAPR compliant platform. We assume it is if :
2337 * - /device_type is "chrp" (please, do NOT use that for future
2338 * non-IBM designs !
2339 * - it has /rtas
2340 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002341 len = prom_getprop(prom.root, "device_type",
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002342 compat, sizeof(compat)-1);
2343 if (len <= 0)
2344 return PLATFORM_GENERIC;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002345 if (prom_strcmp(compat, "chrp"))
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002346 return PLATFORM_GENERIC;
2347
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002348 /* Default to pSeries. We need to know if we are running LPAR */
2349 rtas = call_prom("finddevice", 1, 1, ADDR("/rtas"));
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002350 if (!PHANDLE_VALID(rtas))
2351 return PLATFORM_GENERIC;
2352 x = prom_getproplen(rtas, "ibm,hypertas-functions");
2353 if (x != PROM_ERROR) {
Anton Blanchard4da727a2009-03-31 20:06:14 +00002354 prom_debug("Hypertas detected, assuming LPAR !\n");
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002355 return PLATFORM_PSERIES_LPAR;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002356 }
2357 return PLATFORM_PSERIES;
2358#else
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002359 return PLATFORM_GENERIC;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002360#endif
2361}
2362
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002363static int __init prom_set_color(ihandle ih, int i, int r, int g, int b)
2364{
2365 return call_prom("call-method", 6, 1, ADDR("color!"), ih, i, b, g, r);
2366}
2367
2368/*
2369 * If we have a display that we don't know how to drive,
2370 * we will want to try to execute OF's open method for it
2371 * later. However, OF will probably fall over if we do that
2372 * we've taken over the MMU.
2373 * So we check whether we will need to open the display,
2374 * and if so, open it now.
2375 */
2376static void __init prom_check_displays(void)
2377{
2378 char type[16], *path;
2379 phandle node;
2380 ihandle ih;
2381 int i;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002382
Benjamin Herrenschmidtd00e34b2018-10-15 13:49:56 +11002383 static const unsigned char default_colors[] __initconst = {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002384 0x00, 0x00, 0x00,
2385 0x00, 0x00, 0xaa,
2386 0x00, 0xaa, 0x00,
2387 0x00, 0xaa, 0xaa,
2388 0xaa, 0x00, 0x00,
2389 0xaa, 0x00, 0xaa,
2390 0xaa, 0xaa, 0x00,
2391 0xaa, 0xaa, 0xaa,
2392 0x55, 0x55, 0x55,
2393 0x55, 0x55, 0xff,
2394 0x55, 0xff, 0x55,
2395 0x55, 0xff, 0xff,
2396 0xff, 0x55, 0x55,
2397 0xff, 0x55, 0xff,
2398 0xff, 0xff, 0x55,
2399 0xff, 0xff, 0xff
2400 };
2401 const unsigned char *clut;
2402
Anton Blanchard4da727a2009-03-31 20:06:14 +00002403 prom_debug("Looking for displays\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002404 for (node = 0; prom_next_node(&node); ) {
2405 memset(type, 0, sizeof(type));
2406 prom_getprop(node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002407 if (prom_strcmp(type, "display") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002408 continue;
2409
2410 /* It seems OF doesn't null-terminate the path :-( */
Anton Blanchard5827d412012-11-26 17:40:03 +00002411 path = prom_scratch;
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002412 memset(path, 0, sizeof(prom_scratch));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002413
2414 /*
2415 * leave some room at the end of the path for appending extra
2416 * arguments
2417 */
2418 if (call_prom("package-to-path", 3, 1, node, path,
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002419 sizeof(prom_scratch) - 10) == PROM_ERROR)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002420 continue;
Anton Blanchard1f8737a2009-03-31 20:06:15 +00002421 prom_printf("found display : %s, opening... ", path);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002422
2423 ih = call_prom("open", 1, 1, path);
2424 if (ih == 0) {
2425 prom_printf("failed\n");
2426 continue;
2427 }
2428
2429 /* Success */
2430 prom_printf("done\n");
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002431 prom_setprop(node, path, "linux,opened", NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002432
2433 /* Setup a usable color table when the appropriate
2434 * method is available. Should update this to set-colors */
Anton Blanchard5827d412012-11-26 17:40:03 +00002435 clut = default_colors;
Benjamin Herrenschmidt3f536382011-12-14 13:55:11 +00002436 for (i = 0; i < 16; i++, clut += 3)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002437 if (prom_set_color(ih, i, clut[0], clut[1],
2438 clut[2]) != 0)
2439 break;
2440
2441#ifdef CONFIG_LOGO_LINUX_CLUT224
Anton Blanchard5827d412012-11-26 17:40:03 +00002442 clut = PTRRELOC(logo_linux_clut224.clut);
2443 for (i = 0; i < logo_linux_clut224.clutsize; i++, clut += 3)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002444 if (prom_set_color(ih, i + 32, clut[0], clut[1],
2445 clut[2]) != 0)
2446 break;
2447#endif /* CONFIG_LOGO_LINUX_CLUT224 */
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002448
2449#ifdef CONFIG_PPC_EARLY_DEBUG_BOOTX
2450 if (prom_getprop(node, "linux,boot-display", NULL, 0) !=
2451 PROM_ERROR) {
2452 u32 width, height, pitch, addr;
2453
2454 prom_printf("Setting btext !\n");
Michael Ellerman6c71cfc2020-08-21 20:34:07 +10002455
2456 if (prom_getprop(node, "width", &width, 4) == PROM_ERROR)
2457 return;
2458
2459 if (prom_getprop(node, "height", &height, 4) == PROM_ERROR)
2460 return;
2461
2462 if (prom_getprop(node, "linebytes", &pitch, 4) == PROM_ERROR)
2463 return;
2464
2465 if (prom_getprop(node, "address", &addr, 4) == PROM_ERROR)
2466 return;
2467
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002468 prom_printf("W=%d H=%d LB=%d addr=0x%x\n",
2469 width, height, pitch, addr);
2470 btext_setup_display(width, height, 8, pitch, addr);
Christophe Leroyc21f5a92019-06-03 13:00:51 +00002471 btext_prepare_BAT();
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002472 }
2473#endif /* CONFIG_PPC_EARLY_DEBUG_BOOTX */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002474 }
2475}
2476
2477
2478/* Return (relocated) pointer to this much memory: moves initrd if reqd. */
2479static void __init *make_room(unsigned long *mem_start, unsigned long *mem_end,
2480 unsigned long needed, unsigned long align)
2481{
2482 void *ret;
2483
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002484 *mem_start = ALIGN(*mem_start, align);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002485 while ((*mem_start + needed) > *mem_end) {
2486 unsigned long room, chunk;
2487
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002488 prom_debug("Chunk exhausted, claiming more at %lx...\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002489 alloc_bottom);
2490 room = alloc_top - alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002491 if (room > DEVTREE_CHUNK_SIZE)
2492 room = DEVTREE_CHUNK_SIZE;
2493 if (room < PAGE_SIZE)
Anton Blanchardfbafd722011-07-25 20:47:51 +00002494 prom_panic("No memory for flatten_device_tree "
2495 "(no room)\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002496 chunk = alloc_up(room, 0);
2497 if (chunk == 0)
Anton Blanchardfbafd722011-07-25 20:47:51 +00002498 prom_panic("No memory for flatten_device_tree "
2499 "(claim failed)\n");
Anton Blanchard966728d2011-07-25 20:47:07 +00002500 *mem_end = chunk + room;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002501 }
2502
2503 ret = (void *)*mem_start;
2504 *mem_start += needed;
2505
2506 return ret;
2507}
2508
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002509#define dt_push_token(token, mem_start, mem_end) do { \
2510 void *room = make_room(mem_start, mem_end, 4, 4); \
2511 *(__be32 *)room = cpu_to_be32(token); \
2512 } while(0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002513
2514static unsigned long __init dt_find_string(char *str)
2515{
2516 char *s, *os;
2517
Anton Blanchard5827d412012-11-26 17:40:03 +00002518 s = os = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002519 s += 4;
Anton Blanchard5827d412012-11-26 17:40:03 +00002520 while (s < (char *)dt_string_end) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002521 if (prom_strcmp(s, str) == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002522 return s - os;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002523 s += prom_strlen(s) + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002524 }
2525 return 0;
2526}
2527
2528/*
2529 * The Open Firmware 1275 specification states properties must be 31 bytes or
2530 * less, however not all firmwares obey this. Make it 64 bytes to be safe.
2531 */
2532#define MAX_PROPERTY_NAME 64
2533
2534static void __init scan_dt_build_strings(phandle node,
2535 unsigned long *mem_start,
2536 unsigned long *mem_end)
2537{
2538 char *prev_name, *namep, *sstart;
2539 unsigned long soff;
2540 phandle child;
2541
Anton Blanchard5827d412012-11-26 17:40:03 +00002542 sstart = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002543
2544 /* get and store all property names */
Anton Blanchard5827d412012-11-26 17:40:03 +00002545 prev_name = "";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002546 for (;;) {
2547 /* 64 is max len of name including nul. */
2548 namep = make_room(mem_start, mem_end, MAX_PROPERTY_NAME, 1);
2549 if (call_prom("nextprop", 3, 1, node, prev_name, namep) != 1) {
2550 /* No more nodes: unwind alloc */
2551 *mem_start = (unsigned long)namep;
2552 break;
2553 }
2554
2555 /* skip "name" */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002556 if (prom_strcmp(namep, "name") == 0) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002557 *mem_start = (unsigned long)namep;
Anton Blanchard5827d412012-11-26 17:40:03 +00002558 prev_name = "name";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002559 continue;
2560 }
2561 /* get/create string entry */
2562 soff = dt_find_string(namep);
2563 if (soff != 0) {
2564 *mem_start = (unsigned long)namep;
2565 namep = sstart + soff;
2566 } else {
2567 /* Trim off some if we can */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002568 *mem_start = (unsigned long)namep + prom_strlen(namep) + 1;
Anton Blanchard5827d412012-11-26 17:40:03 +00002569 dt_string_end = *mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002570 }
2571 prev_name = namep;
2572 }
2573
2574 /* do all our children */
2575 child = call_prom("child", 1, 1, node);
2576 while (child != 0) {
2577 scan_dt_build_strings(child, mem_start, mem_end);
2578 child = call_prom("peer", 1, 1, child);
2579 }
2580}
2581
2582static void __init scan_dt_build_struct(phandle node, unsigned long *mem_start,
2583 unsigned long *mem_end)
2584{
2585 phandle child;
2586 char *namep, *prev_name, *sstart, *p, *ep, *lp, *path;
2587 unsigned long soff;
2588 unsigned char *valp;
Benjamin Herrenschmidt8ca2d512018-10-15 13:49:57 +11002589 static char pname[MAX_PROPERTY_NAME] __prombss;
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002590 int l, room, has_phandle = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002591
2592 dt_push_token(OF_DT_BEGIN_NODE, mem_start, mem_end);
2593
2594 /* get the node's full name */
2595 namep = (char *)*mem_start;
Paul Mackerrasc49888202005-10-26 21:52:53 +10002596 room = *mem_end - *mem_start;
2597 if (room > 255)
2598 room = 255;
2599 l = call_prom("package-to-path", 3, 1, node, namep, room);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002600 if (l >= 0) {
2601 /* Didn't fit? Get more room. */
Paul Mackerrasc49888202005-10-26 21:52:53 +10002602 if (l >= room) {
2603 if (l >= *mem_end - *mem_start)
2604 namep = make_room(mem_start, mem_end, l+1, 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002605 call_prom("package-to-path", 3, 1, node, namep, l);
2606 }
2607 namep[l] = '\0';
2608
2609 /* Fixup an Apple bug where they have bogus \0 chars in the
Paul Mackerrasa575b802005-10-23 17:23:21 +10002610 * middle of the path in some properties, and extract
2611 * the unit name (everything after the last '/').
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002612 */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002613 for (lp = p = namep, ep = namep + l; p < ep; p++) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002614 if (*p == '/')
Paul Mackerrasa575b802005-10-23 17:23:21 +10002615 lp = namep;
2616 else if (*p != 0)
2617 *lp++ = *p;
2618 }
2619 *lp = 0;
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002620 *mem_start = ALIGN((unsigned long)lp + 1, 4);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002621 }
2622
2623 /* get it again for debugging */
Anton Blanchard5827d412012-11-26 17:40:03 +00002624 path = prom_scratch;
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002625 memset(path, 0, sizeof(prom_scratch));
2626 call_prom("package-to-path", 3, 1, node, path, sizeof(prom_scratch) - 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002627
2628 /* get and store all properties */
Anton Blanchard5827d412012-11-26 17:40:03 +00002629 prev_name = "";
2630 sstart = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002631 for (;;) {
2632 if (call_prom("nextprop", 3, 1, node, prev_name,
Anton Blanchard5827d412012-11-26 17:40:03 +00002633 pname) != 1)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002634 break;
2635
2636 /* skip "name" */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002637 if (prom_strcmp(pname, "name") == 0) {
Anton Blanchard5827d412012-11-26 17:40:03 +00002638 prev_name = "name";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002639 continue;
2640 }
2641
2642 /* find string offset */
Anton Blanchard5827d412012-11-26 17:40:03 +00002643 soff = dt_find_string(pname);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002644 if (soff == 0) {
2645 prom_printf("WARNING: Can't find string index for"
Anton Blanchard5827d412012-11-26 17:40:03 +00002646 " <%s>, node %s\n", pname, path);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002647 break;
2648 }
2649 prev_name = sstart + soff;
2650
2651 /* get length */
Anton Blanchard5827d412012-11-26 17:40:03 +00002652 l = call_prom("getproplen", 2, 1, node, pname);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002653
2654 /* sanity checks */
2655 if (l == PROM_ERROR)
2656 continue;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002657
2658 /* push property head */
2659 dt_push_token(OF_DT_PROP, mem_start, mem_end);
2660 dt_push_token(l, mem_start, mem_end);
2661 dt_push_token(soff, mem_start, mem_end);
2662
2663 /* push property content */
2664 valp = make_room(mem_start, mem_end, l, 4);
Anton Blanchard5827d412012-11-26 17:40:03 +00002665 call_prom("getprop", 4, 1, node, pname, valp, l);
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002666 *mem_start = ALIGN(*mem_start, 4);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002667
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002668 if (!prom_strcmp(pname, "phandle"))
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002669 has_phandle = 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002670 }
2671
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002672 /* Add a "phandle" property if none already exist */
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002673 if (!has_phandle) {
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002674 soff = dt_find_string("phandle");
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002675 if (soff == 0)
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002676 prom_printf("WARNING: Can't find string index for <phandle> node %s\n", path);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002677 else {
2678 dt_push_token(OF_DT_PROP, mem_start, mem_end);
2679 dt_push_token(4, mem_start, mem_end);
2680 dt_push_token(soff, mem_start, mem_end);
2681 valp = make_room(mem_start, mem_end, 4, 4);
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002682 *(__be32 *)valp = cpu_to_be32(node);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002683 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002684 }
2685
2686 /* do all our children */
2687 child = call_prom("child", 1, 1, node);
2688 while (child != 0) {
2689 scan_dt_build_struct(child, mem_start, mem_end);
2690 child = call_prom("peer", 1, 1, child);
2691 }
2692
2693 dt_push_token(OF_DT_END_NODE, mem_start, mem_end);
2694}
2695
2696static void __init flatten_device_tree(void)
2697{
2698 phandle root;
2699 unsigned long mem_start, mem_end, room;
2700 struct boot_param_header *hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002701 char *namep;
2702 u64 *rsvmap;
2703
2704 /*
2705 * Check how much room we have between alloc top & bottom (+/- a
Anton Blanchardfbafd722011-07-25 20:47:51 +00002706 * few pages), crop to 1MB, as this is our "chunk" size
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002707 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002708 room = alloc_top - alloc_bottom - 0x4000;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002709 if (room > DEVTREE_CHUNK_SIZE)
2710 room = DEVTREE_CHUNK_SIZE;
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002711 prom_debug("starting device tree allocs at %lx\n", alloc_bottom);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002712
2713 /* Now try to claim that */
2714 mem_start = (unsigned long)alloc_up(room, PAGE_SIZE);
2715 if (mem_start == 0)
2716 prom_panic("Can't allocate initial device-tree chunk\n");
Anton Blanchard966728d2011-07-25 20:47:07 +00002717 mem_end = mem_start + room;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002718
2719 /* Get root of tree */
2720 root = call_prom("peer", 1, 1, (phandle)0);
2721 if (root == (phandle)0)
2722 prom_panic ("couldn't get device tree root\n");
2723
2724 /* Build header and make room for mem rsv map */
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002725 mem_start = ALIGN(mem_start, 4);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002726 hdr = make_room(&mem_start, &mem_end,
2727 sizeof(struct boot_param_header), 4);
Anton Blanchard5827d412012-11-26 17:40:03 +00002728 dt_header_start = (unsigned long)hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002729 rsvmap = make_room(&mem_start, &mem_end, sizeof(mem_reserve_map), 8);
2730
2731 /* Start of strings */
2732 mem_start = PAGE_ALIGN(mem_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00002733 dt_string_start = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002734 mem_start += 4; /* hole */
2735
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002736 /* Add "phandle" in there, we'll need it */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002737 namep = make_room(&mem_start, &mem_end, 16, 1);
Michael Ellermanf47d5a42021-06-21 16:49:37 +10002738 prom_strscpy_pad(namep, "phandle", sizeof("phandle"));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002739 mem_start = (unsigned long)namep + prom_strlen(namep) + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002740
2741 /* Build string array */
2742 prom_printf("Building dt strings...\n");
2743 scan_dt_build_strings(root, &mem_start, &mem_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00002744 dt_string_end = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002745
2746 /* Build structure */
2747 mem_start = PAGE_ALIGN(mem_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00002748 dt_struct_start = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002749 prom_printf("Building dt structure...\n");
2750 scan_dt_build_struct(root, &mem_start, &mem_end);
2751 dt_push_token(OF_DT_END, &mem_start, &mem_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00002752 dt_struct_end = PAGE_ALIGN(mem_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002753
2754 /* Finish header */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002755 hdr->boot_cpuid_phys = cpu_to_be32(prom.cpu);
2756 hdr->magic = cpu_to_be32(OF_DT_HEADER);
2757 hdr->totalsize = cpu_to_be32(dt_struct_end - dt_header_start);
2758 hdr->off_dt_struct = cpu_to_be32(dt_struct_start - dt_header_start);
2759 hdr->off_dt_strings = cpu_to_be32(dt_string_start - dt_header_start);
2760 hdr->dt_strings_size = cpu_to_be32(dt_string_end - dt_string_start);
2761 hdr->off_mem_rsvmap = cpu_to_be32(((unsigned long)rsvmap) - dt_header_start);
2762 hdr->version = cpu_to_be32(OF_DT_VERSION);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002763 /* Version 16 is not backward compatible */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002764 hdr->last_comp_version = cpu_to_be32(0x10);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002765
Jimi Xenidis4d1f3f22006-05-18 17:03:05 -05002766 /* Copy the reserve map in */
Anton Blanchard5827d412012-11-26 17:40:03 +00002767 memcpy(rsvmap, mem_reserve_map, sizeof(mem_reserve_map));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002768
2769#ifdef DEBUG_PROM
2770 {
2771 int i;
2772 prom_printf("reserved memory map:\n");
Anton Blanchard5827d412012-11-26 17:40:03 +00002773 for (i = 0; i < mem_reserve_cnt; i++)
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002774 prom_printf(" %llx - %llx\n",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002775 be64_to_cpu(mem_reserve_map[i].base),
2776 be64_to_cpu(mem_reserve_map[i].size));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002777 }
2778#endif
Jimi Xenidis4d1f3f22006-05-18 17:03:05 -05002779 /* Bump mem_reserve_cnt to cause further reservations to fail
2780 * since it's too late.
2781 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002782 mem_reserve_cnt = MEM_RESERVE_MAP_SIZE;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002783
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002784 prom_printf("Device tree strings 0x%lx -> 0x%lx\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002785 dt_string_start, dt_string_end);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002786 prom_printf("Device tree struct 0x%lx -> 0x%lx\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002787 dt_struct_start, dt_struct_end);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002788}
2789
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002790#ifdef CONFIG_PPC_MAPLE
2791/* PIBS Version 1.05.0000 04/26/2005 has an incorrect /ht/isa/ranges property.
2792 * The values are bad, and it doesn't even have the right number of cells. */
2793static void __init fixup_device_tree_maple(void)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002794{
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002795 phandle isa;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002796 u32 rloc = 0x01002000; /* IO space; PCI device = 4 */
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002797 u32 isa_ranges[6];
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002798 char *name;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002799
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002800 name = "/ht@0/isa@4";
2801 isa = call_prom("finddevice", 1, 1, ADDR(name));
2802 if (!PHANDLE_VALID(isa)) {
2803 name = "/ht@0/isa@6";
2804 isa = call_prom("finddevice", 1, 1, ADDR(name));
2805 rloc = 0x01003000; /* IO space; PCI device = 6 */
2806 }
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002807 if (!PHANDLE_VALID(isa))
2808 return;
2809
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002810 if (prom_getproplen(isa, "ranges") != 12)
2811 return;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002812 if (prom_getprop(isa, "ranges", isa_ranges, sizeof(isa_ranges))
2813 == PROM_ERROR)
2814 return;
2815
2816 if (isa_ranges[0] != 0x1 ||
2817 isa_ranges[1] != 0xf4000000 ||
2818 isa_ranges[2] != 0x00010000)
2819 return;
2820
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002821 prom_printf("Fixing up bogus ISA range on Maple/Apache...\n");
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002822
2823 isa_ranges[0] = 0x1;
2824 isa_ranges[1] = 0x0;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002825 isa_ranges[2] = rloc;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002826 isa_ranges[3] = 0x0;
2827 isa_ranges[4] = 0x0;
2828 isa_ranges[5] = 0x00010000;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002829 prom_setprop(isa, name, "ranges",
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002830 isa_ranges, sizeof(isa_ranges));
2831}
Harry Ciao8f101a052009-06-17 16:28:00 -07002832
2833#define CPC925_MC_START 0xf8000000
2834#define CPC925_MC_LENGTH 0x1000000
2835/* The values for memory-controller don't have right number of cells */
2836static void __init fixup_device_tree_maple_memory_controller(void)
2837{
2838 phandle mc;
2839 u32 mc_reg[4];
2840 char *name = "/hostbridge@f8000000";
Harry Ciao8f101a052009-06-17 16:28:00 -07002841 u32 ac, sc;
2842
2843 mc = call_prom("finddevice", 1, 1, ADDR(name));
2844 if (!PHANDLE_VALID(mc))
2845 return;
2846
2847 if (prom_getproplen(mc, "reg") != 8)
2848 return;
2849
Anton Blanchard5827d412012-11-26 17:40:03 +00002850 prom_getprop(prom.root, "#address-cells", &ac, sizeof(ac));
2851 prom_getprop(prom.root, "#size-cells", &sc, sizeof(sc));
Harry Ciao8f101a052009-06-17 16:28:00 -07002852 if ((ac != 2) || (sc != 2))
2853 return;
2854
2855 if (prom_getprop(mc, "reg", mc_reg, sizeof(mc_reg)) == PROM_ERROR)
2856 return;
2857
2858 if (mc_reg[0] != CPC925_MC_START || mc_reg[1] != CPC925_MC_LENGTH)
2859 return;
2860
2861 prom_printf("Fixing up bogus hostbridge on Maple...\n");
2862
2863 mc_reg[0] = 0x0;
2864 mc_reg[1] = CPC925_MC_START;
2865 mc_reg[2] = 0x0;
2866 mc_reg[3] = CPC925_MC_LENGTH;
2867 prom_setprop(mc, name, "reg", mc_reg, sizeof(mc_reg));
2868}
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002869#else
2870#define fixup_device_tree_maple()
Harry Ciao8f101a052009-06-17 16:28:00 -07002871#define fixup_device_tree_maple_memory_controller()
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002872#endif
2873
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002874#ifdef CONFIG_PPC_CHRP
Olaf Heringe4805922007-04-04 18:20:04 +02002875/*
2876 * Pegasos and BriQ lacks the "ranges" property in the isa node
2877 * Pegasos needs decimal IRQ 14/15, not hexadecimal
Olaf Hering556ecf92007-08-18 04:27:17 +10002878 * Pegasos has the IDE configured in legacy mode, but advertised as native
Olaf Heringe4805922007-04-04 18:20:04 +02002879 */
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002880static void __init fixup_device_tree_chrp(void)
2881{
Olaf Heringe4805922007-04-04 18:20:04 +02002882 phandle ph;
2883 u32 prop[6];
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002884 u32 rloc = 0x01006000; /* IO space; PCI device = 12 */
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002885 char *name;
2886 int rc;
2887
2888 name = "/pci@80000000/isa@c";
Olaf Heringe4805922007-04-04 18:20:04 +02002889 ph = call_prom("finddevice", 1, 1, ADDR(name));
2890 if (!PHANDLE_VALID(ph)) {
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002891 name = "/pci@ff500000/isa@6";
Olaf Heringe4805922007-04-04 18:20:04 +02002892 ph = call_prom("finddevice", 1, 1, ADDR(name));
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002893 rloc = 0x01003000; /* IO space; PCI device = 6 */
2894 }
Olaf Heringe4805922007-04-04 18:20:04 +02002895 if (PHANDLE_VALID(ph)) {
2896 rc = prom_getproplen(ph, "ranges");
2897 if (rc == 0 || rc == PROM_ERROR) {
2898 prom_printf("Fixing up missing ISA range on Pegasos...\n");
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002899
Olaf Heringe4805922007-04-04 18:20:04 +02002900 prop[0] = 0x1;
2901 prop[1] = 0x0;
2902 prop[2] = rloc;
2903 prop[3] = 0x0;
2904 prop[4] = 0x0;
2905 prop[5] = 0x00010000;
2906 prom_setprop(ph, name, "ranges", prop, sizeof(prop));
2907 }
2908 }
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002909
Olaf Heringe4805922007-04-04 18:20:04 +02002910 name = "/pci@80000000/ide@C,1";
2911 ph = call_prom("finddevice", 1, 1, ADDR(name));
2912 if (PHANDLE_VALID(ph)) {
2913 prom_printf("Fixing up IDE interrupt on Pegasos...\n");
2914 prop[0] = 14;
2915 prop[1] = 0x0;
Olaf Hering556ecf92007-08-18 04:27:17 +10002916 prom_setprop(ph, name, "interrupts", prop, 2*sizeof(u32));
2917 prom_printf("Fixing up IDE class-code on Pegasos...\n");
2918 rc = prom_getprop(ph, "class-code", prop, sizeof(u32));
2919 if (rc == sizeof(u32)) {
2920 prop[0] &= ~0x5;
2921 prom_setprop(ph, name, "class-code", prop, sizeof(u32));
2922 }
Olaf Heringe4805922007-04-04 18:20:04 +02002923 }
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002924}
2925#else
2926#define fixup_device_tree_chrp()
2927#endif
2928
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002929#if defined(CONFIG_PPC64) && defined(CONFIG_PPC_PMAC)
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002930static void __init fixup_device_tree_pmac(void)
2931{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002932 phandle u3, i2c, mpic;
2933 u32 u3_rev;
2934 u32 interrupts[2];
2935 u32 parent;
2936
2937 /* Some G5s have a missing interrupt definition, fix it up here */
2938 u3 = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000"));
2939 if (!PHANDLE_VALID(u3))
2940 return;
2941 i2c = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000/i2c@f8001000"));
2942 if (!PHANDLE_VALID(i2c))
2943 return;
2944 mpic = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000/mpic@f8040000"));
2945 if (!PHANDLE_VALID(mpic))
2946 return;
2947
2948 /* check if proper rev of u3 */
2949 if (prom_getprop(u3, "device-rev", &u3_rev, sizeof(u3_rev))
2950 == PROM_ERROR)
2951 return;
Benjamin Herrenschmidt7d496972005-11-07 14:36:21 +11002952 if (u3_rev < 0x35 || u3_rev > 0x39)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002953 return;
2954 /* does it need fixup ? */
2955 if (prom_getproplen(i2c, "interrupts") > 0)
2956 return;
2957
2958 prom_printf("fixing up bogus interrupts for u3 i2c...\n");
2959
2960 /* interrupt on this revision of u3 is number 0 and level */
2961 interrupts[0] = 0;
2962 interrupts[1] = 1;
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002963 prom_setprop(i2c, "/u3@0,f8000000/i2c@f8001000", "interrupts",
2964 &interrupts, sizeof(interrupts));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002965 parent = (u32)mpic;
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002966 prom_setprop(i2c, "/u3@0,f8000000/i2c@f8001000", "interrupt-parent",
2967 &parent, sizeof(parent));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002968}
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002969#else
2970#define fixup_device_tree_pmac()
2971#endif
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002972
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002973#ifdef CONFIG_PPC_EFIKA
Grant Likely94d2dde2008-01-24 22:25:32 -07002974/*
2975 * The MPC5200 FEC driver requires an phy-handle property to tell it how
2976 * to talk to the phy. If the phy-handle property is missing, then this
2977 * function is called to add the appropriate nodes and link it to the
2978 * ethernet node.
2979 */
2980static void __init fixup_device_tree_efika_add_phy(void)
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002981{
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002982 u32 node;
2983 char prop[64];
Grant Likely94d2dde2008-01-24 22:25:32 -07002984 int rv;
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002985
Grant Likely94d2dde2008-01-24 22:25:32 -07002986 /* Check if /builtin/ethernet exists - bail if it doesn't */
2987 node = call_prom("finddevice", 1, 1, ADDR("/builtin/ethernet"));
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002988 if (!PHANDLE_VALID(node))
2989 return;
2990
Grant Likely94d2dde2008-01-24 22:25:32 -07002991 /* Check if the phy-handle property exists - bail if it does */
2992 rv = prom_getprop(node, "phy-handle", prop, sizeof(prop));
2993 if (!rv)
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002994 return;
2995
Grant Likely94d2dde2008-01-24 22:25:32 -07002996 /*
2997 * At this point the ethernet device doesn't have a phy described.
2998 * Now we need to add the missing phy node and linkage
2999 */
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003000
Grant Likely94d2dde2008-01-24 22:25:32 -07003001 /* Check for an MDIO bus node - if missing then create one */
Olaf Hering6f4347c2008-01-10 01:06:08 +11003002 node = call_prom("finddevice", 1, 1, ADDR("/builtin/mdio"));
3003 if (!PHANDLE_VALID(node)) {
3004 prom_printf("Adding Ethernet MDIO node\n");
3005 call_prom("interpret", 1, 1,
3006 " s\" /builtin\" find-device"
3007 " new-device"
3008 " 1 encode-int s\" #address-cells\" property"
3009 " 0 encode-int s\" #size-cells\" property"
Grant Likely94d2dde2008-01-24 22:25:32 -07003010 " s\" mdio\" device-name"
3011 " s\" fsl,mpc5200b-mdio\" encode-string"
Olaf Hering6f4347c2008-01-10 01:06:08 +11003012 " s\" compatible\" property"
3013 " 0xf0003000 0x400 reg"
3014 " 0x2 encode-int"
3015 " 0x5 encode-int encode+"
3016 " 0x3 encode-int encode+"
3017 " s\" interrupts\" property"
3018 " finish-device");
Jiapeng Chong4f46d572021-02-24 15:29:21 +08003019 }
Olaf Hering6f4347c2008-01-10 01:06:08 +11003020
Grant Likely94d2dde2008-01-24 22:25:32 -07003021 /* Check for a PHY device node - if missing then create one and
3022 * give it's phandle to the ethernet node */
3023 node = call_prom("finddevice", 1, 1,
3024 ADDR("/builtin/mdio/ethernet-phy"));
Olaf Hering6f4347c2008-01-10 01:06:08 +11003025 if (!PHANDLE_VALID(node)) {
3026 prom_printf("Adding Ethernet PHY node\n");
3027 call_prom("interpret", 1, 1,
3028 " s\" /builtin/mdio\" find-device"
3029 " new-device"
3030 " s\" ethernet-phy\" device-name"
3031 " 0x10 encode-int s\" reg\" property"
3032 " my-self"
3033 " ihandle>phandle"
3034 " finish-device"
3035 " s\" /builtin/ethernet\" find-device"
3036 " encode-int"
3037 " s\" phy-handle\" property"
3038 " device-end");
3039 }
Grant Likely94d2dde2008-01-24 22:25:32 -07003040}
Olaf Hering6f4347c2008-01-10 01:06:08 +11003041
Grant Likely94d2dde2008-01-24 22:25:32 -07003042static void __init fixup_device_tree_efika(void)
3043{
3044 int sound_irq[3] = { 2, 2, 0 };
3045 int bcomm_irq[3*16] = { 3,0,0, 3,1,0, 3,2,0, 3,3,0,
3046 3,4,0, 3,5,0, 3,6,0, 3,7,0,
3047 3,8,0, 3,9,0, 3,10,0, 3,11,0,
3048 3,12,0, 3,13,0, 3,14,0, 3,15,0 };
3049 u32 node;
3050 char prop[64];
3051 int rv, len;
3052
3053 /* Check if we're really running on a EFIKA */
3054 node = call_prom("finddevice", 1, 1, ADDR("/"));
3055 if (!PHANDLE_VALID(node))
3056 return;
3057
3058 rv = prom_getprop(node, "model", prop, sizeof(prop));
3059 if (rv == PROM_ERROR)
3060 return;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003061 if (prom_strcmp(prop, "EFIKA5K2"))
Grant Likely94d2dde2008-01-24 22:25:32 -07003062 return;
3063
3064 prom_printf("Applying EFIKA device tree fixups\n");
3065
3066 /* Claiming to be 'chrp' is death */
3067 node = call_prom("finddevice", 1, 1, ADDR("/"));
3068 rv = prom_getprop(node, "device_type", prop, sizeof(prop));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003069 if (rv != PROM_ERROR && (prom_strcmp(prop, "chrp") == 0))
Grant Likely94d2dde2008-01-24 22:25:32 -07003070 prom_setprop(node, "/", "device_type", "efika", sizeof("efika"));
3071
David Woodhouse7f4392c2008-04-14 02:52:38 +10003072 /* CODEGEN,description is exposed in /proc/cpuinfo so
3073 fix that too */
3074 rv = prom_getprop(node, "CODEGEN,description", prop, sizeof(prop));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003075 if (rv != PROM_ERROR && (prom_strstr(prop, "CHRP")))
David Woodhouse7f4392c2008-04-14 02:52:38 +10003076 prom_setprop(node, "/", "CODEGEN,description",
3077 "Efika 5200B PowerPC System",
3078 sizeof("Efika 5200B PowerPC System"));
3079
Grant Likely94d2dde2008-01-24 22:25:32 -07003080 /* Fixup bestcomm interrupts property */
3081 node = call_prom("finddevice", 1, 1, ADDR("/builtin/bestcomm"));
3082 if (PHANDLE_VALID(node)) {
3083 len = prom_getproplen(node, "interrupts");
3084 if (len == 12) {
3085 prom_printf("Fixing bestcomm interrupts property\n");
3086 prom_setprop(node, "/builtin/bestcom", "interrupts",
3087 bcomm_irq, sizeof(bcomm_irq));
3088 }
3089 }
3090
3091 /* Fixup sound interrupts property */
3092 node = call_prom("finddevice", 1, 1, ADDR("/builtin/sound"));
3093 if (PHANDLE_VALID(node)) {
3094 rv = prom_getprop(node, "interrupts", prop, sizeof(prop));
3095 if (rv == PROM_ERROR) {
3096 prom_printf("Adding sound interrupts property\n");
3097 prom_setprop(node, "/builtin/sound", "interrupts",
3098 sound_irq, sizeof(sound_irq));
3099 }
3100 }
3101
3102 /* Make sure ethernet phy-handle property exists */
3103 fixup_device_tree_efika_add_phy();
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003104}
3105#else
3106#define fixup_device_tree_efika()
3107#endif
3108
Darren Stevens50246782016-08-31 13:24:40 +01003109#ifdef CONFIG_PPC_PASEMI_NEMO
3110/*
3111 * CFE supplied on Nemo is broken in several ways, biggest
3112 * problem is that it reassigns ISA interrupts to unused mpic ints.
3113 * Add an interrupt-controller property for the io-bridge to use
3114 * and correct the ints so we can attach them to an irq_domain
3115 */
3116static void __init fixup_device_tree_pasemi(void)
3117{
3118 u32 interrupts[2], parent, rval, val = 0;
3119 char *name, *pci_name;
3120 phandle iob, node;
3121
3122 /* Find the root pci node */
3123 name = "/pxp@0,e0000000";
3124 iob = call_prom("finddevice", 1, 1, ADDR(name));
3125 if (!PHANDLE_VALID(iob))
3126 return;
3127
3128 /* check if interrupt-controller node set yet */
3129 if (prom_getproplen(iob, "interrupt-controller") !=PROM_ERROR)
3130 return;
3131
3132 prom_printf("adding interrupt-controller property for SB600...\n");
3133
3134 prom_setprop(iob, name, "interrupt-controller", &val, 0);
3135
3136 pci_name = "/pxp@0,e0000000/pci@11";
3137 node = call_prom("finddevice", 1, 1, ADDR(pci_name));
3138 parent = ADDR(iob);
3139
3140 for( ; prom_next_node(&node); ) {
3141 /* scan each node for one with an interrupt */
3142 if (!PHANDLE_VALID(node))
3143 continue;
3144
3145 rval = prom_getproplen(node, "interrupts");
3146 if (rval == 0 || rval == PROM_ERROR)
3147 continue;
3148
3149 prom_getprop(node, "interrupts", &interrupts, sizeof(interrupts));
3150 if ((interrupts[0] < 212) || (interrupts[0] > 222))
3151 continue;
3152
3153 /* found a node, update both interrupts and interrupt-parent */
3154 if ((interrupts[0] >= 212) && (interrupts[0] <= 215))
3155 interrupts[0] -= 203;
3156 if ((interrupts[0] >= 216) && (interrupts[0] <= 220))
3157 interrupts[0] -= 213;
3158 if (interrupts[0] == 221)
3159 interrupts[0] = 14;
3160 if (interrupts[0] == 222)
3161 interrupts[0] = 8;
3162
3163 prom_setprop(node, pci_name, "interrupts", interrupts,
3164 sizeof(interrupts));
3165 prom_setprop(node, pci_name, "interrupt-parent", &parent,
3166 sizeof(parent));
3167 }
Darren Stevens687e16b2016-08-31 13:24:45 +01003168
3169 /*
3170 * The io-bridge has device_type set to 'io-bridge' change it to 'isa'
3171 * so that generic isa-bridge code can add the SB600 and its on-board
3172 * peripherals.
3173 */
3174 name = "/pxp@0,e0000000/io-bridge@0";
3175 iob = call_prom("finddevice", 1, 1, ADDR(name));
3176 if (!PHANDLE_VALID(iob))
3177 return;
3178
3179 /* device_type is already set, just change it. */
3180
3181 prom_printf("Changing device_type of SB600 node...\n");
3182
3183 prom_setprop(iob, name, "device_type", "isa", sizeof("isa"));
Darren Stevens50246782016-08-31 13:24:40 +01003184}
3185#else /* !CONFIG_PPC_PASEMI_NEMO */
3186static inline void fixup_device_tree_pasemi(void) { }
3187#endif
3188
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003189static void __init fixup_device_tree(void)
3190{
3191 fixup_device_tree_maple();
Harry Ciao8f101a052009-06-17 16:28:00 -07003192 fixup_device_tree_maple_memory_controller();
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10003193 fixup_device_tree_chrp();
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003194 fixup_device_tree_pmac();
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003195 fixup_device_tree_efika();
Darren Stevens50246782016-08-31 13:24:40 +01003196 fixup_device_tree_pasemi();
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003197}
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003198
3199static void __init prom_find_boot_cpu(void)
3200{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003201 __be32 rval;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003202 ihandle prom_cpu;
3203 phandle cpu_pkg;
3204
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003205 rval = 0;
3206 if (prom_getprop(prom.chosen, "cpu", &rval, sizeof(rval)) <= 0)
Paul Mackerrasa575b802005-10-23 17:23:21 +10003207 return;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003208 prom_cpu = be32_to_cpu(rval);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003209
3210 cpu_pkg = call_prom("instance-to-package", 1, 1, prom_cpu);
3211
Darren Stevensaf2b7fa2017-01-23 19:42:54 +00003212 if (!PHANDLE_VALID(cpu_pkg))
3213 return;
3214
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003215 prom_getprop(cpu_pkg, "reg", &rval, sizeof(rval));
3216 prom.cpu = be32_to_cpu(rval);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003217
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02003218 prom_debug("Booting CPU hw index = %d\n", prom.cpu);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003219}
3220
3221static void __init prom_check_initrd(unsigned long r3, unsigned long r4)
3222{
3223#ifdef CONFIG_BLK_DEV_INITRD
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003224 if (r3 && r4 && r4 != 0xdeadbeef) {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003225 __be64 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003226
Anton Blanchard5827d412012-11-26 17:40:03 +00003227 prom_initrd_start = is_kernel_addr(r3) ? __pa(r3) : r3;
3228 prom_initrd_end = prom_initrd_start + r4;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003229
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003230 val = cpu_to_be64(prom_initrd_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00003231 prom_setprop(prom.chosen, "/chosen", "linux,initrd-start",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003232 &val, sizeof(val));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003233 val = cpu_to_be64(prom_initrd_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00003234 prom_setprop(prom.chosen, "/chosen", "linux,initrd-end",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003235 &val, sizeof(val));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003236
Anton Blanchard5827d412012-11-26 17:40:03 +00003237 reserve_mem(prom_initrd_start,
3238 prom_initrd_end - prom_initrd_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003239
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02003240 prom_debug("initrd_start=0x%lx\n", prom_initrd_start);
3241 prom_debug("initrd_end=0x%lx\n", prom_initrd_end);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003242 }
3243#endif /* CONFIG_BLK_DEV_INITRD */
3244}
3245
Ram Pai6a9c9302019-08-19 23:13:14 -03003246#ifdef CONFIG_PPC_SVM
3247/*
3248 * Perform the Enter Secure Mode ultracall.
3249 */
3250static int enter_secure_mode(unsigned long kbase, unsigned long fdt)
3251{
3252 register unsigned long r3 asm("r3") = UV_ESM;
3253 register unsigned long r4 asm("r4") = kbase;
3254 register unsigned long r5 asm("r5") = fdt;
3255
3256 asm volatile("sc 2" : "+r"(r3) : "r"(r4), "r"(r5));
3257
3258 return r3;
3259}
3260
3261/*
3262 * Call the Ultravisor to transfer us to secure memory if we have an ESM blob.
3263 */
Vladis Dronovaff77952020-07-29 15:37:41 +02003264static void __init setup_secure_guest(unsigned long kbase, unsigned long fdt)
Ram Pai6a9c9302019-08-19 23:13:14 -03003265{
3266 int ret;
3267
3268 if (!prom_svm_enable)
3269 return;
3270
3271 /* Switch to secure mode. */
3272 prom_printf("Switching to secure mode.\n");
3273
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003274 /*
3275 * The ultravisor will do an integrity check of the kernel image but we
3276 * relocated it so the check will fail. Restore the original image by
3277 * relocating it back to the kernel virtual base address.
3278 */
Michael Ellerman24d33ac2021-06-23 23:04:54 +10003279 relocate(KERNELBASE);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003280
Ram Pai6a9c9302019-08-19 23:13:14 -03003281 ret = enter_secure_mode(kbase, fdt);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003282
3283 /* Relocate the kernel again. */
Michael Ellerman24d33ac2021-06-23 23:04:54 +10003284 relocate(kbase);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003285
Ram Pai6a9c9302019-08-19 23:13:14 -03003286 if (ret != U_SUCCESS) {
3287 prom_printf("Returned %d from switching to secure mode.\n", ret);
3288 prom_rtas_os_term("Switch to secure mode failed.\n");
3289 }
3290}
3291#else
Vladis Dronovaff77952020-07-29 15:37:41 +02003292static void __init setup_secure_guest(unsigned long kbase, unsigned long fdt)
Ram Pai6a9c9302019-08-19 23:13:14 -03003293{
3294}
3295#endif /* CONFIG_PPC_SVM */
3296
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003297/*
3298 * We enter here early on, when the Open Firmware prom is still
3299 * handling exceptions and the MMU hash table for us.
3300 */
3301
3302unsigned long __init prom_init(unsigned long r3, unsigned long r4,
3303 unsigned long pp,
Paul Mackerras549e8152008-08-30 11:43:47 +10003304 unsigned long r6, unsigned long r7,
3305 unsigned long kbase)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003306{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003307 unsigned long hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003308
3309#ifdef CONFIG_PPC32
Paul Mackerras549e8152008-08-30 11:43:47 +10003310 unsigned long offset = reloc_offset();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003311 reloc_got2(offset);
3312#endif
3313
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003314 /*
3315 * First zero the BSS
3316 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003317 memset(&__bss_start, 0, __bss_stop - __bss_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003318
3319 /*
3320 * Init interface to Open Firmware, get some node references,
3321 * like /chosen
3322 */
3323 prom_init_client_services(pp);
3324
3325 /*
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003326 * See if this OF is old enough that we need to do explicit maps
3327 * and other workarounds
3328 */
3329 prom_find_mmu();
3330
3331 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003332 * Init prom stdout device
3333 */
3334 prom_init_stdout();
3335
Anton Blanchard5827d412012-11-26 17:40:03 +00003336 prom_printf("Preparing to boot %s", linux_banner);
Michael Ellermane7943fb2009-03-04 19:02:01 +00003337
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003338 /*
3339 * Get default machine type. At this point, we do not differentiate
3340 * between pSeries SMP and pSeries LPAR
3341 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003342 of_platform = prom_find_machine_type();
3343 prom_printf("Detected machine type: %x\n", of_platform);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003344
Suzuki Poulose0f890c82011-12-14 22:57:15 +00003345#ifndef CONFIG_NONSTATIC_KERNEL
Olaf Heringadd60ef2006-03-23 22:03:57 +01003346 /* Bail if this is a kdump kernel. */
3347 if (PHYSICAL_START > 0)
3348 prom_panic("Error: You can't boot a kdump kernel from OF!\n");
Paul Mackerras549e8152008-08-30 11:43:47 +10003349#endif
Olaf Heringadd60ef2006-03-23 22:03:57 +01003350
3351 /*
3352 * Check for an initrd
3353 */
3354 prom_check_initrd(r3, r4);
3355
Suraj Jitindar Singh12cc9fd2017-02-28 17:03:47 +11003356 /*
3357 * Do early parsing of command line
3358 */
3359 early_cmdline_parse();
3360
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003361#ifdef CONFIG_PPC_PSERIES
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003362 /*
3363 * On pSeries, inform the firmware about our capabilities
3364 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003365 if (of_platform == PLATFORM_PSERIES ||
3366 of_platform == PLATFORM_PSERIES_LPAR)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003367 prom_send_capabilities();
3368#endif
3369
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003370 /*
Arnd Bergmannf3f66f52005-10-31 20:08:37 -05003371 * Copy the CPU hold code
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003372 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003373 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras549e8152008-08-30 11:43:47 +10003374 copy_and_flush(0, kbase, 0x100, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003375
3376 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003377 * Initialize memory management within prom_init
3378 */
3379 prom_init_mem();
3380
3381 /*
3382 * Determine which cpu is actually running right _now_
3383 */
3384 prom_find_boot_cpu();
3385
3386 /*
3387 * Initialize display devices
3388 */
3389 prom_check_displays();
3390
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003391#if defined(CONFIG_PPC64) && defined(__BIG_ENDIAN__)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003392 /*
3393 * Initialize IOMMU (TCE tables) on pSeries. Do that before anything else
3394 * that uses the allocator, we need to make sure we get the top of memory
3395 * available for us here...
3396 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003397 if (of_platform == PLATFORM_PSERIES)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003398 prom_initialize_tce_table();
3399#endif
3400
3401 /*
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003402 * On non-powermacs, try to instantiate RTAS. PowerMacs don't
3403 * have a usable RTAS implementation.
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003404 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003405 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003406 prom_instantiate_rtas();
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003407
Ashley Lai4a727422012-08-14 18:34:57 -05003408#ifdef CONFIG_PPC64
3409 /* instantiate sml */
3410 prom_instantiate_sml();
3411#endif
3412
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003413 /*
3414 * On non-powermacs, put all CPUs in spin-loops.
3415 *
3416 * PowerMacs use a different mechanism to spin CPUs
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10003417 *
3418 * (This must be done after instanciating RTAS)
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003419 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003420 if (of_platform != PLATFORM_POWERMAC)
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003421 prom_hold_cpus();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003422
3423 /*
3424 * Fill in some infos for use by the kernel later on
3425 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003426 if (prom_memory_limit) {
3427 __be64 val = cpu_to_be64(prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00003428 prom_setprop(prom.chosen, "/chosen", "linux,memory-limit",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003429 &val, sizeof(val));
3430 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003431#ifdef CONFIG_PPC64
Anton Blanchard5827d412012-11-26 17:40:03 +00003432 if (prom_iommu_off)
3433 prom_setprop(prom.chosen, "/chosen", "linux,iommu-off",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003434 NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003435
Anton Blanchard5827d412012-11-26 17:40:03 +00003436 if (prom_iommu_force_on)
3437 prom_setprop(prom.chosen, "/chosen", "linux,iommu-force-on",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003438 NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003439
Anton Blanchard5827d412012-11-26 17:40:03 +00003440 if (prom_tce_alloc_start) {
3441 prom_setprop(prom.chosen, "/chosen", "linux,tce-alloc-start",
3442 &prom_tce_alloc_start,
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003443 sizeof(prom_tce_alloc_start));
Anton Blanchard5827d412012-11-26 17:40:03 +00003444 prom_setprop(prom.chosen, "/chosen", "linux,tce-alloc-end",
3445 &prom_tce_alloc_end,
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003446 sizeof(prom_tce_alloc_end));
3447 }
3448#endif
3449
3450 /*
3451 * Fixup any known bugs in the device-tree
3452 */
3453 fixup_device_tree();
3454
3455 /*
3456 * Now finally create the flattened device-tree
3457 */
Anton Blanchard1f8737a2009-03-31 20:06:15 +00003458 prom_printf("copying OF device tree...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003459 flatten_device_tree();
3460
Paul Mackerras3825ac02005-11-08 22:48:08 +11003461 /*
3462 * in case stdin is USB and still active on IBM machines...
3463 * Unfortunately quiesce crashes on some powermacs if we have
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003464 * closed stdin already (in particular the powerbook 101).
Paul Mackerras3825ac02005-11-08 22:48:08 +11003465 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003466 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras3825ac02005-11-08 22:48:08 +11003467 prom_close_stdin();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003468
3469 /*
3470 * Call OF "quiesce" method to shut down pending DMA's from
3471 * devices etc...
3472 */
Michael Ellerman7e862d72015-03-30 17:38:09 +11003473 prom_printf("Quiescing Open Firmware ...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003474 call_prom("quiesce", 0, 0);
3475
3476 /*
3477 * And finally, call the kernel passing it the flattened device
3478 * tree and NULL as r5, thus triggering the new entry point which
3479 * is common to us and kexec
3480 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003481 hdr = dt_header_start;
Benjamin Herrenschmidt40dfef62011-11-29 18:22:56 +00003482
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003483 prom_printf("Booting Linux via __start() @ 0x%lx ...\n", kbase);
3484 prom_debug("->dt_header_start=0x%lx\n", hdr);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003485
3486#ifdef CONFIG_PPC32
3487 reloc_got2(-offset);
3488#endif
3489
Ram Pai6a9c9302019-08-19 23:13:14 -03003490 /* Move to secure memory if we're supposed to be secure guests. */
3491 setup_secure_guest(kbase, hdr);
3492
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00003493 __start(hdr, kbase, 0, 0, 0, 0, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003494
3495 return 0;
3496}