kernel-entry-init.h 3.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * 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) 2005-2008 Cavium Networks, Inc
 */
#ifndef __ASM_MACH_CAVIUM_OCTEON_KERNEL_ENTRY_H
#define __ASM_MACH_CAVIUM_OCTEON_KERNEL_ENTRY_H

#define CP0_CVMCTL_REG $9, 7
#define CP0_CVMMEMCTL_REG $11,7
#define CP0_PRID_REG $15, 0
14
#define CP0_DCACHE_ERR_REG $27, 1
15 16 17
#define CP0_PRID_OCTEON_PASS1 0x000d0000
#define CP0_PRID_OCTEON_CN30XX 0x000d0200

R
Ralf Baechle 已提交
18
.macro	kernel_entry_setup
19 20 21 22 23 24 25 26 27 28 29
	# Registers set by bootloader:
	# (only 32 bits set by bootloader, all addresses are physical
	# addresses, and need to have the appropriate memory region set
	# by the kernel
	# a0 = argc
	# a1 = argv (kseg0 compat addr)
	# a2 = 1 if init core, zero otherwise
	# a3 = address of boot descriptor block
	.set push
	.set arch=octeon
	# Read the cavium mem control register
R
Ralf Baechle 已提交
30
	dmfc0	v0, CP0_CVMMEMCTL_REG
31
	# Clear the lower 6 bits, the CVMSEG size
R
Ralf Baechle 已提交
32 33 34 35
	dins	v0, $0, 0, 6
	ori	v0, CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE
	dmtc0	v0, CP0_CVMMEMCTL_REG	# Write the cavium mem control register
	dmfc0	v0, CP0_CVMCTL_REG	# Read the cavium control register
36
	# Disable unaligned load/store support but leave HW fixup enabled
37
	# Needed for octeon specific memcpy
38 39
	or  v0, v0, 0x5001
	xor v0, v0, 0x1001
40 41
	# First clear off CvmCtl[IPPCI] bit and move the performance
	# counters interrupt to IRQ 6
42
	dli	v1, ~(7 << 7)
43 44
	and	v0, v0, v1
	ori	v0, v0, (6 << 7)
45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66

	mfc0	v1, CP0_PRID_REG
	and	t1, v1, 0xfff8
	xor	t1, t1, 0x9000		# 63-P1
	beqz	t1, 4f
	and	t1, v1, 0xfff8
	xor	t1, t1, 0x9008		# 63-P2
	beqz	t1, 4f
	and	t1, v1, 0xfff8
	xor	t1, t1, 0x9100		# 68-P1
	beqz	t1, 4f
	and	t1, v1, 0xff00
	xor	t1, t1, 0x9200		# 66-PX
	bnez	t1, 5f			# Skip WAR for others.
	and	t1, v1, 0x00ff
	slti	t1, t1, 2		# 66-P1.2 and later good.
	beqz	t1, 5f

4:	# core-16057 work around
	or	v0, v0, 0x2000		# Set IPREF bit.

5:	# No core-16057 work around
67
	# Write the cavium control register
R
Ralf Baechle 已提交
68
	dmtc0	v0, CP0_CVMCTL_REG
69 70
	sync
	# Flush dcache after config change
R
Ralf Baechle 已提交
71
	cache	9, 0($0)
72 73 74 75 76 77 78 79 80 81 82
	# Zero all of CVMSEG to make sure parity is correct
	dli	v0, CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE
	dsll	v0, 7
	beqz	v0, 2f
1:	dsubu	v0, 8
	sd	$0, -32768(v0)
	bnez	v0, 1b
2:
	mfc0	v0, CP0_PRID_REG
	bbit0	v0, 15, 1f
	# OCTEON II or better have bit 15 set.  Clear the error bits.
83 84 85
	and	t1, v0, 0xff00
	dli	v0, 0x9500
	bge	t1, v0, 1f  # OCTEON III has no DCACHE_ERR_REG COP0
86 87 88
	dli	v0, 0x27
	dmtc0	v0, CP0_DCACHE_ERR_REG
1:
89
	# Get my core id
R
Ralf Baechle 已提交
90
	rdhwr	v0, $0
91
	# Jump the master to kernel_entry
R
Ralf Baechle 已提交
92
	bne	a2, zero, octeon_main_processor
93 94 95 96 97 98 99 100 101 102
	nop

#ifdef CONFIG_SMP

	#
	# All cores other than the master need to wait here for SMP bootstrap
	# to begin
	#

octeon_spin_wait_boot:
S
Steven J. Hill 已提交
103 104 105 106 107 108 109 110 111 112 113 114 115
#ifdef CONFIG_RELOCATABLE
	PTR_LA	t0, octeon_processor_relocated_kernel_entry
	LONG_L	t0, (t0)
	beq	zero, t0, 1f
	nop

	jr	t0
	nop
1:
#endif /* CONFIG_RELOCATABLE */

	# This is the variable where the next core to boot is stored
	PTR_LA	t0, octeon_processor_boot
116
	# Get the core id of the next to be booted
R
Ralf Baechle 已提交
117
	LONG_L	t1, (t0)
118 119 120 121
	# Keep looping if it isn't me
	bne t1, v0, octeon_spin_wait_boot
	nop
	# Get my GP from the global variable
R
Ralf Baechle 已提交
122 123
	PTR_LA	t0, octeon_processor_gp
	LONG_L	gp, (t0)
124
	# Get my SP from the global variable
R
Ralf Baechle 已提交
125 126
	PTR_LA	t0, octeon_processor_sp
	LONG_L	sp, (t0)
127
	# Set the SP global variable to zero so the master knows we've started
R
Ralf Baechle 已提交
128
	LONG_S	zero, (t0)
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
#ifdef __OCTEON__
	syncw
	syncw
#else
	sync
#endif
	# Jump to the normal Linux SMP entry point
	j   smp_bootstrap
	nop
#else /* CONFIG_SMP */

	#
	# Someone tried to boot SMP with a non SMP kernel. All extra cores
	# will halt here.
	#
octeon_wait_forever:
	wait
	b   octeon_wait_forever
	nop

#endif /* CONFIG_SMP */
octeon_main_processor:
	.set pop
.endm

/*
155
 * Do SMP slave processor setup necessary before we can safely execute C code.
156
 */
R
Ralf Baechle 已提交
157
	.macro	smp_slave_setup
158 159 160
	.endm

#endif /* __ASM_MACH_CAVIUM_OCTEON_KERNEL_ENTRY_H */