blob: 0ac5faacc909cac6497a7aa14c113bd9dbc7037b [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
Alexey Dobriyanc0891ac2021-08-02 23:40:32 +030017#include <linux/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
Nick Childd2769602021-12-16 17:00:16 -0500675static void __init 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
Nick Childd2769602021-12-16 17:00:16 -0500685static char *__init tohex(unsigned int x)
Paul Mackerrasa23414b2005-11-10 12:00:55 +1100686{
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
Nick Childd2769602021-12-16 17:00:16 -0500731static unsigned long __init 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
Nick Childd2769602021-12-16 17:00:16 -0500756static unsigned long __init 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
Aneesh Kumar K.V1c6b5a72021-08-12 18:52:23 +05301099 .associativity = OV5_FEAT(OV5_FORM1_AFFINITY) | OV5_FEAT(OV5_PRRN) |
1100 OV5_FEAT(OV5_FORM2_AFFINITY),
Michael Roth3dbbaf22017-02-20 19:12:18 -06001101 .bin_opts = OV5_FEAT(OV5_RESIZE_HPT) | OV5_FEAT(OV5_HP_EVT),
Michael Ellerman76ffb572016-11-18 23:15:42 +11001102 .micro_checkpoint = 0,
1103 .reserved0 = 0,
1104 .max_cpus = cpu_to_be32(NR_CPUS), /* number of cores supported */
1105 .papr_level = 0,
1106 .reserved1 = 0,
1107 .platform_facilities = OV5_FEAT(OV5_PFO_HW_RNG) | OV5_FEAT(OV5_PFO_HW_ENCR) | OV5_FEAT(OV5_PFO_HW_842),
1108 .reserved2 = 0,
1109 .reserved3 = 0,
1110 .subprocessors = 1,
Tyrel Datwyler0a87ccd2019-11-10 23:21:37 -06001111 .byte22 = OV5_FEAT(OV5_DRMEM_V2) | OV5_FEAT(OV5_DRC_INFO),
Paul Mackerrascc3d2942017-01-30 21:21:36 +11001112 .intarch = 0,
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001113 .mmu = 0,
1114 .hash_ext = 0,
1115 .radix_ext = 0,
Michael Ellerman76ffb572016-11-18 23:15:42 +11001116 },
1117
1118 /* option vector 6: IBM PAPR hints */
1119 .vec6_len = VECTOR_LENGTH(sizeof(struct option_vector6)),
1120 .vec6 = {
1121 .reserved = 0,
1122 .secondary_pteg = 0,
1123 .os_name = OV6_LINUX,
1124 },
Michael Ellermanffaacd92021-06-21 16:49:38 +10001125
1126 /* option vector 7: OS Identification */
1127 .vec7_len = VECTOR_LENGTH(sizeof(struct option_vector7)),
Michael Ellerman76ffb572016-11-18 23:15:42 +11001128};
1129
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001130static struct ibm_arch_vec __prombss ibm_architecture_vec ____cacheline_aligned;
1131
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001132/* Old method - ELF header with PT_NOTE sections only works on BE */
1133#ifdef __BIG_ENDIAN__
Benjamin Herrenschmidt30c69ca2018-05-31 14:33:40 +10001134static const struct fake_elf {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001135 Elf32_Ehdr elfhdr;
1136 Elf32_Phdr phdr[2];
1137 struct chrpnote {
1138 u32 namesz;
1139 u32 descsz;
1140 u32 type;
1141 char name[8]; /* "PowerPC" */
1142 struct chrpdesc {
1143 u32 real_mode;
1144 u32 real_base;
1145 u32 real_size;
1146 u32 virt_base;
1147 u32 virt_size;
1148 u32 load_base;
1149 } chrpdesc;
1150 } chrpnote;
1151 struct rpanote {
1152 u32 namesz;
1153 u32 descsz;
1154 u32 type;
1155 char name[24]; /* "IBM,RPA-Client-Config" */
1156 struct rpadesc {
1157 u32 lpar_affinity;
1158 u32 min_rmo_size;
1159 u32 min_rmo_percent;
1160 u32 max_pft_size;
1161 u32 splpar;
1162 u32 min_load;
1163 u32 new_mem_def;
1164 u32 ignore_me;
1165 } rpadesc;
1166 } rpanote;
Benjamin Herrenschmidtd00e34b2018-10-15 13:49:56 +11001167} fake_elf __initconst = {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001168 .elfhdr = {
1169 .e_ident = { 0x7f, 'E', 'L', 'F',
1170 ELFCLASS32, ELFDATA2MSB, EV_CURRENT },
1171 .e_type = ET_EXEC, /* yeah right */
1172 .e_machine = EM_PPC,
1173 .e_version = EV_CURRENT,
1174 .e_phoff = offsetof(struct fake_elf, phdr),
1175 .e_phentsize = sizeof(Elf32_Phdr),
1176 .e_phnum = 2
1177 },
1178 .phdr = {
1179 [0] = {
1180 .p_type = PT_NOTE,
1181 .p_offset = offsetof(struct fake_elf, chrpnote),
1182 .p_filesz = sizeof(struct chrpnote)
1183 }, [1] = {
1184 .p_type = PT_NOTE,
1185 .p_offset = offsetof(struct fake_elf, rpanote),
1186 .p_filesz = sizeof(struct rpanote)
1187 }
1188 },
1189 .chrpnote = {
1190 .namesz = sizeof("PowerPC"),
1191 .descsz = sizeof(struct chrpdesc),
1192 .type = 0x1275,
1193 .name = "PowerPC",
1194 .chrpdesc = {
1195 .real_mode = ~0U, /* ~0 means "don't care" */
1196 .real_base = ~0U,
1197 .real_size = ~0U,
1198 .virt_base = ~0U,
1199 .virt_size = ~0U,
1200 .load_base = ~0U
1201 },
1202 },
1203 .rpanote = {
1204 .namesz = sizeof("IBM,RPA-Client-Config"),
1205 .descsz = sizeof(struct rpadesc),
1206 .type = 0x12759999,
1207 .name = "IBM,RPA-Client-Config",
1208 .rpadesc = {
Paul Mackerras5663a122008-10-31 22:27:17 +11001209 .lpar_affinity = 0,
1210 .min_rmo_size = 64, /* in megabytes */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001211 .min_rmo_percent = 0,
Paul Mackerras5663a122008-10-31 22:27:17 +11001212 .max_pft_size = 48, /* 2^48 bytes max PFT size */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001213 .splpar = 1,
1214 .min_load = ~0U,
Paul Mackerras5663a122008-10-31 22:27:17 +11001215 .new_mem_def = 0
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001216 }
1217 }
1218};
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001219#endif /* __BIG_ENDIAN__ */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001220
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001221static int __init prom_count_smt_threads(void)
1222{
1223 phandle node;
1224 char type[64];
1225 unsigned int plen;
1226
1227 /* Pick up th first CPU node we can find */
1228 for (node = 0; prom_next_node(&node); ) {
1229 type[0] = 0;
1230 prom_getprop(node, "device_type", type, sizeof(type));
1231
Christophe Leroy450e7dd2019-04-26 16:23:28 +00001232 if (prom_strcmp(type, "cpu"))
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001233 continue;
1234 /*
1235 * There is an entry for each smt thread, each entry being
1236 * 4 bytes long. All cpus should have the same number of
1237 * smt threads, so return after finding the first.
1238 */
1239 plen = prom_getproplen(node, "ibm,ppc-interrupt-server#s");
1240 if (plen == PROM_ERROR)
1241 break;
1242 plen >>= 2;
Michael Neuling2c48a7d2010-07-27 18:26:21 +00001243 prom_debug("Found %lu smt threads per core\n", (unsigned long)plen);
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001244
1245 /* Sanity check */
1246 if (plen < 1 || plen > 64) {
Michael Neuling2c48a7d2010-07-27 18:26:21 +00001247 prom_printf("Threads per core %lu out of bounds, assuming 1\n",
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001248 (unsigned long)plen);
1249 return 1;
1250 }
1251 return plen;
1252 }
1253 prom_debug("No threads found, assuming 1 per core\n");
1254
1255 return 1;
1256
1257}
1258
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001259static void __init prom_parse_mmu_model(u8 val,
1260 struct platform_support *support)
1261{
1262 switch (val) {
1263 case OV5_FEAT(OV5_MMU_DYNAMIC):
1264 case OV5_FEAT(OV5_MMU_EITHER): /* Either Available */
1265 prom_debug("MMU - either supported\n");
1266 support->radix_mmu = !prom_radix_disable;
1267 support->hash_mmu = true;
1268 break;
1269 case OV5_FEAT(OV5_MMU_RADIX): /* Only Radix */
1270 prom_debug("MMU - radix only\n");
1271 if (prom_radix_disable) {
1272 /*
1273 * If we __have__ to do radix, we're better off ignoring
1274 * the command line rather than not booting.
1275 */
1276 prom_printf("WARNING: Ignoring cmdline option disable_radix\n");
1277 }
1278 support->radix_mmu = true;
1279 break;
1280 case OV5_FEAT(OV5_MMU_HASH):
1281 prom_debug("MMU - hash only\n");
1282 support->hash_mmu = true;
1283 break;
1284 default:
1285 prom_debug("Unknown mmu support option: 0x%x\n", val);
1286 break;
1287 }
1288}
1289
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001290static void __init prom_parse_xive_model(u8 val,
1291 struct platform_support *support)
1292{
1293 switch (val) {
1294 case OV5_FEAT(OV5_XIVE_EITHER): /* Either Available */
1295 prom_debug("XIVE - either mode supported\n");
Greg Kurza3bf9fb2019-05-15 12:05:01 +02001296 support->xive = !prom_xive_disable;
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001297 break;
1298 case OV5_FEAT(OV5_XIVE_EXPLOIT): /* Only Exploitation mode */
1299 prom_debug("XIVE - exploitation mode supported\n");
Greg Kurza3bf9fb2019-05-15 12:05:01 +02001300 if (prom_xive_disable) {
1301 /*
1302 * If we __have__ to do XIVE, we're better off ignoring
1303 * the command line rather than not booting.
1304 */
1305 prom_printf("WARNING: Ignoring cmdline option xive=off\n");
1306 }
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001307 support->xive = true;
1308 break;
1309 case OV5_FEAT(OV5_XIVE_LEGACY): /* Only Legacy mode */
1310 prom_debug("XIVE - legacy mode supported\n");
1311 break;
1312 default:
1313 prom_debug("Unknown xive support option: 0x%x\n", val);
1314 break;
1315 }
1316}
1317
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001318static void __init prom_parse_platform_support(u8 index, u8 val,
1319 struct platform_support *support)
1320{
1321 switch (index) {
1322 case OV5_INDX(OV5_MMU_SUPPORT): /* MMU Model */
1323 prom_parse_mmu_model(val & OV5_FEAT(OV5_MMU_SUPPORT), support);
1324 break;
1325 case OV5_INDX(OV5_RADIX_GTSE): /* Radix Extensions */
Aneesh Kumar K.Vbf6b7662020-07-27 14:29:08 +05301326 if (val & OV5_FEAT(OV5_RADIX_GTSE))
1327 support->radix_gtse = !prom_radix_gtse_disable;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001328 break;
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001329 case OV5_INDX(OV5_XIVE_SUPPORT): /* Interrupt mode */
1330 prom_parse_xive_model(val & OV5_FEAT(OV5_XIVE_SUPPORT),
1331 support);
1332 break;
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001333 }
1334}
1335
1336static void __init prom_check_platform_support(void)
1337{
1338 struct platform_support supported = {
1339 .hash_mmu = false,
1340 .radix_mmu = false,
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001341 .radix_gtse = false,
1342 .xive = false
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001343 };
1344 int prop_len = prom_getproplen(prom.chosen,
1345 "ibm,arch-vec-5-platform-support");
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001346
Christophe Leroyadcf5912019-04-26 16:23:29 +00001347 /*
1348 * First copy the architecture vec template
1349 *
1350 * use memcpy() instead of *vec = *vec_template so that GCC replaces it
1351 * by __memcpy() when KASAN is active
1352 */
1353 memcpy(&ibm_architecture_vec, &ibm_architecture_vec_template,
1354 sizeof(ibm_architecture_vec));
Benjamin Herrenschmidta614f522018-10-15 13:49:55 +11001355
Michael Ellermanffaacd92021-06-21 16:49:38 +10001356 prom_strscpy_pad(ibm_architecture_vec.vec7.os_id, linux_banner, 256);
1357
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001358 if (prop_len > 1) {
1359 int i;
Suraj Jitindar Singhab912392018-09-05 12:09:50 +10001360 u8 vec[8];
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001361 prom_debug("Found ibm,arch-vec-5-platform-support, len: %d\n",
1362 prop_len);
Suraj Jitindar Singhab912392018-09-05 12:09:50 +10001363 if (prop_len > sizeof(vec))
1364 prom_printf("WARNING: ibm,arch-vec-5-platform-support longer than expected (len: %d)\n",
1365 prop_len);
Cédric Le Goatered5b00a2021-01-22 08:50:29 +01001366 prom_getprop(prom.chosen, "ibm,arch-vec-5-platform-support", &vec, sizeof(vec));
1367 for (i = 0; i < prop_len; i += 2) {
1368 prom_debug("%d: index = 0x%x val = 0x%x\n", i / 2, vec[i], vec[i + 1]);
1369 prom_parse_platform_support(vec[i], vec[i + 1], &supported);
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001370 }
1371 }
1372
Bharata B Rao029ab302020-07-03 11:06:06 +05301373 if (supported.radix_mmu && IS_ENABLED(CONFIG_PPC_RADIX_MMU)) {
1374 /* Radix preferred - Check if GTSE is also supported */
1375 prom_debug("Asking for radix\n");
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001376 ibm_architecture_vec.vec5.mmu = OV5_FEAT(OV5_MMU_RADIX);
Bharata B Rao029ab302020-07-03 11:06:06 +05301377 if (supported.radix_gtse)
1378 ibm_architecture_vec.vec5.radix_ext =
1379 OV5_FEAT(OV5_RADIX_GTSE);
1380 else
1381 prom_debug("Radix GTSE isn't supported\n");
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001382 } else if (supported.hash_mmu) {
1383 /* Default to hash mmu (if we can) */
1384 prom_debug("Asking for hash\n");
1385 ibm_architecture_vec.vec5.mmu = OV5_FEAT(OV5_MMU_HASH);
1386 } else {
1387 /* We're probably on a legacy hypervisor */
1388 prom_debug("Assuming legacy hash support\n");
1389 }
Cédric Le Goaterac5e5a52017-08-30 21:46:16 +02001390
1391 if (supported.xive) {
1392 prom_debug("Asking for XIVE\n");
1393 ibm_architecture_vec.vec5.intarch = OV5_FEAT(OV5_XIVE_EXPLOIT);
1394 }
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001395}
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001396
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001397static void __init prom_send_capabilities(void)
1398{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001399 ihandle root;
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001400 prom_arg_t ret;
Laurent Dufourdbd0c5d2013-09-17 11:52:48 +02001401 u32 cores;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001402
Suraj Jitindar Singh014d02c2017-02-28 17:03:48 +11001403 /* Check ibm,arch-vec-5-platform-support and fixup vec5 if required */
1404 prom_check_platform_support();
1405
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001406 root = call_prom("open", 1, 1, ADDR("/"));
1407 if (root != 0) {
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001408 /* We need to tell the FW about the number of cores we support.
1409 *
1410 * To do that, we count the number of threads on the first core
1411 * (we assume this is the same for all cores) and use it to
1412 * divide NR_CPUS.
1413 */
Laurent Dufourdbd0c5d2013-09-17 11:52:48 +02001414
Michael Ellerman76ffb572016-11-18 23:15:42 +11001415 cores = DIV_ROUND_UP(NR_CPUS, prom_count_smt_threads());
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001416 prom_printf("Max number of cores passed to firmware: %u (NR_CPUS = %d)\n",
Michael Ellerman76ffb572016-11-18 23:15:42 +11001417 cores, NR_CPUS);
1418
1419 ibm_architecture_vec.vec5.max_cpus = cpu_to_be32(cores);
Benjamin Herrenschmidtefec9592010-02-04 14:33:54 +11001420
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001421 /* try calling the ibm,client-architecture-support method */
Anton Blanchard049d0492009-09-21 20:47:39 +00001422 prom_printf("Calling ibm,client-architecture-support...");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001423 if (call_prom_ret("call-method", 3, 2, &ret,
1424 ADDR("ibm,client-architecture-support"),
Benjamin Herrenschmidt33b74972006-06-07 12:01:32 +10001425 root,
Michael Ellerman76ffb572016-11-18 23:15:42 +11001426 ADDR(&ibm_architecture_vec)) == 0) {
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001427 /* the call exists... */
1428 if (ret)
Anton Blanchard4da727a2009-03-31 20:06:14 +00001429 prom_printf("\nWARNING: ibm,client-architecture"
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001430 "-support call FAILED!\n");
1431 call_prom("close", 1, 0, root);
Anton Blanchard4da727a2009-03-31 20:06:14 +00001432 prom_printf(" done\n");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001433 return;
1434 }
1435 call_prom("close", 1, 0, root);
Anton Blanchard049d0492009-09-21 20:47:39 +00001436 prom_printf(" not implemented\n");
Paul Mackerrasf709bfa2006-04-28 16:28:35 +10001437 }
1438
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001439#ifdef __BIG_ENDIAN__
1440 {
1441 ihandle elfloader;
1442
1443 /* no ibm,client-architecture-support call, try the old way */
1444 elfloader = call_prom("open", 1, 1,
1445 ADDR("/packages/elf-loader"));
1446 if (elfloader == 0) {
1447 prom_printf("couldn't open /packages/elf-loader\n");
1448 return;
1449 }
1450 call_prom("call-method", 3, 1, ADDR("process-elf-header"),
1451 elfloader, ADDR(&fake_elf));
1452 call_prom("close", 1, 0, elfloader);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001453 }
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001454#endif /* __BIG_ENDIAN__ */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001455}
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11001456#endif /* CONFIG_PPC_PSERIES */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001457
1458/*
1459 * Memory allocation strategy... our layout is normally:
1460 *
1461 * at 14Mb or more we have vmlinux, then a gap and initrd. In some
1462 * rare cases, initrd might end up being before the kernel though.
1463 * We assume this won't override the final kernel at 0, we have no
1464 * provision to handle that in this version, but it should hopefully
1465 * never happen.
1466 *
1467 * alloc_top is set to the top of RMO, eventually shrink down if the
1468 * TCEs overlap
1469 *
1470 * alloc_bottom is set to the top of kernel/initrd
1471 *
1472 * from there, allocations are done this way : rtas is allocated
1473 * topmost, and the device-tree is allocated from the bottom. We try
1474 * to grow the device-tree allocation as we progress. If we can't,
1475 * then we fail, we don't currently have a facility to restart
1476 * elsewhere, but that shouldn't be necessary.
1477 *
1478 * Note that calls to reserve_mem have to be done explicitly, memory
1479 * allocated with either alloc_up or alloc_down isn't automatically
1480 * reserved.
1481 */
1482
1483
1484/*
1485 * Allocates memory in the RMO upward from the kernel/initrd
1486 *
1487 * When align is 0, this is a special case, it means to allocate in place
1488 * at the current location of alloc_bottom or fail (that is basically
1489 * extending the previous allocation). Used for the device-tree flattening
1490 */
1491static unsigned long __init alloc_up(unsigned long size, unsigned long align)
1492{
Anton Blanchard5827d412012-11-26 17:40:03 +00001493 unsigned long base = alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001494 unsigned long addr = 0;
1495
Paul Mackerrasc49888202005-10-26 21:52:53 +10001496 if (align)
Christophe Leroyb7115312020-04-20 18:36:36 +00001497 base = ALIGN(base, align);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001498 prom_debug("%s(%lx, %lx)\n", __func__, size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001499 if (ram_top == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001500 prom_panic("alloc_up() called with mem not initialized\n");
1501
1502 if (align)
Christophe Leroyb7115312020-04-20 18:36:36 +00001503 base = ALIGN(alloc_bottom, align);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001504 else
Anton Blanchard5827d412012-11-26 17:40:03 +00001505 base = alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001506
Anton Blanchard5827d412012-11-26 17:40:03 +00001507 for(; (base + size) <= alloc_top;
Christophe Leroyb7115312020-04-20 18:36:36 +00001508 base = ALIGN(base + 0x100000, align)) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001509 prom_debug(" trying: 0x%lx\n\r", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001510 addr = (unsigned long)prom_claim(base, size, 0);
Paul Mackerrasc49888202005-10-26 21:52:53 +10001511 if (addr != PROM_ERROR && addr != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001512 break;
1513 addr = 0;
1514 if (align == 0)
1515 break;
1516 }
1517 if (addr == 0)
1518 return 0;
Anton Blanchard5827d412012-11-26 17:40:03 +00001519 alloc_bottom = addr + size;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001520
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001521 prom_debug(" -> %lx\n", addr);
1522 prom_debug(" alloc_bottom : %lx\n", alloc_bottom);
1523 prom_debug(" alloc_top : %lx\n", alloc_top);
1524 prom_debug(" alloc_top_hi : %lx\n", alloc_top_high);
1525 prom_debug(" rmo_top : %lx\n", rmo_top);
1526 prom_debug(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001527
1528 return addr;
1529}
1530
1531/*
1532 * Allocates memory downward, either from top of RMO, or if highmem
1533 * is set, from the top of RAM. Note that this one doesn't handle
1534 * failures. It does claim memory if highmem is not set.
1535 */
1536static unsigned long __init alloc_down(unsigned long size, unsigned long align,
1537 int highmem)
1538{
1539 unsigned long base, addr = 0;
1540
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001541 prom_debug("%s(%lx, %lx, %s)\n", __func__, size, align,
Anton Blanchard5827d412012-11-26 17:40:03 +00001542 highmem ? "(high)" : "(low)");
1543 if (ram_top == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001544 prom_panic("alloc_down() called with mem not initialized\n");
1545
1546 if (highmem) {
1547 /* Carve out storage for the TCE table. */
Christophe Leroye96d9042020-04-20 18:36:35 +00001548 addr = ALIGN_DOWN(alloc_top_high - size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001549 if (addr <= alloc_bottom)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001550 return 0;
1551 /* Will we bump into the RMO ? If yes, check out that we
1552 * didn't overlap existing allocations there, if we did,
1553 * we are dead, we must be the first in town !
1554 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001555 if (addr < rmo_top) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001556 /* Good, we are first */
Anton Blanchard5827d412012-11-26 17:40:03 +00001557 if (alloc_top == rmo_top)
1558 alloc_top = rmo_top = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001559 else
1560 return 0;
1561 }
Anton Blanchard5827d412012-11-26 17:40:03 +00001562 alloc_top_high = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001563 goto bail;
1564 }
1565
Christophe Leroye96d9042020-04-20 18:36:35 +00001566 base = ALIGN_DOWN(alloc_top - size, align);
Anton Blanchard5827d412012-11-26 17:40:03 +00001567 for (; base > alloc_bottom;
Christophe Leroye96d9042020-04-20 18:36:35 +00001568 base = ALIGN_DOWN(base - 0x100000, align)) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001569 prom_debug(" trying: 0x%lx\n\r", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001570 addr = (unsigned long)prom_claim(base, size, 0);
Paul Mackerrasc49888202005-10-26 21:52:53 +10001571 if (addr != PROM_ERROR && addr != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001572 break;
1573 addr = 0;
1574 }
1575 if (addr == 0)
1576 return 0;
Anton Blanchard5827d412012-11-26 17:40:03 +00001577 alloc_top = addr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001578
1579 bail:
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001580 prom_debug(" -> %lx\n", addr);
1581 prom_debug(" alloc_bottom : %lx\n", alloc_bottom);
1582 prom_debug(" alloc_top : %lx\n", alloc_top);
1583 prom_debug(" alloc_top_hi : %lx\n", alloc_top_high);
1584 prom_debug(" rmo_top : %lx\n", rmo_top);
1585 prom_debug(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001586
1587 return addr;
1588}
1589
1590/*
1591 * Parse a "reg" cell
1592 */
1593static unsigned long __init prom_next_cell(int s, cell_t **cellp)
1594{
1595 cell_t *p = *cellp;
1596 unsigned long r = 0;
1597
1598 /* Ignore more than 2 cells */
1599 while (s > sizeof(unsigned long) / 4) {
1600 p++;
1601 s--;
1602 }
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001603 r = be32_to_cpu(*p++);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001604#ifdef CONFIG_PPC64
Paul Mackerras35499c02005-10-22 16:02:39 +10001605 if (s > 1) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001606 r <<= 32;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001607 r |= be32_to_cpu(*(p++));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001608 }
1609#endif
1610 *cellp = p;
1611 return r;
1612}
1613
1614/*
1615 * Very dumb function for adding to the memory reserve list, but
1616 * we don't need anything smarter at this point
1617 *
1618 * XXX Eventually check for collisions. They should NEVER happen.
1619 * If problems seem to show up, it would be a good start to track
1620 * them down.
1621 */
Michael Ellerman0108d3f2007-05-07 15:58:28 +10001622static void __init reserve_mem(u64 base, u64 size)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001623{
Kumar Galacbbcf342006-01-11 17:57:13 -06001624 u64 top = base + size;
Anton Blanchard5827d412012-11-26 17:40:03 +00001625 unsigned long cnt = mem_reserve_cnt;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001626
1627 if (size == 0)
1628 return;
1629
1630 /* We need to always keep one empty entry so that we
1631 * have our terminator with "size" set to 0 since we are
1632 * dumb and just copy this entire array to the boot params
1633 */
Christophe Leroye96d9042020-04-20 18:36:35 +00001634 base = ALIGN_DOWN(base, PAGE_SIZE);
Christophe Leroyb7115312020-04-20 18:36:36 +00001635 top = ALIGN(top, PAGE_SIZE);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001636 size = top - base;
1637
1638 if (cnt >= (MEM_RESERVE_MAP_SIZE - 1))
1639 prom_panic("Memory reserve map exhausted !\n");
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001640 mem_reserve_map[cnt].base = cpu_to_be64(base);
1641 mem_reserve_map[cnt].size = cpu_to_be64(size);
Anton Blanchard5827d412012-11-26 17:40:03 +00001642 mem_reserve_cnt = cnt + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001643}
1644
1645/*
Adrian Bunkb3c2ffd2006-06-30 18:20:44 +02001646 * Initialize memory allocation mechanism, parse "memory" nodes and
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001647 * obtain that way the top of memory and RMO to setup out local allocator
1648 */
1649static void __init prom_init_mem(void)
1650{
1651 phandle node;
Mathieu Malaterreeab00a22018-04-04 22:08:35 +02001652 char type[64];
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001653 unsigned int plen;
1654 cell_t *p, *endp;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001655 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001656 u32 rac, rsc;
1657
1658 /*
1659 * We iterate the memory nodes to find
1660 * 1) top of RMO (first node)
1661 * 2) top of memory
1662 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001663 val = cpu_to_be32(2);
1664 prom_getprop(prom.root, "#address-cells", &val, sizeof(val));
1665 rac = be32_to_cpu(val);
1666 val = cpu_to_be32(1);
1667 prom_getprop(prom.root, "#size-cells", &val, sizeof(rsc));
1668 rsc = be32_to_cpu(val);
1669 prom_debug("root_addr_cells: %x\n", rac);
1670 prom_debug("root_size_cells: %x\n", rsc);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001671
1672 prom_debug("scanning memory:\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001673
1674 for (node = 0; prom_next_node(&node); ) {
1675 type[0] = 0;
1676 prom_getprop(node, "device_type", type, sizeof(type));
1677
Paul Mackerrasc49888202005-10-26 21:52:53 +10001678 if (type[0] == 0) {
1679 /*
1680 * CHRP Longtrail machines have no device_type
1681 * on the memory node, so check the name instead...
1682 */
1683 prom_getprop(node, "name", type, sizeof(type));
1684 }
Christophe Leroy450e7dd2019-04-26 16:23:28 +00001685 if (prom_strcmp(type, "memory"))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001686 continue;
Paul Mackerrasc49888202005-10-26 21:52:53 +10001687
Anton Blanchard5827d412012-11-26 17:40:03 +00001688 plen = prom_getprop(node, "reg", regbuf, sizeof(regbuf));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001689 if (plen > sizeof(regbuf)) {
1690 prom_printf("memory node too large for buffer !\n");
1691 plen = sizeof(regbuf);
1692 }
Anton Blanchard5827d412012-11-26 17:40:03 +00001693 p = regbuf;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001694 endp = p + (plen / sizeof(cell_t));
1695
1696#ifdef DEBUG_PROM
Mathieu Malaterrec806a6f2019-05-23 12:25:20 +02001697 memset(prom_scratch, 0, sizeof(prom_scratch));
1698 call_prom("package-to-path", 3, 1, node, prom_scratch,
1699 sizeof(prom_scratch) - 1);
1700 prom_debug(" node %s :\n", prom_scratch);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001701#endif /* DEBUG_PROM */
1702
1703 while ((endp - p) >= (rac + rsc)) {
1704 unsigned long base, size;
1705
1706 base = prom_next_cell(rac, &p);
1707 size = prom_next_cell(rsc, &p);
1708
1709 if (size == 0)
1710 continue;
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001711 prom_debug(" %lx %lx\n", base, size);
Anton Blanchard5827d412012-11-26 17:40:03 +00001712 if (base == 0 && (of_platform & PLATFORM_LPAR))
1713 rmo_top = size;
1714 if ((base + size) > ram_top)
1715 ram_top = base + size;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001716 }
1717 }
1718
Anton Blanchard5827d412012-11-26 17:40:03 +00001719 alloc_bottom = PAGE_ALIGN((unsigned long)&_end + 0x4000);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001720
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001721 /*
Benjamin Krillcf687872009-07-27 22:02:39 +00001722 * If prom_memory_limit is set we reduce the upper limits *except* for
1723 * alloc_top_high. This must be the real top of RAM so we can put
1724 * TCE's up there.
1725 */
1726
Anton Blanchard5827d412012-11-26 17:40:03 +00001727 alloc_top_high = ram_top;
Benjamin Krillcf687872009-07-27 22:02:39 +00001728
Anton Blanchard5827d412012-11-26 17:40:03 +00001729 if (prom_memory_limit) {
1730 if (prom_memory_limit <= alloc_bottom) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001731 prom_printf("Ignoring mem=%lx <= alloc_bottom.\n",
1732 prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00001733 prom_memory_limit = 0;
1734 } else if (prom_memory_limit >= ram_top) {
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001735 prom_printf("Ignoring mem=%lx >= ram_top.\n",
1736 prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00001737 prom_memory_limit = 0;
Benjamin Krillcf687872009-07-27 22:02:39 +00001738 } else {
Anton Blanchard5827d412012-11-26 17:40:03 +00001739 ram_top = prom_memory_limit;
1740 rmo_top = min(rmo_top, prom_memory_limit);
Benjamin Krillcf687872009-07-27 22:02:39 +00001741 }
1742 }
1743
1744 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001745 * Setup our top alloc point, that is top of RMO or top of
1746 * segment 0 when running non-LPAR.
1747 * Some RS64 machines have buggy firmware where claims up at
1748 * 1GB fail. Cap at 768MB as a workaround.
1749 * Since 768MB is plenty of room, and we need to cap to something
1750 * reasonable on 32-bit, cap at 768MB on all machines.
1751 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001752 if (!rmo_top)
1753 rmo_top = ram_top;
1754 rmo_top = min(0x30000000ul, rmo_top);
1755 alloc_top = rmo_top;
1756 alloc_top_high = ram_top;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001757
Paul Mackerras64968f62011-12-13 17:54:13 +00001758 /*
1759 * Check if we have an initrd after the kernel but still inside
1760 * the RMO. If we do move our bottom point to after it.
1761 */
Anton Blanchard5827d412012-11-26 17:40:03 +00001762 if (prom_initrd_start &&
1763 prom_initrd_start < rmo_top &&
1764 prom_initrd_end > alloc_bottom)
1765 alloc_bottom = PAGE_ALIGN(prom_initrd_end);
Paul Mackerras64968f62011-12-13 17:54:13 +00001766
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001767 prom_printf("memory layout at init:\n");
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001768 prom_printf(" memory_limit : %lx (16 MB aligned)\n",
1769 prom_memory_limit);
1770 prom_printf(" alloc_bottom : %lx\n", alloc_bottom);
1771 prom_printf(" alloc_top : %lx\n", alloc_top);
1772 prom_printf(" alloc_top_hi : %lx\n", alloc_top_high);
1773 prom_printf(" rmo_top : %lx\n", rmo_top);
1774 prom_printf(" ram_top : %lx\n", ram_top);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001775}
1776
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001777static void __init prom_close_stdin(void)
1778{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001779 __be32 val;
1780 ihandle stdin;
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001781
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001782 if (prom_getprop(prom.chosen, "stdin", &val, sizeof(val)) > 0) {
1783 stdin = be32_to_cpu(val);
1784 call_prom("close", 1, 0, stdin);
1785 }
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00001786}
1787
Ram Pai6a9c9302019-08-19 23:13:14 -03001788#ifdef CONFIG_PPC_SVM
Nick Childd2769602021-12-16 17:00:16 -05001789static int __init prom_rtas_hcall(uint64_t args)
Ram Pai6a9c9302019-08-19 23:13:14 -03001790{
1791 register uint64_t arg1 asm("r3") = H_RTAS;
1792 register uint64_t arg2 asm("r4") = args;
1793
1794 asm volatile("sc 1\n" : "=r" (arg1) :
1795 "r" (arg1),
1796 "r" (arg2) :);
Nicholas Piggin59dc5bf2021-06-18 01:51:03 +10001797 srr_regs_clobbered();
1798
Ram Pai6a9c9302019-08-19 23:13:14 -03001799 return arg1;
1800}
1801
1802static struct rtas_args __prombss os_term_args;
1803
1804static void __init prom_rtas_os_term(char *str)
1805{
1806 phandle rtas_node;
1807 __be32 val;
1808 u32 token;
1809
1810 prom_debug("%s: start...\n", __func__);
1811 rtas_node = call_prom("finddevice", 1, 1, ADDR("/rtas"));
1812 prom_debug("rtas_node: %x\n", rtas_node);
1813 if (!PHANDLE_VALID(rtas_node))
1814 return;
1815
1816 val = 0;
1817 prom_getprop(rtas_node, "ibm,os-term", &val, sizeof(val));
1818 token = be32_to_cpu(val);
1819 prom_debug("ibm,os-term: %x\n", token);
1820 if (token == 0)
1821 prom_panic("Could not get token for ibm,os-term\n");
1822 os_term_args.token = cpu_to_be32(token);
Alexey Kardashevskiy74bb84e2020-03-12 18:44:04 +11001823 os_term_args.nargs = cpu_to_be32(1);
1824 os_term_args.nret = cpu_to_be32(1);
1825 os_term_args.args[0] = cpu_to_be32(__pa(str));
Ram Pai6a9c9302019-08-19 23:13:14 -03001826 prom_rtas_hcall((uint64_t)&os_term_args);
1827}
1828#endif /* CONFIG_PPC_SVM */
1829
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001830/*
1831 * Allocate room for and instantiate RTAS
1832 */
1833static void __init prom_instantiate_rtas(void)
1834{
1835 phandle rtas_node;
1836 ihandle rtas_inst;
1837 u32 base, entry = 0;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001838 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001839 u32 size = 0;
1840
1841 prom_debug("prom_instantiate_rtas: start...\n");
1842
1843 rtas_node = call_prom("finddevice", 1, 1, ADDR("/rtas"));
1844 prom_debug("rtas_node: %x\n", rtas_node);
1845 if (!PHANDLE_VALID(rtas_node))
1846 return;
1847
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001848 val = 0;
1849 prom_getprop(rtas_node, "rtas-size", &val, sizeof(size));
1850 size = be32_to_cpu(val);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001851 if (size == 0)
1852 return;
1853
1854 base = alloc_down(size, PAGE_SIZE, 0);
Anton Blanchard6d1e2c62011-11-14 12:55:47 +00001855 if (base == 0)
1856 prom_panic("Could not allocate memory for RTAS\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001857
1858 rtas_inst = call_prom("open", 1, 1, ADDR("/rtas"));
1859 if (!IHANDLE_VALID(rtas_inst)) {
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001860 prom_printf("opening rtas package failed (%x)\n", rtas_inst);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001861 return;
1862 }
1863
Anton Blanchard1f8737a2009-03-31 20:06:15 +00001864 prom_printf("instantiating rtas at 0x%x...", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001865
1866 if (call_prom_ret("call-method", 3, 2, &entry,
1867 ADDR("instantiate-rtas"),
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001868 rtas_inst, base) != 0
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001869 || entry == 0) {
1870 prom_printf(" failed\n");
1871 return;
1872 }
1873 prom_printf(" done\n");
1874
1875 reserve_mem(base, size);
1876
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001877 val = cpu_to_be32(base);
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001878 prom_setprop(rtas_node, "/rtas", "linux,rtas-base",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001879 &val, sizeof(val));
1880 val = cpu_to_be32(entry);
Paul Mackerrasa23414b2005-11-10 12:00:55 +11001881 prom_setprop(rtas_node, "/rtas", "linux,rtas-entry",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001882 &val, sizeof(val));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001883
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10001884 /* Check if it supports "query-cpu-stopped-state" */
1885 if (prom_getprop(rtas_node, "query-cpu-stopped-state",
1886 &val, sizeof(val)) != PROM_ERROR)
1887 rtas_has_query_cpu_stopped = true;
1888
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001889 prom_debug("rtas base = 0x%x\n", base);
1890 prom_debug("rtas entry = 0x%x\n", entry);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001891 prom_debug("rtas size = 0x%x\n", size);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001892
1893 prom_debug("prom_instantiate_rtas: end...\n");
1894}
1895
1896#ifdef CONFIG_PPC64
1897/*
Ashley Lai4a727422012-08-14 18:34:57 -05001898 * Allocate room for and instantiate Stored Measurement Log (SML)
1899 */
1900static void __init prom_instantiate_sml(void)
1901{
1902 phandle ibmvtpm_node;
1903 ihandle ibmvtpm_inst;
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001904 u32 entry = 0, size = 0, succ = 0;
Ashley Lai4a727422012-08-14 18:34:57 -05001905 u64 base;
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001906 __be32 val;
Ashley Lai4a727422012-08-14 18:34:57 -05001907
1908 prom_debug("prom_instantiate_sml: start...\n");
1909
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001910 ibmvtpm_node = call_prom("finddevice", 1, 1, ADDR("/vdevice/vtpm"));
Ashley Lai4a727422012-08-14 18:34:57 -05001911 prom_debug("ibmvtpm_node: %x\n", ibmvtpm_node);
1912 if (!PHANDLE_VALID(ibmvtpm_node))
1913 return;
1914
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001915 ibmvtpm_inst = call_prom("open", 1, 1, ADDR("/vdevice/vtpm"));
Ashley Lai4a727422012-08-14 18:34:57 -05001916 if (!IHANDLE_VALID(ibmvtpm_inst)) {
1917 prom_printf("opening vtpm package failed (%x)\n", ibmvtpm_inst);
1918 return;
1919 }
1920
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001921 if (prom_getprop(ibmvtpm_node, "ibm,sml-efi-reformat-supported",
1922 &val, sizeof(val)) != PROM_ERROR) {
1923 if (call_prom_ret("call-method", 2, 2, &succ,
1924 ADDR("reformat-sml-to-efi-alignment"),
1925 ibmvtpm_inst) != 0 || succ == 0) {
1926 prom_printf("Reformat SML to EFI alignment failed\n");
1927 return;
1928 }
Hon Ching \(Vicky\) Lob4ed0462015-10-07 20:11:53 -04001929
Hon Ching \(Vicky\) Lo9e5d4af2015-10-07 20:11:54 -04001930 if (call_prom_ret("call-method", 2, 2, &size,
1931 ADDR("sml-get-allocated-size"),
1932 ibmvtpm_inst) != 0 || size == 0) {
1933 prom_printf("SML get allocated size failed\n");
1934 return;
1935 }
1936 } else {
1937 if (call_prom_ret("call-method", 2, 2, &size,
1938 ADDR("sml-get-handover-size"),
1939 ibmvtpm_inst) != 0 || size == 0) {
1940 prom_printf("SML get handover size failed\n");
1941 return;
1942 }
Ashley Lai4a727422012-08-14 18:34:57 -05001943 }
1944
1945 base = alloc_down(size, PAGE_SIZE, 0);
1946 if (base == 0)
1947 prom_panic("Could not allocate memory for sml\n");
1948
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001949 prom_printf("instantiating sml at 0x%llx...", base);
Ashley Lai4a727422012-08-14 18:34:57 -05001950
Hon Ching \(Vicky\) Lo9e5d4af2015-10-07 20:11:54 -04001951 memset((void *)base, 0, size);
1952
Ashley Lai4a727422012-08-14 18:34:57 -05001953 if (call_prom_ret("call-method", 4, 2, &entry,
1954 ADDR("sml-handover"),
1955 ibmvtpm_inst, size, base) != 0 || entry == 0) {
1956 prom_printf("SML handover failed\n");
1957 return;
1958 }
1959 prom_printf(" done\n");
1960
1961 reserve_mem(base, size);
1962
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001963 prom_setprop(ibmvtpm_node, "/vdevice/vtpm", "linux,sml-base",
Ashley Lai4a727422012-08-14 18:34:57 -05001964 &base, sizeof(base));
Hon Ching \(Vicky\) Lo2f82e982015-10-07 20:11:52 -04001965 prom_setprop(ibmvtpm_node, "/vdevice/vtpm", "linux,sml-size",
Ashley Lai4a727422012-08-14 18:34:57 -05001966 &size, sizeof(size));
1967
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02001968 prom_debug("sml base = 0x%llx\n", base);
1969 prom_debug("sml size = 0x%x\n", size);
Ashley Lai4a727422012-08-14 18:34:57 -05001970
1971 prom_debug("prom_instantiate_sml: end...\n");
1972}
1973
1974/*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001975 * Allocate room for and initialize TCE tables
1976 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10001977#ifdef __BIG_ENDIAN__
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001978static void __init prom_initialize_tce_table(void)
1979{
1980 phandle node;
1981 ihandle phb_node;
1982 char compatible[64], type[64], model[64];
Anton Blanchard5827d412012-11-26 17:40:03 +00001983 char *path = prom_scratch;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001984 u64 base, align;
1985 u32 minalign, minsize;
1986 u64 tce_entry, *tce_entryp;
1987 u64 local_alloc_top, local_alloc_bottom;
1988 u64 i;
1989
Anton Blanchard5827d412012-11-26 17:40:03 +00001990 if (prom_iommu_off)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001991 return;
1992
1993 prom_debug("starting prom_initialize_tce_table\n");
1994
1995 /* Cache current top of allocs so we reserve a single block */
Anton Blanchard5827d412012-11-26 17:40:03 +00001996 local_alloc_top = alloc_top_high;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10001997 local_alloc_bottom = local_alloc_top;
1998
1999 /* Search all nodes looking for PHBs. */
2000 for (node = 0; prom_next_node(&node); ) {
2001 compatible[0] = 0;
2002 type[0] = 0;
2003 model[0] = 0;
2004 prom_getprop(node, "compatible",
2005 compatible, sizeof(compatible));
2006 prom_getprop(node, "device_type", type, sizeof(type));
2007 prom_getprop(node, "model", model, sizeof(model));
2008
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002009 if ((type[0] == 0) || (prom_strstr(type, "pci") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002010 continue;
2011
Linas Vepstase788ff12007-09-07 03:45:21 +10002012 /* Keep the old logic intact to avoid regression. */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002013 if (compatible[0] != 0) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002014 if ((prom_strstr(compatible, "python") == NULL) &&
2015 (prom_strstr(compatible, "Speedwagon") == NULL) &&
2016 (prom_strstr(compatible, "Winnipeg") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002017 continue;
2018 } else if (model[0] != 0) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002019 if ((prom_strstr(model, "ython") == NULL) &&
2020 (prom_strstr(model, "peedwagon") == NULL) &&
2021 (prom_strstr(model, "innipeg") == NULL))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002022 continue;
2023 }
2024
2025 if (prom_getprop(node, "tce-table-minalign", &minalign,
2026 sizeof(minalign)) == PROM_ERROR)
2027 minalign = 0;
2028 if (prom_getprop(node, "tce-table-minsize", &minsize,
2029 sizeof(minsize)) == PROM_ERROR)
2030 minsize = 4UL << 20;
2031
2032 /*
2033 * Even though we read what OF wants, we just set the table
2034 * size to 4 MB. This is enough to map 2GB of PCI DMA space.
2035 * By doing this, we avoid the pitfalls of trying to DMA to
2036 * MMIO space and the DMA alias hole.
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002037 */
Nicholas Piggin471d7ff2018-02-21 05:08:29 +10002038 minsize = 4UL << 20;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002039
2040 /* Align to the greater of the align or size */
2041 align = max(minalign, minsize);
2042 base = alloc_down(minsize, align, 1);
2043 if (base == 0)
2044 prom_panic("ERROR, cannot find space for TCE table.\n");
2045 if (base < local_alloc_bottom)
2046 local_alloc_bottom = base;
2047
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002048 /* It seems OF doesn't null-terminate the path :-( */
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002049 memset(path, 0, sizeof(prom_scratch));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002050 /* Call OF to setup the TCE hardware */
2051 if (call_prom("package-to-path", 3, 1, node,
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002052 path, sizeof(prom_scratch) - 1) == PROM_ERROR) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002053 prom_printf("package-to-path failed\n");
2054 }
2055
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002056 /* Save away the TCE table attributes for later use. */
2057 prom_setprop(node, path, "linux,tce-base", &base, sizeof(base));
2058 prom_setprop(node, path, "linux,tce-size", &minsize, sizeof(minsize));
2059
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002060 prom_debug("TCE table: %s\n", path);
2061 prom_debug("\tnode = 0x%x\n", node);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002062 prom_debug("\tbase = 0x%llx\n", base);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002063 prom_debug("\tsize = 0x%x\n", minsize);
2064
2065 /* Initialize the table to have a one-to-one mapping
2066 * over the allocated size.
2067 */
Ingo Molnar2b931fb2009-01-06 13:56:52 +00002068 tce_entryp = (u64 *)base;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002069 for (i = 0; i < (minsize >> 3) ;tce_entryp++, i++) {
2070 tce_entry = (i << PAGE_SHIFT);
2071 tce_entry |= 0x3;
2072 *tce_entryp = tce_entry;
2073 }
2074
2075 prom_printf("opening PHB %s", path);
2076 phb_node = call_prom("open", 1, 1, path);
2077 if (phb_node == 0)
2078 prom_printf("... failed\n");
2079 else
2080 prom_printf("... done\n");
2081
2082 call_prom("call-method", 6, 0, ADDR("set-64-bit-addressing"),
2083 phb_node, -1, minsize,
2084 (u32) base, (u32) (base >> 32));
2085 call_prom("close", 1, 0, phb_node);
2086 }
2087
2088 reserve_mem(local_alloc_bottom, local_alloc_top - local_alloc_bottom);
2089
Michael Ellerman2babf5c2006-05-17 18:00:46 +10002090 /* These are only really needed if there is a memory limit in
2091 * effect, but we don't know so export them always. */
Anton Blanchard5827d412012-11-26 17:40:03 +00002092 prom_tce_alloc_start = local_alloc_bottom;
2093 prom_tce_alloc_end = local_alloc_top;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002094
2095 /* Flag the first invalid entry */
2096 prom_debug("ending prom_initialize_tce_table\n");
2097}
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002098#endif /* __BIG_ENDIAN__ */
2099#endif /* CONFIG_PPC64 */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002100
2101/*
2102 * With CHRP SMP we need to use the OF to start the other processors.
2103 * We can't wait until smp_boot_cpus (the OF is trashed by then)
2104 * so we have to put the processors into a holding pattern controlled
2105 * by the kernel (not OF) before we destroy the OF.
2106 *
2107 * This uses a chunk of low memory, puts some holding pattern
2108 * code there and sends the other processors off to there until
2109 * smp_boot_cpus tells them to do something. The holding pattern
2110 * checks that address until its cpu # is there, when it is that
2111 * cpu jumps to __secondary_start(). smp_boot_cpus() takes care
2112 * of setting those values.
2113 *
2114 * We also use physical address 0x4 here to tell when a cpu
2115 * is in its holding pattern code.
2116 *
2117 * -- Cort
2118 */
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002119/*
2120 * We want to reference the copy of __secondary_hold_* in the
2121 * 0 - 0x100 address range
2122 */
2123#define LOW_ADDR(x) (((unsigned long) &(x)) & 0xff)
2124
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002125static void __init prom_hold_cpus(void)
2126{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002127 unsigned long i;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002128 phandle node;
2129 char type[64];
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002130 unsigned long *spinloop
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002131 = (void *) LOW_ADDR(__secondary_hold_spinloop);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002132 unsigned long *acknowledge
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002133 = (void *) LOW_ADDR(__secondary_hold_acknowledge);
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002134 unsigned long secondary_hold = LOW_ADDR(__secondary_hold);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002135
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10002136 /*
2137 * On pseries, if RTAS supports "query-cpu-stopped-state",
2138 * we skip this stage, the CPUs will be started by the
2139 * kernel using RTAS.
2140 */
2141 if ((of_platform == PLATFORM_PSERIES ||
2142 of_platform == PLATFORM_PSERIES_LPAR) &&
2143 rtas_has_query_cpu_stopped) {
2144 prom_printf("prom_hold_cpus: skipped\n");
2145 return;
2146 }
2147
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002148 prom_debug("prom_hold_cpus: start...\n");
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002149 prom_debug(" 1) spinloop = 0x%lx\n", (unsigned long)spinloop);
2150 prom_debug(" 1) *spinloop = 0x%lx\n", *spinloop);
2151 prom_debug(" 1) acknowledge = 0x%lx\n",
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002152 (unsigned long)acknowledge);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002153 prom_debug(" 1) *acknowledge = 0x%lx\n", *acknowledge);
2154 prom_debug(" 1) secondary_hold = 0x%lx\n", secondary_hold);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002155
2156 /* Set the common spinloop variable, so all of the secondary cpus
2157 * will block when they are awakened from their OF spinloop.
2158 * This must occur for both SMP and non SMP kernels, since OF will
2159 * be trashed when we move the kernel.
2160 */
2161 *spinloop = 0;
2162
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002163 /* look for cpus */
2164 for (node = 0; prom_next_node(&node); ) {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002165 unsigned int cpu_no;
2166 __be32 reg;
2167
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002168 type[0] = 0;
2169 prom_getprop(node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002170 if (prom_strcmp(type, "cpu") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002171 continue;
2172
2173 /* Skip non-configured cpus. */
2174 if (prom_getprop(node, "status", type, sizeof(type)) > 0)
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002175 if (prom_strcmp(type, "okay") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002176 continue;
2177
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002178 reg = cpu_to_be32(-1); /* make sparse happy */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002179 prom_getprop(node, "reg", &reg, sizeof(reg));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002180 cpu_no = be32_to_cpu(reg);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002181
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002182 prom_debug("cpu hw idx = %u\n", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002183
2184 /* Init the acknowledge var which will be reset by
2185 * the secondary cpu when it awakens from its OF
2186 * spinloop.
2187 */
2188 *acknowledge = (unsigned long)-1;
2189
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002190 if (cpu_no != prom.cpu) {
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00002191 /* Primary Thread of non-boot cpu or any thread */
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002192 prom_printf("starting cpu hw idx %u... ", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002193 call_prom("start-cpu", 3, 0, node,
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002194 secondary_hold, cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002195
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002196 for (i = 0; (i < 100000000) &&
2197 (*acknowledge == ((unsigned long)-1)); i++ )
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002198 mb();
2199
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002200 if (*acknowledge == cpu_no)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002201 prom_printf("done\n");
Paul Mackerrasbbd0abd2005-10-26 21:45:56 +10002202 else
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002203 prom_printf("failed: %lx\n", *acknowledge);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002204 }
2205#ifdef CONFIG_SMP
2206 else
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002207 prom_printf("boot cpu hw idx %u\n", cpu_no);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002208#endif /* CONFIG_SMP */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002209 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002210
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002211 prom_debug("prom_hold_cpus: end...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002212}
2213
2214
2215static void __init prom_init_client_services(unsigned long pp)
2216{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002217 /* Get a handle to the prom entry point before anything else */
Anton Blanchard5827d412012-11-26 17:40:03 +00002218 prom_entry = pp;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002219
2220 /* get a handle for the stdout device */
Anton Blanchard5827d412012-11-26 17:40:03 +00002221 prom.chosen = call_prom("finddevice", 1, 1, ADDR("/chosen"));
2222 if (!PHANDLE_VALID(prom.chosen))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002223 prom_panic("cannot find chosen"); /* msg won't be printed :( */
2224
2225 /* get device tree root */
Anton Blanchard5827d412012-11-26 17:40:03 +00002226 prom.root = call_prom("finddevice", 1, 1, ADDR("/"));
2227 if (!PHANDLE_VALID(prom.root))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002228 prom_panic("cannot find device tree root"); /* msg won't be printed :( */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002229
Anton Blanchard5827d412012-11-26 17:40:03 +00002230 prom.mmumap = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002231}
2232
Paul Mackerrasa575b802005-10-23 17:23:21 +10002233#ifdef CONFIG_PPC32
2234/*
2235 * For really old powermacs, we need to map things we claim.
2236 * For that, we need the ihandle of the mmu.
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002237 * Also, on the longtrail, we need to work around other bugs.
Paul Mackerrasa575b802005-10-23 17:23:21 +10002238 */
2239static void __init prom_find_mmu(void)
2240{
Paul Mackerrasa575b802005-10-23 17:23:21 +10002241 phandle oprom;
2242 char version[64];
2243
2244 oprom = call_prom("finddevice", 1, 1, ADDR("/openprom"));
2245 if (!PHANDLE_VALID(oprom))
2246 return;
2247 if (prom_getprop(oprom, "model", version, sizeof(version)) <= 0)
2248 return;
2249 version[sizeof(version) - 1] = 0;
Paul Mackerrasa575b802005-10-23 17:23:21 +10002250 /* XXX might need to add other versions here */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002251 if (prom_strcmp(version, "Open Firmware, 1.0.5") == 0)
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002252 of_workarounds = OF_WA_CLAIM;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002253 else if (prom_strncmp(version, "FirmWorks,3.", 12) == 0) {
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002254 of_workarounds = OF_WA_CLAIM | OF_WA_LONGTRAIL;
2255 call_prom("interpret", 1, 1, "dev /memory 0 to allow-reclaim");
2256 } else
Paul Mackerrasa575b802005-10-23 17:23:21 +10002257 return;
Anton Blanchard5827d412012-11-26 17:40:03 +00002258 prom.memory = call_prom("open", 1, 1, ADDR("/memory"));
2259 prom_getprop(prom.chosen, "mmu", &prom.mmumap,
2260 sizeof(prom.mmumap));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002261 prom.mmumap = be32_to_cpu(prom.mmumap);
Anton Blanchard5827d412012-11-26 17:40:03 +00002262 if (!IHANDLE_VALID(prom.memory) || !IHANDLE_VALID(prom.mmumap))
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002263 of_workarounds &= ~OF_WA_CLAIM; /* hmmm */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002264}
2265#else
2266#define prom_find_mmu()
2267#endif
2268
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002269static void __init prom_init_stdout(void)
2270{
Anton Blanchard5827d412012-11-26 17:40:03 +00002271 char *path = of_stdout_device;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002272 char type[16];
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002273 phandle stdout_node;
2274 __be32 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002275
Anton Blanchard5827d412012-11-26 17:40:03 +00002276 if (prom_getprop(prom.chosen, "stdout", &val, sizeof(val)) <= 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002277 prom_panic("cannot find stdout");
2278
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002279 prom.stdout = be32_to_cpu(val);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002280
2281 /* Get the full OF pathname of the stdout device */
2282 memset(path, 0, 256);
Anton Blanchard5827d412012-11-26 17:40:03 +00002283 call_prom("instance-to-path", 3, 1, prom.stdout, path, 255);
Anton Blanchard5827d412012-11-26 17:40:03 +00002284 prom_printf("OF stdout device is: %s\n", of_stdout_device);
2285 prom_setprop(prom.chosen, "/chosen", "linux,stdout-path",
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002286 path, prom_strlen(path) + 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002287
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002288 /* instance-to-package fails on PA-Semi */
2289 stdout_node = call_prom("instance-to-package", 1, 1, prom.stdout);
2290 if (stdout_node != PROM_ERROR) {
2291 val = cpu_to_be32(stdout_node);
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002292
2293 /* If it's a display, note it */
2294 memset(type, 0, sizeof(type));
2295 prom_getprop(stdout_node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002296 if (prom_strcmp(type, "display") == 0)
Benjamin Herrenschmidt10348f52014-01-13 09:49:17 +11002297 prom_setprop(stdout_node, path, "linux,boot-display", NULL, 0);
2298 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002299}
2300
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002301static int __init prom_find_machine_type(void)
2302{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002303 char compat[256];
2304 int len, i = 0;
Benjamin Herrenschmidt21fe3302005-11-07 16:41:59 +11002305#ifdef CONFIG_PPC64
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002306 phandle rtas;
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002307 int x;
Benjamin Herrenschmidt21fe3302005-11-07 16:41:59 +11002308#endif
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002309
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002310 /* Look for a PowerMac or a Cell */
Anton Blanchard5827d412012-11-26 17:40:03 +00002311 len = prom_getprop(prom.root, "compatible",
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002312 compat, sizeof(compat)-1);
2313 if (len > 0) {
2314 compat[len] = 0;
2315 while (i < len) {
2316 char *p = &compat[i];
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002317 int sl = prom_strlen(p);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002318 if (sl == 0)
2319 break;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002320 if (prom_strstr(p, "Power Macintosh") ||
2321 prom_strstr(p, "MacRISC"))
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002322 return PLATFORM_POWERMAC;
Arnd Bergmann133dda12006-06-07 12:04:18 +10002323#ifdef CONFIG_PPC64
2324 /* We must make sure we don't detect the IBM Cell
2325 * blades as pSeries due to some firmware issues,
2326 * so we do it here.
2327 */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002328 if (prom_strstr(p, "IBM,CBEA") ||
2329 prom_strstr(p, "IBM,CPBW-1.0"))
Arnd Bergmann133dda12006-06-07 12:04:18 +10002330 return PLATFORM_GENERIC;
2331#endif /* CONFIG_PPC64 */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002332 i += sl + 1;
2333 }
2334 }
2335#ifdef CONFIG_PPC64
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002336 /* Try to figure out if it's an IBM pSeries or any other
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002337 * PAPR compliant platform. We assume it is if :
2338 * - /device_type is "chrp" (please, do NOT use that for future
2339 * non-IBM designs !
2340 * - it has /rtas
2341 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002342 len = prom_getprop(prom.root, "device_type",
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002343 compat, sizeof(compat)-1);
2344 if (len <= 0)
2345 return PLATFORM_GENERIC;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002346 if (prom_strcmp(compat, "chrp"))
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002347 return PLATFORM_GENERIC;
2348
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002349 /* Default to pSeries. We need to know if we are running LPAR */
2350 rtas = call_prom("finddevice", 1, 1, ADDR("/rtas"));
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002351 if (!PHANDLE_VALID(rtas))
2352 return PLATFORM_GENERIC;
2353 x = prom_getproplen(rtas, "ibm,hypertas-functions");
2354 if (x != PROM_ERROR) {
Anton Blanchard4da727a2009-03-31 20:06:14 +00002355 prom_debug("Hypertas detected, assuming LPAR !\n");
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002356 return PLATFORM_PSERIES_LPAR;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002357 }
2358 return PLATFORM_PSERIES;
2359#else
Benjamin Herrenschmidte8222502006-03-28 23:15:54 +11002360 return PLATFORM_GENERIC;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002361#endif
2362}
2363
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002364static int __init prom_set_color(ihandle ih, int i, int r, int g, int b)
2365{
2366 return call_prom("call-method", 6, 1, ADDR("color!"), ih, i, b, g, r);
2367}
2368
2369/*
2370 * If we have a display that we don't know how to drive,
2371 * we will want to try to execute OF's open method for it
2372 * later. However, OF will probably fall over if we do that
2373 * we've taken over the MMU.
2374 * So we check whether we will need to open the display,
2375 * and if so, open it now.
2376 */
2377static void __init prom_check_displays(void)
2378{
2379 char type[16], *path;
2380 phandle node;
2381 ihandle ih;
2382 int i;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002383
Benjamin Herrenschmidtd00e34b2018-10-15 13:49:56 +11002384 static const unsigned char default_colors[] __initconst = {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002385 0x00, 0x00, 0x00,
2386 0x00, 0x00, 0xaa,
2387 0x00, 0xaa, 0x00,
2388 0x00, 0xaa, 0xaa,
2389 0xaa, 0x00, 0x00,
2390 0xaa, 0x00, 0xaa,
2391 0xaa, 0xaa, 0x00,
2392 0xaa, 0xaa, 0xaa,
2393 0x55, 0x55, 0x55,
2394 0x55, 0x55, 0xff,
2395 0x55, 0xff, 0x55,
2396 0x55, 0xff, 0xff,
2397 0xff, 0x55, 0x55,
2398 0xff, 0x55, 0xff,
2399 0xff, 0xff, 0x55,
2400 0xff, 0xff, 0xff
2401 };
2402 const unsigned char *clut;
2403
Anton Blanchard4da727a2009-03-31 20:06:14 +00002404 prom_debug("Looking for displays\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002405 for (node = 0; prom_next_node(&node); ) {
2406 memset(type, 0, sizeof(type));
2407 prom_getprop(node, "device_type", type, sizeof(type));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002408 if (prom_strcmp(type, "display") != 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002409 continue;
2410
2411 /* It seems OF doesn't null-terminate the path :-( */
Anton Blanchard5827d412012-11-26 17:40:03 +00002412 path = prom_scratch;
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002413 memset(path, 0, sizeof(prom_scratch));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002414
2415 /*
2416 * leave some room at the end of the path for appending extra
2417 * arguments
2418 */
2419 if (call_prom("package-to-path", 3, 1, node, path,
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002420 sizeof(prom_scratch) - 10) == PROM_ERROR)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002421 continue;
Anton Blanchard1f8737a2009-03-31 20:06:15 +00002422 prom_printf("found display : %s, opening... ", path);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002423
2424 ih = call_prom("open", 1, 1, path);
2425 if (ih == 0) {
2426 prom_printf("failed\n");
2427 continue;
2428 }
2429
2430 /* Success */
2431 prom_printf("done\n");
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002432 prom_setprop(node, path, "linux,opened", NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002433
2434 /* Setup a usable color table when the appropriate
2435 * method is available. Should update this to set-colors */
Anton Blanchard5827d412012-11-26 17:40:03 +00002436 clut = default_colors;
Benjamin Herrenschmidt3f536382011-12-14 13:55:11 +00002437 for (i = 0; i < 16; i++, clut += 3)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002438 if (prom_set_color(ih, i, clut[0], clut[1],
2439 clut[2]) != 0)
2440 break;
2441
2442#ifdef CONFIG_LOGO_LINUX_CLUT224
Anton Blanchard5827d412012-11-26 17:40:03 +00002443 clut = PTRRELOC(logo_linux_clut224.clut);
2444 for (i = 0; i < logo_linux_clut224.clutsize; i++, clut += 3)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002445 if (prom_set_color(ih, i + 32, clut[0], clut[1],
2446 clut[2]) != 0)
2447 break;
2448#endif /* CONFIG_LOGO_LINUX_CLUT224 */
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002449
2450#ifdef CONFIG_PPC_EARLY_DEBUG_BOOTX
2451 if (prom_getprop(node, "linux,boot-display", NULL, 0) !=
2452 PROM_ERROR) {
2453 u32 width, height, pitch, addr;
2454
2455 prom_printf("Setting btext !\n");
Michael Ellerman6c71cfc2020-08-21 20:34:07 +10002456
2457 if (prom_getprop(node, "width", &width, 4) == PROM_ERROR)
2458 return;
2459
2460 if (prom_getprop(node, "height", &height, 4) == PROM_ERROR)
2461 return;
2462
2463 if (prom_getprop(node, "linebytes", &pitch, 4) == PROM_ERROR)
2464 return;
2465
2466 if (prom_getprop(node, "address", &addr, 4) == PROM_ERROR)
2467 return;
2468
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002469 prom_printf("W=%d H=%d LB=%d addr=0x%x\n",
2470 width, height, pitch, addr);
2471 btext_setup_display(width, height, 8, pitch, addr);
Christophe Leroyc21f5a92019-06-03 13:00:51 +00002472 btext_prepare_BAT();
Benjamin Herrenschmidt7191b612013-07-25 12:12:32 +10002473 }
2474#endif /* CONFIG_PPC_EARLY_DEBUG_BOOTX */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002475 }
2476}
2477
2478
2479/* Return (relocated) pointer to this much memory: moves initrd if reqd. */
2480static void __init *make_room(unsigned long *mem_start, unsigned long *mem_end,
2481 unsigned long needed, unsigned long align)
2482{
2483 void *ret;
2484
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002485 *mem_start = ALIGN(*mem_start, align);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002486 while ((*mem_start + needed) > *mem_end) {
2487 unsigned long room, chunk;
2488
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002489 prom_debug("Chunk exhausted, claiming more at %lx...\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002490 alloc_bottom);
2491 room = alloc_top - alloc_bottom;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002492 if (room > DEVTREE_CHUNK_SIZE)
2493 room = DEVTREE_CHUNK_SIZE;
2494 if (room < PAGE_SIZE)
Anton Blanchardfbafd722011-07-25 20:47:51 +00002495 prom_panic("No memory for flatten_device_tree "
2496 "(no room)\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002497 chunk = alloc_up(room, 0);
2498 if (chunk == 0)
Anton Blanchardfbafd722011-07-25 20:47:51 +00002499 prom_panic("No memory for flatten_device_tree "
2500 "(claim failed)\n");
Anton Blanchard966728d2011-07-25 20:47:07 +00002501 *mem_end = chunk + room;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002502 }
2503
2504 ret = (void *)*mem_start;
2505 *mem_start += needed;
2506
2507 return ret;
2508}
2509
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002510#define dt_push_token(token, mem_start, mem_end) do { \
2511 void *room = make_room(mem_start, mem_end, 4, 4); \
2512 *(__be32 *)room = cpu_to_be32(token); \
2513 } while(0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002514
2515static unsigned long __init dt_find_string(char *str)
2516{
2517 char *s, *os;
2518
Anton Blanchard5827d412012-11-26 17:40:03 +00002519 s = os = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002520 s += 4;
Anton Blanchard5827d412012-11-26 17:40:03 +00002521 while (s < (char *)dt_string_end) {
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002522 if (prom_strcmp(s, str) == 0)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002523 return s - os;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002524 s += prom_strlen(s) + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002525 }
2526 return 0;
2527}
2528
2529/*
2530 * The Open Firmware 1275 specification states properties must be 31 bytes or
2531 * less, however not all firmwares obey this. Make it 64 bytes to be safe.
2532 */
2533#define MAX_PROPERTY_NAME 64
2534
2535static void __init scan_dt_build_strings(phandle node,
2536 unsigned long *mem_start,
2537 unsigned long *mem_end)
2538{
2539 char *prev_name, *namep, *sstart;
2540 unsigned long soff;
2541 phandle child;
2542
Anton Blanchard5827d412012-11-26 17:40:03 +00002543 sstart = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002544
2545 /* get and store all property names */
Anton Blanchard5827d412012-11-26 17:40:03 +00002546 prev_name = "";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002547 for (;;) {
2548 /* 64 is max len of name including nul. */
2549 namep = make_room(mem_start, mem_end, MAX_PROPERTY_NAME, 1);
2550 if (call_prom("nextprop", 3, 1, node, prev_name, namep) != 1) {
2551 /* No more nodes: unwind alloc */
2552 *mem_start = (unsigned long)namep;
2553 break;
2554 }
2555
2556 /* skip "name" */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002557 if (prom_strcmp(namep, "name") == 0) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002558 *mem_start = (unsigned long)namep;
Anton Blanchard5827d412012-11-26 17:40:03 +00002559 prev_name = "name";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002560 continue;
2561 }
2562 /* get/create string entry */
2563 soff = dt_find_string(namep);
2564 if (soff != 0) {
2565 *mem_start = (unsigned long)namep;
2566 namep = sstart + soff;
2567 } else {
2568 /* Trim off some if we can */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002569 *mem_start = (unsigned long)namep + prom_strlen(namep) + 1;
Anton Blanchard5827d412012-11-26 17:40:03 +00002570 dt_string_end = *mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002571 }
2572 prev_name = namep;
2573 }
2574
2575 /* do all our children */
2576 child = call_prom("child", 1, 1, node);
2577 while (child != 0) {
2578 scan_dt_build_strings(child, mem_start, mem_end);
2579 child = call_prom("peer", 1, 1, child);
2580 }
2581}
2582
2583static void __init scan_dt_build_struct(phandle node, unsigned long *mem_start,
2584 unsigned long *mem_end)
2585{
2586 phandle child;
2587 char *namep, *prev_name, *sstart, *p, *ep, *lp, *path;
2588 unsigned long soff;
2589 unsigned char *valp;
Benjamin Herrenschmidt8ca2d512018-10-15 13:49:57 +11002590 static char pname[MAX_PROPERTY_NAME] __prombss;
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002591 int l, room, has_phandle = 0;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002592
2593 dt_push_token(OF_DT_BEGIN_NODE, mem_start, mem_end);
2594
2595 /* get the node's full name */
2596 namep = (char *)*mem_start;
Paul Mackerrasc49888202005-10-26 21:52:53 +10002597 room = *mem_end - *mem_start;
2598 if (room > 255)
2599 room = 255;
2600 l = call_prom("package-to-path", 3, 1, node, namep, room);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002601 if (l >= 0) {
2602 /* Didn't fit? Get more room. */
Paul Mackerrasc49888202005-10-26 21:52:53 +10002603 if (l >= room) {
2604 if (l >= *mem_end - *mem_start)
2605 namep = make_room(mem_start, mem_end, l+1, 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002606 call_prom("package-to-path", 3, 1, node, namep, l);
2607 }
2608 namep[l] = '\0';
2609
2610 /* Fixup an Apple bug where they have bogus \0 chars in the
Paul Mackerrasa575b802005-10-23 17:23:21 +10002611 * middle of the path in some properties, and extract
2612 * the unit name (everything after the last '/').
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002613 */
Paul Mackerrasa575b802005-10-23 17:23:21 +10002614 for (lp = p = namep, ep = namep + l; p < ep; p++) {
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002615 if (*p == '/')
Paul Mackerrasa575b802005-10-23 17:23:21 +10002616 lp = namep;
2617 else if (*p != 0)
2618 *lp++ = *p;
2619 }
2620 *lp = 0;
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002621 *mem_start = ALIGN((unsigned long)lp + 1, 4);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002622 }
2623
2624 /* get it again for debugging */
Anton Blanchard5827d412012-11-26 17:40:03 +00002625 path = prom_scratch;
Christophe Leroyd7fbe2a2019-04-02 09:08:38 +00002626 memset(path, 0, sizeof(prom_scratch));
2627 call_prom("package-to-path", 3, 1, node, path, sizeof(prom_scratch) - 1);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002628
2629 /* get and store all properties */
Anton Blanchard5827d412012-11-26 17:40:03 +00002630 prev_name = "";
2631 sstart = (char *)dt_string_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002632 for (;;) {
2633 if (call_prom("nextprop", 3, 1, node, prev_name,
Anton Blanchard5827d412012-11-26 17:40:03 +00002634 pname) != 1)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002635 break;
2636
2637 /* skip "name" */
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002638 if (prom_strcmp(pname, "name") == 0) {
Anton Blanchard5827d412012-11-26 17:40:03 +00002639 prev_name = "name";
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002640 continue;
2641 }
2642
2643 /* find string offset */
Anton Blanchard5827d412012-11-26 17:40:03 +00002644 soff = dt_find_string(pname);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002645 if (soff == 0) {
2646 prom_printf("WARNING: Can't find string index for"
Anton Blanchard5827d412012-11-26 17:40:03 +00002647 " <%s>, node %s\n", pname, path);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002648 break;
2649 }
2650 prev_name = sstart + soff;
2651
2652 /* get length */
Anton Blanchard5827d412012-11-26 17:40:03 +00002653 l = call_prom("getproplen", 2, 1, node, pname);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002654
2655 /* sanity checks */
2656 if (l == PROM_ERROR)
2657 continue;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002658
2659 /* push property head */
2660 dt_push_token(OF_DT_PROP, mem_start, mem_end);
2661 dt_push_token(l, mem_start, mem_end);
2662 dt_push_token(soff, mem_start, mem_end);
2663
2664 /* push property content */
2665 valp = make_room(mem_start, mem_end, l, 4);
Anton Blanchard5827d412012-11-26 17:40:03 +00002666 call_prom("getprop", 4, 1, node, pname, valp, l);
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002667 *mem_start = ALIGN(*mem_start, 4);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002668
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002669 if (!prom_strcmp(pname, "phandle"))
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002670 has_phandle = 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002671 }
2672
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002673 /* Add a "phandle" property if none already exist */
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002674 if (!has_phandle) {
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002675 soff = dt_find_string("phandle");
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002676 if (soff == 0)
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002677 prom_printf("WARNING: Can't find string index for <phandle> node %s\n", path);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002678 else {
2679 dt_push_token(OF_DT_PROP, mem_start, mem_end);
2680 dt_push_token(4, mem_start, mem_end);
2681 dt_push_token(soff, mem_start, mem_end);
2682 valp = make_room(mem_start, mem_end, 4, 4);
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002683 *(__be32 *)valp = cpu_to_be32(node);
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00002684 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002685 }
2686
2687 /* do all our children */
2688 child = call_prom("child", 1, 1, node);
2689 while (child != 0) {
2690 scan_dt_build_struct(child, mem_start, mem_end);
2691 child = call_prom("peer", 1, 1, child);
2692 }
2693
2694 dt_push_token(OF_DT_END_NODE, mem_start, mem_end);
2695}
2696
2697static void __init flatten_device_tree(void)
2698{
2699 phandle root;
2700 unsigned long mem_start, mem_end, room;
2701 struct boot_param_header *hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002702 char *namep;
2703 u64 *rsvmap;
2704
2705 /*
2706 * Check how much room we have between alloc top & bottom (+/- a
Anton Blanchardfbafd722011-07-25 20:47:51 +00002707 * few pages), crop to 1MB, as this is our "chunk" size
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002708 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002709 room = alloc_top - alloc_bottom - 0x4000;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002710 if (room > DEVTREE_CHUNK_SIZE)
2711 room = DEVTREE_CHUNK_SIZE;
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002712 prom_debug("starting device tree allocs at %lx\n", alloc_bottom);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002713
2714 /* Now try to claim that */
2715 mem_start = (unsigned long)alloc_up(room, PAGE_SIZE);
2716 if (mem_start == 0)
2717 prom_panic("Can't allocate initial device-tree chunk\n");
Anton Blanchard966728d2011-07-25 20:47:07 +00002718 mem_end = mem_start + room;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002719
2720 /* Get root of tree */
2721 root = call_prom("peer", 1, 1, (phandle)0);
2722 if (root == (phandle)0)
2723 prom_panic ("couldn't get device tree root\n");
2724
2725 /* Build header and make room for mem rsv map */
Christophe Leroyd3f3d3b2020-04-20 18:36:37 +00002726 mem_start = ALIGN(mem_start, 4);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002727 hdr = make_room(&mem_start, &mem_end,
2728 sizeof(struct boot_param_header), 4);
Anton Blanchard5827d412012-11-26 17:40:03 +00002729 dt_header_start = (unsigned long)hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002730 rsvmap = make_room(&mem_start, &mem_end, sizeof(mem_reserve_map), 8);
2731
2732 /* Start of strings */
2733 mem_start = PAGE_ALIGN(mem_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00002734 dt_string_start = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002735 mem_start += 4; /* hole */
2736
Benjamin Herrenschmidtf1f208e2018-10-15 13:50:00 +11002737 /* Add "phandle" in there, we'll need it */
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002738 namep = make_room(&mem_start, &mem_end, 16, 1);
Michael Ellermanf47d5a42021-06-21 16:49:37 +10002739 prom_strscpy_pad(namep, "phandle", sizeof("phandle"));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00002740 mem_start = (unsigned long)namep + prom_strlen(namep) + 1;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002741
2742 /* Build string array */
2743 prom_printf("Building dt strings...\n");
2744 scan_dt_build_strings(root, &mem_start, &mem_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00002745 dt_string_end = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002746
2747 /* Build structure */
2748 mem_start = PAGE_ALIGN(mem_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00002749 dt_struct_start = mem_start;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002750 prom_printf("Building dt structure...\n");
2751 scan_dt_build_struct(root, &mem_start, &mem_end);
2752 dt_push_token(OF_DT_END, &mem_start, &mem_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00002753 dt_struct_end = PAGE_ALIGN(mem_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002754
2755 /* Finish header */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002756 hdr->boot_cpuid_phys = cpu_to_be32(prom.cpu);
2757 hdr->magic = cpu_to_be32(OF_DT_HEADER);
2758 hdr->totalsize = cpu_to_be32(dt_struct_end - dt_header_start);
2759 hdr->off_dt_struct = cpu_to_be32(dt_struct_start - dt_header_start);
2760 hdr->off_dt_strings = cpu_to_be32(dt_string_start - dt_header_start);
2761 hdr->dt_strings_size = cpu_to_be32(dt_string_end - dt_string_start);
2762 hdr->off_mem_rsvmap = cpu_to_be32(((unsigned long)rsvmap) - dt_header_start);
2763 hdr->version = cpu_to_be32(OF_DT_VERSION);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002764 /* Version 16 is not backward compatible */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002765 hdr->last_comp_version = cpu_to_be32(0x10);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002766
Jimi Xenidis4d1f3f22006-05-18 17:03:05 -05002767 /* Copy the reserve map in */
Anton Blanchard5827d412012-11-26 17:40:03 +00002768 memcpy(rsvmap, mem_reserve_map, sizeof(mem_reserve_map));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002769
2770#ifdef DEBUG_PROM
2771 {
2772 int i;
2773 prom_printf("reserved memory map:\n");
Anton Blanchard5827d412012-11-26 17:40:03 +00002774 for (i = 0; i < mem_reserve_cnt; i++)
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002775 prom_printf(" %llx - %llx\n",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10002776 be64_to_cpu(mem_reserve_map[i].base),
2777 be64_to_cpu(mem_reserve_map[i].size));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002778 }
2779#endif
Jimi Xenidis4d1f3f22006-05-18 17:03:05 -05002780 /* Bump mem_reserve_cnt to cause further reservations to fail
2781 * since it's too late.
2782 */
Anton Blanchard5827d412012-11-26 17:40:03 +00002783 mem_reserve_cnt = MEM_RESERVE_MAP_SIZE;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002784
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002785 prom_printf("Device tree strings 0x%lx -> 0x%lx\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002786 dt_string_start, dt_string_end);
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02002787 prom_printf("Device tree struct 0x%lx -> 0x%lx\n",
Anton Blanchard5827d412012-11-26 17:40:03 +00002788 dt_struct_start, dt_struct_end);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002789}
2790
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002791#ifdef CONFIG_PPC_MAPLE
2792/* PIBS Version 1.05.0000 04/26/2005 has an incorrect /ht/isa/ranges property.
2793 * The values are bad, and it doesn't even have the right number of cells. */
2794static void __init fixup_device_tree_maple(void)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002795{
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002796 phandle isa;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002797 u32 rloc = 0x01002000; /* IO space; PCI device = 4 */
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002798 u32 isa_ranges[6];
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002799 char *name;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002800
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002801 name = "/ht@0/isa@4";
2802 isa = call_prom("finddevice", 1, 1, ADDR(name));
2803 if (!PHANDLE_VALID(isa)) {
2804 name = "/ht@0/isa@6";
2805 isa = call_prom("finddevice", 1, 1, ADDR(name));
2806 rloc = 0x01003000; /* IO space; PCI device = 6 */
2807 }
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002808 if (!PHANDLE_VALID(isa))
2809 return;
2810
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002811 if (prom_getproplen(isa, "ranges") != 12)
2812 return;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002813 if (prom_getprop(isa, "ranges", isa_ranges, sizeof(isa_ranges))
2814 == PROM_ERROR)
2815 return;
2816
2817 if (isa_ranges[0] != 0x1 ||
2818 isa_ranges[1] != 0xf4000000 ||
2819 isa_ranges[2] != 0x00010000)
2820 return;
2821
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002822 prom_printf("Fixing up bogus ISA range on Maple/Apache...\n");
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002823
2824 isa_ranges[0] = 0x1;
2825 isa_ranges[1] = 0x0;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002826 isa_ranges[2] = rloc;
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002827 isa_ranges[3] = 0x0;
2828 isa_ranges[4] = 0x0;
2829 isa_ranges[5] = 0x00010000;
Benjamin Herrenschmidt980a6512006-07-03 17:22:05 +10002830 prom_setprop(isa, name, "ranges",
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002831 isa_ranges, sizeof(isa_ranges));
2832}
Harry Ciao8f101a052009-06-17 16:28:00 -07002833
2834#define CPC925_MC_START 0xf8000000
2835#define CPC925_MC_LENGTH 0x1000000
2836/* The values for memory-controller don't have right number of cells */
2837static void __init fixup_device_tree_maple_memory_controller(void)
2838{
2839 phandle mc;
2840 u32 mc_reg[4];
2841 char *name = "/hostbridge@f8000000";
Harry Ciao8f101a052009-06-17 16:28:00 -07002842 u32 ac, sc;
2843
2844 mc = call_prom("finddevice", 1, 1, ADDR(name));
2845 if (!PHANDLE_VALID(mc))
2846 return;
2847
2848 if (prom_getproplen(mc, "reg") != 8)
2849 return;
2850
Anton Blanchard5827d412012-11-26 17:40:03 +00002851 prom_getprop(prom.root, "#address-cells", &ac, sizeof(ac));
2852 prom_getprop(prom.root, "#size-cells", &sc, sizeof(sc));
Harry Ciao8f101a052009-06-17 16:28:00 -07002853 if ((ac != 2) || (sc != 2))
2854 return;
2855
2856 if (prom_getprop(mc, "reg", mc_reg, sizeof(mc_reg)) == PROM_ERROR)
2857 return;
2858
2859 if (mc_reg[0] != CPC925_MC_START || mc_reg[1] != CPC925_MC_LENGTH)
2860 return;
2861
2862 prom_printf("Fixing up bogus hostbridge on Maple...\n");
2863
2864 mc_reg[0] = 0x0;
2865 mc_reg[1] = CPC925_MC_START;
2866 mc_reg[2] = 0x0;
2867 mc_reg[3] = CPC925_MC_LENGTH;
2868 prom_setprop(mc, name, "reg", mc_reg, sizeof(mc_reg));
2869}
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002870#else
2871#define fixup_device_tree_maple()
Harry Ciao8f101a052009-06-17 16:28:00 -07002872#define fixup_device_tree_maple_memory_controller()
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002873#endif
2874
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002875#ifdef CONFIG_PPC_CHRP
Olaf Heringe4805922007-04-04 18:20:04 +02002876/*
2877 * Pegasos and BriQ lacks the "ranges" property in the isa node
2878 * Pegasos needs decimal IRQ 14/15, not hexadecimal
Olaf Hering556ecf92007-08-18 04:27:17 +10002879 * Pegasos has the IDE configured in legacy mode, but advertised as native
Olaf Heringe4805922007-04-04 18:20:04 +02002880 */
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002881static void __init fixup_device_tree_chrp(void)
2882{
Olaf Heringe4805922007-04-04 18:20:04 +02002883 phandle ph;
2884 u32 prop[6];
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002885 u32 rloc = 0x01006000; /* IO space; PCI device = 12 */
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002886 char *name;
2887 int rc;
2888
2889 name = "/pci@80000000/isa@c";
Olaf Heringe4805922007-04-04 18:20:04 +02002890 ph = call_prom("finddevice", 1, 1, ADDR(name));
2891 if (!PHANDLE_VALID(ph)) {
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002892 name = "/pci@ff500000/isa@6";
Olaf Heringe4805922007-04-04 18:20:04 +02002893 ph = call_prom("finddevice", 1, 1, ADDR(name));
Benjamin Herrenschmidt26c50322006-07-04 14:16:28 +10002894 rloc = 0x01003000; /* IO space; PCI device = 6 */
2895 }
Olaf Heringe4805922007-04-04 18:20:04 +02002896 if (PHANDLE_VALID(ph)) {
2897 rc = prom_getproplen(ph, "ranges");
2898 if (rc == 0 || rc == PROM_ERROR) {
2899 prom_printf("Fixing up missing ISA range on Pegasos...\n");
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002900
Olaf Heringe4805922007-04-04 18:20:04 +02002901 prop[0] = 0x1;
2902 prop[1] = 0x0;
2903 prop[2] = rloc;
2904 prop[3] = 0x0;
2905 prop[4] = 0x0;
2906 prop[5] = 0x00010000;
2907 prom_setprop(ph, name, "ranges", prop, sizeof(prop));
2908 }
2909 }
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002910
Olaf Heringe4805922007-04-04 18:20:04 +02002911 name = "/pci@80000000/ide@C,1";
2912 ph = call_prom("finddevice", 1, 1, ADDR(name));
2913 if (PHANDLE_VALID(ph)) {
2914 prom_printf("Fixing up IDE interrupt on Pegasos...\n");
2915 prop[0] = 14;
2916 prop[1] = 0x0;
Olaf Hering556ecf92007-08-18 04:27:17 +10002917 prom_setprop(ph, name, "interrupts", prop, 2*sizeof(u32));
2918 prom_printf("Fixing up IDE class-code on Pegasos...\n");
2919 rc = prom_getprop(ph, "class-code", prop, sizeof(u32));
2920 if (rc == sizeof(u32)) {
2921 prop[0] &= ~0x5;
2922 prom_setprop(ph, name, "class-code", prop, sizeof(u32));
2923 }
Olaf Heringe4805922007-04-04 18:20:04 +02002924 }
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10002925}
2926#else
2927#define fixup_device_tree_chrp()
2928#endif
2929
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002930#if defined(CONFIG_PPC64) && defined(CONFIG_PPC_PMAC)
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002931static void __init fixup_device_tree_pmac(void)
2932{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002933 phandle u3, i2c, mpic;
2934 u32 u3_rev;
2935 u32 interrupts[2];
2936 u32 parent;
2937
2938 /* Some G5s have a missing interrupt definition, fix it up here */
2939 u3 = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000"));
2940 if (!PHANDLE_VALID(u3))
2941 return;
2942 i2c = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000/i2c@f8001000"));
2943 if (!PHANDLE_VALID(i2c))
2944 return;
2945 mpic = call_prom("finddevice", 1, 1, ADDR("/u3@0,f8000000/mpic@f8040000"));
2946 if (!PHANDLE_VALID(mpic))
2947 return;
2948
2949 /* check if proper rev of u3 */
2950 if (prom_getprop(u3, "device-rev", &u3_rev, sizeof(u3_rev))
2951 == PROM_ERROR)
2952 return;
Benjamin Herrenschmidt7d496972005-11-07 14:36:21 +11002953 if (u3_rev < 0x35 || u3_rev > 0x39)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002954 return;
2955 /* does it need fixup ? */
2956 if (prom_getproplen(i2c, "interrupts") > 0)
2957 return;
2958
2959 prom_printf("fixing up bogus interrupts for u3 i2c...\n");
2960
2961 /* interrupt on this revision of u3 is number 0 and level */
2962 interrupts[0] = 0;
2963 interrupts[1] = 1;
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002964 prom_setprop(i2c, "/u3@0,f8000000/i2c@f8001000", "interrupts",
2965 &interrupts, sizeof(interrupts));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002966 parent = (u32)mpic;
Paul Mackerrasa23414b2005-11-10 12:00:55 +11002967 prom_setprop(i2c, "/u3@0,f8000000/i2c@f8001000", "interrupt-parent",
2968 &parent, sizeof(parent));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002969}
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05002970#else
2971#define fixup_device_tree_pmac()
2972#endif
Paul Mackerras9b6b5632005-10-06 12:06:20 +10002973
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002974#ifdef CONFIG_PPC_EFIKA
Grant Likely94d2dde2008-01-24 22:25:32 -07002975/*
2976 * The MPC5200 FEC driver requires an phy-handle property to tell it how
2977 * to talk to the phy. If the phy-handle property is missing, then this
2978 * function is called to add the appropriate nodes and link it to the
2979 * ethernet node.
2980 */
2981static void __init fixup_device_tree_efika_add_phy(void)
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002982{
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002983 u32 node;
2984 char prop[64];
Grant Likely94d2dde2008-01-24 22:25:32 -07002985 int rv;
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002986
Grant Likely94d2dde2008-01-24 22:25:32 -07002987 /* Check if /builtin/ethernet exists - bail if it doesn't */
2988 node = call_prom("finddevice", 1, 1, ADDR("/builtin/ethernet"));
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002989 if (!PHANDLE_VALID(node))
2990 return;
2991
Grant Likely94d2dde2008-01-24 22:25:32 -07002992 /* Check if the phy-handle property exists - bail if it does */
2993 rv = prom_getprop(node, "phy-handle", prop, sizeof(prop));
Peiwei Hu869fb7e2021-11-19 17:12:18 +08002994 if (rv <= 0)
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01002995 return;
2996
Grant Likely94d2dde2008-01-24 22:25:32 -07002997 /*
2998 * At this point the ethernet device doesn't have a phy described.
2999 * Now we need to add the missing phy node and linkage
3000 */
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003001
Grant Likely94d2dde2008-01-24 22:25:32 -07003002 /* Check for an MDIO bus node - if missing then create one */
Olaf Hering6f4347c2008-01-10 01:06:08 +11003003 node = call_prom("finddevice", 1, 1, ADDR("/builtin/mdio"));
3004 if (!PHANDLE_VALID(node)) {
3005 prom_printf("Adding Ethernet MDIO node\n");
3006 call_prom("interpret", 1, 1,
3007 " s\" /builtin\" find-device"
3008 " new-device"
3009 " 1 encode-int s\" #address-cells\" property"
3010 " 0 encode-int s\" #size-cells\" property"
Grant Likely94d2dde2008-01-24 22:25:32 -07003011 " s\" mdio\" device-name"
3012 " s\" fsl,mpc5200b-mdio\" encode-string"
Olaf Hering6f4347c2008-01-10 01:06:08 +11003013 " s\" compatible\" property"
3014 " 0xf0003000 0x400 reg"
3015 " 0x2 encode-int"
3016 " 0x5 encode-int encode+"
3017 " 0x3 encode-int encode+"
3018 " s\" interrupts\" property"
3019 " finish-device");
Jiapeng Chong4f46d572021-02-24 15:29:21 +08003020 }
Olaf Hering6f4347c2008-01-10 01:06:08 +11003021
Grant Likely94d2dde2008-01-24 22:25:32 -07003022 /* Check for a PHY device node - if missing then create one and
3023 * give it's phandle to the ethernet node */
3024 node = call_prom("finddevice", 1, 1,
3025 ADDR("/builtin/mdio/ethernet-phy"));
Olaf Hering6f4347c2008-01-10 01:06:08 +11003026 if (!PHANDLE_VALID(node)) {
3027 prom_printf("Adding Ethernet PHY node\n");
3028 call_prom("interpret", 1, 1,
3029 " s\" /builtin/mdio\" find-device"
3030 " new-device"
3031 " s\" ethernet-phy\" device-name"
3032 " 0x10 encode-int s\" reg\" property"
3033 " my-self"
3034 " ihandle>phandle"
3035 " finish-device"
3036 " s\" /builtin/ethernet\" find-device"
3037 " encode-int"
3038 " s\" phy-handle\" property"
3039 " device-end");
3040 }
Grant Likely94d2dde2008-01-24 22:25:32 -07003041}
Olaf Hering6f4347c2008-01-10 01:06:08 +11003042
Grant Likely94d2dde2008-01-24 22:25:32 -07003043static void __init fixup_device_tree_efika(void)
3044{
3045 int sound_irq[3] = { 2, 2, 0 };
3046 int bcomm_irq[3*16] = { 3,0,0, 3,1,0, 3,2,0, 3,3,0,
3047 3,4,0, 3,5,0, 3,6,0, 3,7,0,
3048 3,8,0, 3,9,0, 3,10,0, 3,11,0,
3049 3,12,0, 3,13,0, 3,14,0, 3,15,0 };
3050 u32 node;
3051 char prop[64];
3052 int rv, len;
3053
3054 /* Check if we're really running on a EFIKA */
3055 node = call_prom("finddevice", 1, 1, ADDR("/"));
3056 if (!PHANDLE_VALID(node))
3057 return;
3058
3059 rv = prom_getprop(node, "model", prop, sizeof(prop));
3060 if (rv == PROM_ERROR)
3061 return;
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003062 if (prom_strcmp(prop, "EFIKA5K2"))
Grant Likely94d2dde2008-01-24 22:25:32 -07003063 return;
3064
3065 prom_printf("Applying EFIKA device tree fixups\n");
3066
3067 /* Claiming to be 'chrp' is death */
3068 node = call_prom("finddevice", 1, 1, ADDR("/"));
3069 rv = prom_getprop(node, "device_type", prop, sizeof(prop));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003070 if (rv != PROM_ERROR && (prom_strcmp(prop, "chrp") == 0))
Grant Likely94d2dde2008-01-24 22:25:32 -07003071 prom_setprop(node, "/", "device_type", "efika", sizeof("efika"));
3072
David Woodhouse7f4392c2008-04-14 02:52:38 +10003073 /* CODEGEN,description is exposed in /proc/cpuinfo so
3074 fix that too */
3075 rv = prom_getprop(node, "CODEGEN,description", prop, sizeof(prop));
Christophe Leroy450e7dd2019-04-26 16:23:28 +00003076 if (rv != PROM_ERROR && (prom_strstr(prop, "CHRP")))
David Woodhouse7f4392c2008-04-14 02:52:38 +10003077 prom_setprop(node, "/", "CODEGEN,description",
3078 "Efika 5200B PowerPC System",
3079 sizeof("Efika 5200B PowerPC System"));
3080
Grant Likely94d2dde2008-01-24 22:25:32 -07003081 /* Fixup bestcomm interrupts property */
3082 node = call_prom("finddevice", 1, 1, ADDR("/builtin/bestcomm"));
3083 if (PHANDLE_VALID(node)) {
3084 len = prom_getproplen(node, "interrupts");
3085 if (len == 12) {
3086 prom_printf("Fixing bestcomm interrupts property\n");
3087 prom_setprop(node, "/builtin/bestcom", "interrupts",
3088 bcomm_irq, sizeof(bcomm_irq));
3089 }
3090 }
3091
3092 /* Fixup sound interrupts property */
3093 node = call_prom("finddevice", 1, 1, ADDR("/builtin/sound"));
3094 if (PHANDLE_VALID(node)) {
3095 rv = prom_getprop(node, "interrupts", prop, sizeof(prop));
3096 if (rv == PROM_ERROR) {
3097 prom_printf("Adding sound interrupts property\n");
3098 prom_setprop(node, "/builtin/sound", "interrupts",
3099 sound_irq, sizeof(sound_irq));
3100 }
3101 }
3102
3103 /* Make sure ethernet phy-handle property exists */
3104 fixup_device_tree_efika_add_phy();
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003105}
3106#else
3107#define fixup_device_tree_efika()
3108#endif
3109
Darren Stevens50246782016-08-31 13:24:40 +01003110#ifdef CONFIG_PPC_PASEMI_NEMO
3111/*
3112 * CFE supplied on Nemo is broken in several ways, biggest
3113 * problem is that it reassigns ISA interrupts to unused mpic ints.
3114 * Add an interrupt-controller property for the io-bridge to use
3115 * and correct the ints so we can attach them to an irq_domain
3116 */
3117static void __init fixup_device_tree_pasemi(void)
3118{
3119 u32 interrupts[2], parent, rval, val = 0;
3120 char *name, *pci_name;
3121 phandle iob, node;
3122
3123 /* Find the root pci node */
3124 name = "/pxp@0,e0000000";
3125 iob = call_prom("finddevice", 1, 1, ADDR(name));
3126 if (!PHANDLE_VALID(iob))
3127 return;
3128
3129 /* check if interrupt-controller node set yet */
3130 if (prom_getproplen(iob, "interrupt-controller") !=PROM_ERROR)
3131 return;
3132
3133 prom_printf("adding interrupt-controller property for SB600...\n");
3134
3135 prom_setprop(iob, name, "interrupt-controller", &val, 0);
3136
3137 pci_name = "/pxp@0,e0000000/pci@11";
3138 node = call_prom("finddevice", 1, 1, ADDR(pci_name));
3139 parent = ADDR(iob);
3140
3141 for( ; prom_next_node(&node); ) {
3142 /* scan each node for one with an interrupt */
3143 if (!PHANDLE_VALID(node))
3144 continue;
3145
3146 rval = prom_getproplen(node, "interrupts");
3147 if (rval == 0 || rval == PROM_ERROR)
3148 continue;
3149
3150 prom_getprop(node, "interrupts", &interrupts, sizeof(interrupts));
3151 if ((interrupts[0] < 212) || (interrupts[0] > 222))
3152 continue;
3153
3154 /* found a node, update both interrupts and interrupt-parent */
3155 if ((interrupts[0] >= 212) && (interrupts[0] <= 215))
3156 interrupts[0] -= 203;
3157 if ((interrupts[0] >= 216) && (interrupts[0] <= 220))
3158 interrupts[0] -= 213;
3159 if (interrupts[0] == 221)
3160 interrupts[0] = 14;
3161 if (interrupts[0] == 222)
3162 interrupts[0] = 8;
3163
3164 prom_setprop(node, pci_name, "interrupts", interrupts,
3165 sizeof(interrupts));
3166 prom_setprop(node, pci_name, "interrupt-parent", &parent,
3167 sizeof(parent));
3168 }
Darren Stevens687e16b2016-08-31 13:24:45 +01003169
3170 /*
3171 * The io-bridge has device_type set to 'io-bridge' change it to 'isa'
3172 * so that generic isa-bridge code can add the SB600 and its on-board
3173 * peripherals.
3174 */
3175 name = "/pxp@0,e0000000/io-bridge@0";
3176 iob = call_prom("finddevice", 1, 1, ADDR(name));
3177 if (!PHANDLE_VALID(iob))
3178 return;
3179
3180 /* device_type is already set, just change it. */
3181
3182 prom_printf("Changing device_type of SB600 node...\n");
3183
3184 prom_setprop(iob, name, "device_type", "isa", sizeof("isa"));
Darren Stevens50246782016-08-31 13:24:40 +01003185}
3186#else /* !CONFIG_PPC_PASEMI_NEMO */
3187static inline void fixup_device_tree_pasemi(void) { }
3188#endif
3189
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003190static void __init fixup_device_tree(void)
3191{
3192 fixup_device_tree_maple();
Harry Ciao8f101a052009-06-17 16:28:00 -07003193 fixup_device_tree_maple_memory_controller();
Benjamin Herrenschmidte8c0acf2006-07-04 14:06:29 +10003194 fixup_device_tree_chrp();
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003195 fixup_device_tree_pmac();
Sylvain Munaut88fd2a92007-02-12 23:13:20 +01003196 fixup_device_tree_efika();
Darren Stevens50246782016-08-31 13:24:40 +01003197 fixup_device_tree_pasemi();
Hollis Blanchard54f4ee12006-05-25 16:36:53 -05003198}
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003199
3200static void __init prom_find_boot_cpu(void)
3201{
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003202 __be32 rval;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003203 ihandle prom_cpu;
3204 phandle cpu_pkg;
3205
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003206 rval = 0;
3207 if (prom_getprop(prom.chosen, "cpu", &rval, sizeof(rval)) <= 0)
Paul Mackerrasa575b802005-10-23 17:23:21 +10003208 return;
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003209 prom_cpu = be32_to_cpu(rval);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003210
3211 cpu_pkg = call_prom("instance-to-package", 1, 1, prom_cpu);
3212
Darren Stevensaf2b7fa2017-01-23 19:42:54 +00003213 if (!PHANDLE_VALID(cpu_pkg))
3214 return;
3215
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003216 prom_getprop(cpu_pkg, "reg", &rval, sizeof(rval));
3217 prom.cpu = be32_to_cpu(rval);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003218
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02003219 prom_debug("Booting CPU hw index = %d\n", prom.cpu);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003220}
3221
3222static void __init prom_check_initrd(unsigned long r3, unsigned long r4)
3223{
3224#ifdef CONFIG_BLK_DEV_INITRD
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003225 if (r3 && r4 && r4 != 0xdeadbeef) {
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003226 __be64 val;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003227
Anton Blanchard5827d412012-11-26 17:40:03 +00003228 prom_initrd_start = is_kernel_addr(r3) ? __pa(r3) : r3;
3229 prom_initrd_end = prom_initrd_start + r4;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003230
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003231 val = cpu_to_be64(prom_initrd_start);
Anton Blanchard5827d412012-11-26 17:40:03 +00003232 prom_setprop(prom.chosen, "/chosen", "linux,initrd-start",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003233 &val, sizeof(val));
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003234 val = cpu_to_be64(prom_initrd_end);
Anton Blanchard5827d412012-11-26 17:40:03 +00003235 prom_setprop(prom.chosen, "/chosen", "linux,initrd-end",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003236 &val, sizeof(val));
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003237
Anton Blanchard5827d412012-11-26 17:40:03 +00003238 reserve_mem(prom_initrd_start,
3239 prom_initrd_end - prom_initrd_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003240
Mathieu Malaterreeae5f702018-04-06 22:12:19 +02003241 prom_debug("initrd_start=0x%lx\n", prom_initrd_start);
3242 prom_debug("initrd_end=0x%lx\n", prom_initrd_end);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003243 }
3244#endif /* CONFIG_BLK_DEV_INITRD */
3245}
3246
Ram Pai6a9c9302019-08-19 23:13:14 -03003247#ifdef CONFIG_PPC_SVM
3248/*
3249 * Perform the Enter Secure Mode ultracall.
3250 */
Nick Childd2769602021-12-16 17:00:16 -05003251static int __init enter_secure_mode(unsigned long kbase, unsigned long fdt)
Ram Pai6a9c9302019-08-19 23:13:14 -03003252{
3253 register unsigned long r3 asm("r3") = UV_ESM;
3254 register unsigned long r4 asm("r4") = kbase;
3255 register unsigned long r5 asm("r5") = fdt;
3256
3257 asm volatile("sc 2" : "+r"(r3) : "r"(r4), "r"(r5));
3258
3259 return r3;
3260}
3261
3262/*
3263 * Call the Ultravisor to transfer us to secure memory if we have an ESM blob.
3264 */
Vladis Dronovaff77952020-07-29 15:37:41 +02003265static void __init setup_secure_guest(unsigned long kbase, unsigned long fdt)
Ram Pai6a9c9302019-08-19 23:13:14 -03003266{
3267 int ret;
3268
3269 if (!prom_svm_enable)
3270 return;
3271
3272 /* Switch to secure mode. */
3273 prom_printf("Switching to secure mode.\n");
3274
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003275 /*
3276 * The ultravisor will do an integrity check of the kernel image but we
3277 * relocated it so the check will fail. Restore the original image by
3278 * relocating it back to the kernel virtual base address.
3279 */
Michael Ellerman24d33ac2021-06-23 23:04:54 +10003280 relocate(KERNELBASE);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003281
Ram Pai6a9c9302019-08-19 23:13:14 -03003282 ret = enter_secure_mode(kbase, fdt);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003283
3284 /* Relocate the kernel again. */
Michael Ellerman24d33ac2021-06-23 23:04:54 +10003285 relocate(kbase);
Thiago Jung Bauermann05d9a952019-09-11 13:34:33 -03003286
Ram Pai6a9c9302019-08-19 23:13:14 -03003287 if (ret != U_SUCCESS) {
3288 prom_printf("Returned %d from switching to secure mode.\n", ret);
3289 prom_rtas_os_term("Switch to secure mode failed.\n");
3290 }
3291}
3292#else
Vladis Dronovaff77952020-07-29 15:37:41 +02003293static void __init setup_secure_guest(unsigned long kbase, unsigned long fdt)
Ram Pai6a9c9302019-08-19 23:13:14 -03003294{
3295}
3296#endif /* CONFIG_PPC_SVM */
3297
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003298/*
3299 * We enter here early on, when the Open Firmware prom is still
3300 * handling exceptions and the MMU hash table for us.
3301 */
3302
3303unsigned long __init prom_init(unsigned long r3, unsigned long r4,
3304 unsigned long pp,
Paul Mackerras549e8152008-08-30 11:43:47 +10003305 unsigned long r6, unsigned long r7,
3306 unsigned long kbase)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003307{
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003308 unsigned long hdr;
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003309
3310#ifdef CONFIG_PPC32
Paul Mackerras549e8152008-08-30 11:43:47 +10003311 unsigned long offset = reloc_offset();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003312 reloc_got2(offset);
3313#endif
3314
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003315 /*
3316 * First zero the BSS
3317 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003318 memset(&__bss_start, 0, __bss_stop - __bss_start);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003319
3320 /*
3321 * Init interface to Open Firmware, get some node references,
3322 * like /chosen
3323 */
3324 prom_init_client_services(pp);
3325
3326 /*
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003327 * See if this OF is old enough that we need to do explicit maps
3328 * and other workarounds
3329 */
3330 prom_find_mmu();
3331
3332 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003333 * Init prom stdout device
3334 */
3335 prom_init_stdout();
3336
Anton Blanchard5827d412012-11-26 17:40:03 +00003337 prom_printf("Preparing to boot %s", linux_banner);
Michael Ellermane7943fb2009-03-04 19:02:01 +00003338
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003339 /*
3340 * Get default machine type. At this point, we do not differentiate
3341 * between pSeries SMP and pSeries LPAR
3342 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003343 of_platform = prom_find_machine_type();
3344 prom_printf("Detected machine type: %x\n", of_platform);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003345
Suzuki Poulose0f890c82011-12-14 22:57:15 +00003346#ifndef CONFIG_NONSTATIC_KERNEL
Olaf Heringadd60ef2006-03-23 22:03:57 +01003347 /* Bail if this is a kdump kernel. */
3348 if (PHYSICAL_START > 0)
3349 prom_panic("Error: You can't boot a kdump kernel from OF!\n");
Paul Mackerras549e8152008-08-30 11:43:47 +10003350#endif
Olaf Heringadd60ef2006-03-23 22:03:57 +01003351
3352 /*
3353 * Check for an initrd
3354 */
3355 prom_check_initrd(r3, r4);
3356
Suraj Jitindar Singh12cc9fd2017-02-28 17:03:47 +11003357 /*
3358 * Do early parsing of command line
3359 */
3360 early_cmdline_parse();
3361
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003362#ifdef CONFIG_PPC_PSERIES
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003363 /*
3364 * On pSeries, inform the firmware about our capabilities
3365 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003366 if (of_platform == PLATFORM_PSERIES ||
3367 of_platform == PLATFORM_PSERIES_LPAR)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003368 prom_send_capabilities();
3369#endif
3370
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003371 /*
Arnd Bergmannf3f66f52005-10-31 20:08:37 -05003372 * Copy the CPU hold code
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003373 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003374 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras549e8152008-08-30 11:43:47 +10003375 copy_and_flush(0, kbase, 0x100, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003376
3377 /*
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003378 * Initialize memory management within prom_init
3379 */
3380 prom_init_mem();
3381
3382 /*
3383 * Determine which cpu is actually running right _now_
3384 */
3385 prom_find_boot_cpu();
3386
3387 /*
3388 * Initialize display devices
3389 */
3390 prom_check_displays();
3391
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003392#if defined(CONFIG_PPC64) && defined(__BIG_ENDIAN__)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003393 /*
3394 * Initialize IOMMU (TCE tables) on pSeries. Do that before anything else
3395 * that uses the allocator, we need to make sure we get the top of memory
3396 * available for us here...
3397 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003398 if (of_platform == PLATFORM_PSERIES)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003399 prom_initialize_tce_table();
3400#endif
3401
3402 /*
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003403 * On non-powermacs, try to instantiate RTAS. PowerMacs don't
3404 * have a usable RTAS implementation.
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003405 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003406 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003407 prom_instantiate_rtas();
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003408
Ashley Lai4a727422012-08-14 18:34:57 -05003409#ifdef CONFIG_PPC64
3410 /* instantiate sml */
3411 prom_instantiate_sml();
3412#endif
3413
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003414 /*
3415 * On non-powermacs, put all CPUs in spin-loops.
3416 *
3417 * PowerMacs use a different mechanism to spin CPUs
Benjamin Herrenschmidtdbe78b42013-09-25 14:02:50 +10003418 *
3419 * (This must be done after instanciating RTAS)
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003420 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003421 if (of_platform != PLATFORM_POWERMAC)
Benjamin Herrenschmidt27f44882011-09-19 18:27:58 +00003422 prom_hold_cpus();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003423
3424 /*
3425 * Fill in some infos for use by the kernel later on
3426 */
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003427 if (prom_memory_limit) {
3428 __be64 val = cpu_to_be64(prom_memory_limit);
Anton Blanchard5827d412012-11-26 17:40:03 +00003429 prom_setprop(prom.chosen, "/chosen", "linux,memory-limit",
Benjamin Herrenschmidt493adff2013-08-07 02:01:38 +10003430 &val, sizeof(val));
3431 }
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003432#ifdef CONFIG_PPC64
Anton Blanchard5827d412012-11-26 17:40:03 +00003433 if (prom_iommu_off)
3434 prom_setprop(prom.chosen, "/chosen", "linux,iommu-off",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003435 NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003436
Anton Blanchard5827d412012-11-26 17:40:03 +00003437 if (prom_iommu_force_on)
3438 prom_setprop(prom.chosen, "/chosen", "linux,iommu-force-on",
Paul Mackerrasa23414b2005-11-10 12:00:55 +11003439 NULL, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003440
Anton Blanchard5827d412012-11-26 17:40:03 +00003441 if (prom_tce_alloc_start) {
3442 prom_setprop(prom.chosen, "/chosen", "linux,tce-alloc-start",
3443 &prom_tce_alloc_start,
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003444 sizeof(prom_tce_alloc_start));
Anton Blanchard5827d412012-11-26 17:40:03 +00003445 prom_setprop(prom.chosen, "/chosen", "linux,tce-alloc-end",
3446 &prom_tce_alloc_end,
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003447 sizeof(prom_tce_alloc_end));
3448 }
3449#endif
3450
3451 /*
3452 * Fixup any known bugs in the device-tree
3453 */
3454 fixup_device_tree();
3455
3456 /*
3457 * Now finally create the flattened device-tree
3458 */
Anton Blanchard1f8737a2009-03-31 20:06:15 +00003459 prom_printf("copying OF device tree...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003460 flatten_device_tree();
3461
Paul Mackerras3825ac02005-11-08 22:48:08 +11003462 /*
3463 * in case stdin is USB and still active on IBM machines...
3464 * Unfortunately quiesce crashes on some powermacs if we have
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003465 * closed stdin already (in particular the powerbook 101).
Paul Mackerras3825ac02005-11-08 22:48:08 +11003466 */
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003467 if (of_platform != PLATFORM_POWERMAC)
Paul Mackerras3825ac02005-11-08 22:48:08 +11003468 prom_close_stdin();
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003469
3470 /*
3471 * Call OF "quiesce" method to shut down pending DMA's from
3472 * devices etc...
3473 */
Michael Ellerman7e862d72015-03-30 17:38:09 +11003474 prom_printf("Quiescing Open Firmware ...\n");
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003475 call_prom("quiesce", 0, 0);
3476
3477 /*
3478 * And finally, call the kernel passing it the flattened device
3479 * tree and NULL as r5, thus triggering the new entry point which
3480 * is common to us and kexec
3481 */
Anton Blanchard5827d412012-11-26 17:40:03 +00003482 hdr = dt_header_start;
Benjamin Herrenschmidt40dfef62011-11-29 18:22:56 +00003483
Benjamin Herrenschmidt11fdb302018-10-15 13:49:53 +11003484 prom_printf("Booting Linux via __start() @ 0x%lx ...\n", kbase);
3485 prom_debug("->dt_header_start=0x%lx\n", hdr);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003486
3487#ifdef CONFIG_PPC32
3488 reloc_got2(-offset);
3489#endif
3490
Ram Pai6a9c9302019-08-19 23:13:14 -03003491 /* Move to secure memory if we're supposed to be secure guests. */
3492 setup_secure_guest(kbase, hdr);
3493
Benjamin Herrenschmidt6e35d5d2011-09-19 18:28:01 +00003494 __start(hdr, kbase, 0, 0, 0, 0, 0);
Paul Mackerras9b6b5632005-10-06 12:06:20 +10003495
3496 return 0;
3497}