summaryrefslogtreecommitdiffstats
path: root/arch/mips/kernel/setup.c
blob: 44d0f0dafa62fe93985b3e64afa8225a043566ed (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
/*
 * This file is subject to the terms and conditions of the GNU General Public
 * License.  See the file "COPYING" in the main directory of this archive
 * for more details.
 *
 * Copyright (C) 1995  Linus Torvalds
 * Copyright (C) 1995, 1996, 1997, 1998, 1999, 2000  Ralf Baechle
 * Copyright (C) 1996  Stoned Elipot
 */
#include <linux/config.h>
#include <linux/errno.h>
#include <linux/hdreg.h>
#include <linux/init.h>
#include <linux/ioport.h>
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/stddef.h>
#include <linux/string.h>
#include <linux/unistd.h>
#include <linux/ptrace.h>
#include <linux/malloc.h>
#include <linux/user.h>
#include <linux/utsname.h>
#include <linux/a.out.h>
#include <linux/tty.h>
#include <linux/bootmem.h>
#ifdef CONFIG_BLK_DEV_RAM
#include <linux/blk.h>
#endif
#include <linux/ide.h>
#ifdef CONFIG_RTC
#include <linux/timex.h>
#endif

#include <asm/asm.h>
#include <asm/bootinfo.h>
#include <asm/cachectl.h>
#include <asm/cpu.h>
#include <asm/io.h>
#include <asm/stackframe.h>
#include <asm/system.h>
#include <asm/cpu.h>
#ifdef CONFIG_SGI_IP22
#include <asm/sgialib.h>
#endif

struct mips_cpuinfo boot_cpu_data = { NULL, NULL, 0 };

/*
 * Not all of the MIPS CPUs have the "wait" instruction available. Moreover,
 * the implementation of the "wait" feature differs between CPU families. This
 * points to the function that implements CPU specific wait. 
 * The wait instruction stops the pipeline and reduces the power consumption of
 * the CPU very much.
 */
void (*cpu_wait)(void) = NULL;

/*
 * Do we have a cyclecounter available?
 */
char cyclecounter_available;

/*
 * There are several bus types available for MIPS machines.  "RISC PC"
 * type machines have ISA, EISA, VLB or PCI available, DECstations
 * have Turbochannel or Q-Bus, SGI has GIO, there are lots of VME
 * boxes ...
 * This flag is set if a EISA slots are available.
 */
int EISA_bus = 0;

struct screen_info screen_info;

#ifdef CONFIG_BLK_DEV_FD
extern struct fd_ops no_fd_ops;
struct fd_ops *fd_ops;
#endif

#if defined(CONFIG_BLK_DEV_IDE) || defined(CONFIG_BLK_DEV_IDE_MODULE)
extern struct ide_ops no_ide_ops;
struct ide_ops *ide_ops;
#endif

extern struct rtc_ops no_rtc_ops;
struct rtc_ops *rtc_ops;

#ifdef CONFIG_PC_KEYB
extern struct kbd_ops no_kbd_ops;
struct kbd_ops *kbd_ops;
#endif

/*
 * Setup information
 *
 * These are initialized so they are in the .data section
 */
unsigned long mips_memory_upper = KSEG0; /* this is set by kernel_entry() */
unsigned long mips_machtype = MACH_UNKNOWN;
unsigned long mips_machgroup = MACH_GROUP_UNKNOWN;

unsigned char aux_device_present;
extern int _end;

static char command_line[CL_SIZE] = { 0, };
       char saved_command_line[CL_SIZE];
extern char arcs_cmdline[CL_SIZE];

/*
 * The board specific setup routine sets irq_setup to point to a board
 * specific setup routine.
 */
void (*irq_setup)(void);

/*
 * mips_io_port_base is the begin of the address space to which x86 style
 * I/O ports are mapped.
 */
unsigned long mips_io_port_base;

/*
 * isa_slot_offset is the address where E(ISA) busaddress 0 is is mapped
 * for the processor.
 */
unsigned long isa_slot_offset;

extern void sgi_sysinit(void);
extern void SetUpBootInfo(void);
extern void loadmmu(void);
extern asmlinkage void start_kernel(void);
extern int prom_init(int, char **, char **, int *);

/*
 * Probe whether cpu has config register by trying to play with
 * alternate cache bit and see whether it matters.
 * It's used by cpu_probe to distinguish between R3000A and R3081.
 */
static inline int cpu_has_confreg(void)
{
#ifdef CONFIG_CPU_R3000
	extern unsigned long r3k_cache_size(unsigned long);
	unsigned long size1, size2; 
	unsigned long cfg = read_32bit_cp0_register(CP0_CONF);

	size1 = r3k_cache_size(ST0_DE);
	write_32bit_cp0_register(CP0_CONF, cfg^CONF_AC);
	size2 = r3k_cache_size(ST0_DE);
	write_32bit_cp0_register(CP0_CONF, cfg);
	return size1 != size2;
#else
	return 0;
#endif
}

/* declaration of the global struct */
struct mips_cpu mips_cpu = {PRID_IMP_UNKNOWN, CPU_UNKNOWN, 0, 0, 0,
			    {0,0,0,0}, {0,0,0,0}, {0,0,0,0}, {0,0,0,0}};

/* Shortcut for assembler access to mips_cpu.options */
int *cpuoptions = &mips_cpu.options;

#define R4K_OPTS (MIPS_CPU_TLB | MIPS_CPU_4KEX | MIPS_CPU_4KTLB \
		| MIPS_CPU_COUNTER | MIPS_CPU_CACHE_CDEX)

static inline void cpu_probe(void)
{
	unsigned long config1;

	mips_cpu.processor_id = read_32bit_cp0_register(CP0_PRID);
	switch (mips_cpu.processor_id & 0xff00) {
	case PRID_IMP_R2000:
		mips_cpu.cputype = CPU_R2000;
		mips_cpu.isa_level = MIPS_CPU_ISA_I;
		mips_cpu.options = MIPS_CPU_TLB;
		mips_cpu.tlbsize = 32;
		break;
	case PRID_IMP_R3000:
		if ((mips_cpu.processor_id & 0xff) == PRID_REV_R3000A)
			if (cpu_has_confreg())
				mips_cpu.cputype = CPU_R3081E;
			else
				mips_cpu.cputype = CPU_R3000A;
		else
			 mips_cpu.cputype = CPU_R3000;
		mips_cpu.isa_level = MIPS_CPU_ISA_I;
		mips_cpu.options = MIPS_CPU_TLB;
		mips_cpu.tlbsize = 32;
		break;
	case PRID_IMP_R4000:
		if ((mips_cpu.processor_id & 0xff) == PRID_REV_R4400)
			mips_cpu.cputype = CPU_R4400SC;
		else
			mips_cpu.cputype = CPU_R4000SC;
		mips_cpu.isa_level = MIPS_CPU_ISA_III;
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU | MIPS_CPU_32FPR |
		                   MIPS_CPU_WATCH | MIPS_CPU_VCE;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R4600:
		mips_cpu.cputype = CPU_R4600;
		mips_cpu.isa_level = MIPS_CPU_ISA_III;
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R4650:
		mips_cpu.cputype = CPU_R4650;
		mips_cpu.isa_level = MIPS_CPU_ISA_III;
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R4700:
		mips_cpu.cputype = CPU_R4700;
		mips_cpu.isa_level = MIPS_CPU_ISA_III;
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU | MIPS_CPU_32FPR;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R5000:
		mips_cpu.cputype = CPU_R5000;
		mips_cpu.isa_level = MIPS_CPU_ISA_IV; 
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU | MIPS_CPU_32FPR;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R5432:
	        mips_cpu.cputype = CPU_R5432;
		mips_cpu.isa_level = MIPS_CPU_ISA_IV; 
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU | MIPS_CPU_32FPR;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_NEVADA:
		mips_cpu.cputype = CPU_NEVADA;
		mips_cpu.isa_level = MIPS_CPU_ISA_IV; 
		mips_cpu.options = R4K_OPTS | MIPS_CPU_FPU | MIPS_CPU_32FPR | 
		                   MIPS_CPU_DIVEC;
		mips_cpu.tlbsize = 48;
		break;
	case PRID_IMP_R6000:
		mips_cpu.cputype = CPU_R6000;
		mips_cpu.isa_level = MIPS_CPU_ISA_II;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_FPU;
		mips_cpu.tlbsize = 32;
		break;
	case PRID_IMP_R6000A:
		mips_cpu.cputype = CPU_R6000A;
		mips_cpu.isa_level = MIPS_CPU_ISA_II;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_FPU;
		mips_cpu.tlbsize = 32;
		break;
	case PRID_IMP_RM7000:
		mips_cpu.cputype = CPU_RM7000;
	        mips_cpu.isa_level = MIPS_CPU_ISA_IV;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_4KEX |
		                   MIPS_CPU_FPU | MIPS_CPU_32FPR |
		                   MIPS_CPU_COUNTER;
		break;
	case PRID_IMP_R8000:
		mips_cpu.cputype = CPU_R8000;
	        mips_cpu.isa_level = MIPS_CPU_ISA_IV;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_4KEX |
		                   MIPS_CPU_FPU | MIPS_CPU_32FPR;
		mips_cpu.tlbsize = 384;      /* has wierd TLB: 3-way x 128 */
		break;
	case PRID_IMP_R10000:
		mips_cpu.cputype = CPU_R10000;
		mips_cpu.cputype = MIPS_CPU_ISA_IV;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_4KEX | 
		                   MIPS_CPU_FPU | MIPS_CPU_32FPR | 
		                   MIPS_CPU_COUNTER | MIPS_CPU_WATCH;
		mips_cpu.tlbsize = 64;
		break;
#ifdef CONFIG_CPU_MIPS32
	case PRID_IMP_4KC:
		mips_cpu.cputype = CPU_4KC;
		mips_cpu.cputype = MIPS_CPU_ISA_M32;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_4KEX | 
		                   MIPS_CPU_4KTLB | MIPS_CPU_COUNTER | 
		                   MIPS_CPU_DIVEC | MIPS_CPU_WATCH;
		config1 = read_mips32_cp0_config1();
		if (config1 & (1 << 3))
		        mips_cpu.options |= MIPS_CPU_WATCH;
		if (config1 & (1 << 2))
		        mips_cpu.options |= MIPS_CPU_MIPS16;
		if (config1 & 1)
		        mips_cpu.options |= MIPS_CPU_FPU;
		break;
	case PRID_IMP_5KC:
		mips_cpu.cputype = CPU_5KC;
		mips_cpu.cputype = MIPS_CPU_ISA_M64;
		mips_cpu.options = MIPS_CPU_TLB | MIPS_CPU_4KEX | 
		                   MIPS_CPU_4KTLB | MIPS_CPU_COUNTER | 
		                   MIPS_CPU_DIVEC | MIPS_CPU_WATCH;
		config1 = read_mips32_cp0_config1();
		if (config1 & (1 << 3))
		        mips_cpu.options |= MIPS_CPU_WATCH;
		if (config1 & (1 << 2))
		        mips_cpu.options |= MIPS_CPU_MIPS16;
		if (config1 & 1)
		        mips_cpu.options |= MIPS_CPU_FPU;
		break;
#endif
	default:
		mips_cpu.cputype = CPU_UNKNOWN;
	}
}

asmlinkage void __init init_arch(int argc, char **argv, char **envp, int *prom_vec)
{
	unsigned int s;

	/* Determine which MIPS variant we are running on. */
	cpu_probe();

	prom_init(argc, argv, envp, prom_vec);

#ifdef CONFIG_SGI_IP22
	sgi_sysinit();
#endif

	/*
	 * Determine the mmu/cache attached to this machine,
	 * then flush the tlb and caches.  On the r4xx0
	 * variants this also sets CP0_WIRED to zero.
	 */
	loadmmu();

	/* Disable coprocessors and set FPU for 16 FPRs */
	s = read_32bit_cp0_register(CP0_STATUS);
	s &= ~(ST0_CU1|ST0_CU2|ST0_CU3|ST0_KX|ST0_SX|ST0_FR);
	s |= ST0_CU0;
	write_32bit_cp0_register(CP0_STATUS, s);

	/*
	 * Main should never return here, but
	 * just in case, we know what happens.
	 */
	for(;;)
		start_kernel();
}

static void __init default_irq_setup(void)
{
	panic("Unknown machtype in init_IRQ");
}

void __init setup_arch(char **cmdline_p)
{
	void baget_setup(void);
	void decstation_setup(void);
	void deskstation_setup(void);
	void jazz_setup(void);
	void sni_rm200_pci_setup(void);
	void sgi_setup(void);
	void ddb_setup(void);
	void orion_setup(void);
        void ev96100_setup(void);
	void atlas_setup(void);
	void malta_setup(void);

	/* Save defaults for configuration-dependent routines.  */
	irq_setup = default_irq_setup;

#ifdef CONFIG_BLK_DEV_FD
	fd_ops = &no_fd_ops;
#endif

#ifdef CONFIG_BLK_DEV_IDE
	ide_ops = &no_ide_ops;
#endif

#ifdef CONFIG_PC_KEYB
	kbd_ops = &no_kbd_ops;
#endif
	
	rtc_ops = &no_rtc_ops;

	switch(mips_machgroup)
	{
#ifdef CONFIG_BAGET_MIPS
	case MACH_GROUP_BAGET: 
		baget_setup();
		break;
#endif
#ifdef CONFIG_DECSTATION
	case MACH_GROUP_DEC:
		decstation_setup();
		break;
#endif
#ifdef CONFIG_MIPS_ATLAS
	case MACH_GROUP_UNKNOWN:
		atlas_setup();
		break;
#endif
#ifdef CONFIG_MIPS_JAZZ
	case MACH_GROUP_JAZZ:
		jazz_setup();
		break;
#endif
#ifdef CONFIG_MIPS_MALTA
	case MACH_GROUP_UNKNOWN:
		malta_setup();
		break;
#endif
#ifdef CONFIG_SGI_IP22
	/* As of now this is only IP22.  */
	case MACH_GROUP_SGI:
		sgi_setup();
		break;
#endif
#ifdef CONFIG_SNI_RM200_PCI
	case MACH_GROUP_SNI_RM:
		sni_rm200_pci_setup();
		break;
#endif
#ifdef CONFIG_DDB5074
	case MACH_GROUP_NEC_DDB:
		ddb_setup();
		break;
#endif
#ifdef CONFIG_DDB5476
       case MACH_GROUP_NEC_DDB:
               ddb_setup();
               break;
#endif
#ifdef CONFIG_ORION
	case MACH_GROUP_ORION:
		orion_setup();
		break;
#endif
#ifdef CONFIG_MIPS_EV96100
	case MACH_GROUP_GALILEO:
		ev96100_setup();
		break;
#endif
#ifdef CONFIG_MIPS_EV64120
	case MACH_GROUP_GALILEO:
		ev64120_setup();
		break;
#endif
	default:
		panic("Unsupported architecture");
	}

        strncpy (command_line, arcs_cmdline, CL_SIZE);
	memcpy(saved_command_line, command_line, CL_SIZE);
	saved_command_line[CL_SIZE-1] = '\0';

	*cmdline_p = command_line;

#if 0
#ifdef CONFIG_BLK_DEV_INITRD
#error "Fixme, I'm broken."
	tmp = (((unsigned long)&_end + PAGE_SIZE-1) & PAGE_MASK) - 8;
	if (tmp < (unsigned long)&_end)
		tmp += PAGE_SIZE;
	initrd_header = (unsigned long *)tmp;
	if (initrd_header[0] == 0x494E5244) {
		initrd_start = (unsigned long)&initrd_header[2];
		initrd_end = initrd_start + initrd_header[1];
		initrd_below_start_ok = 1;
		if (initrd_end > memory_end) {
			printk("initrd extends beyond end of memory "
			       "(0x%08lx > 0x%08lx)\ndisabling initrd\n",
			       initrd_end,memory_end);
			initrd_start = 0;
		} else
			*memory_start_p = initrd_end;
	}
#endif /* CONFIG_BLK_DEV_INITRD  */
#endif /* 0  */

	paging_init();
}

void r3081_wait(void) 
{
	unsigned long cfg = read_32bit_cp0_register(CP0_CONF);
	write_32bit_cp0_register(CP0_CONF, cfg|CONF_HALT);
}

void r4k_wait(void)
{
	__asm__(".set\tmips3\n\t"
		"wait\n\t"
		".set\tmips0");
}