powerpc updates for 6.5
- Extend KCSAN support to 32-bit and BookE. Add some KCSAN annotations. - Make ELFv2 ABI the default for 64-bit big-endian kernel builds, and use the -mprofile-kernel option (kernel specific ftrace ABI) for big endian ELFv2 kernels. - Add initial Dynamic Execution Control Register (DEXCR) support, and allow the ROP protection instructions to be used on Power 10. - Various other small features and fixes. Thanks to: Aditya Gupta, Aneesh Kumar K.V, Benjamin Gray, Brian King, Christophe Leroy, Colin Ian King, Dmitry Torokhov, Gaurav Batra, Jean Delvare, Joel Stanley, Marco Elver, Masahiro Yamada, Nageswara R Sastry, Nathan Chancellor, Naveen N Rao, Nayna Jain, Nicholas Piggin, Paul Gortmaker, Randy Dunlap, Rob Herring, Rohan McLure, Russell Currey, Sachin Sant, Timothy Pearson, Tom Rix, Uwe Kleine-König. -----BEGIN PGP SIGNATURE----- iQJHBAABCAAxFiEEJFGtCPCthwEv2Y/bUevqPMjhpYAFAmSeqQMTHG1wZUBlbGxl cm1hbi5pZC5hdQAKCRBR6+o8yOGlgKukD/sGUceX6gIc7UcjWhL1ZCVco0bsgLjT JrY1NenisGKjwwRd/o+2+h3ziJDoO5AsQfT72EiNLhaYJhnlb1d0vXzsvN0THc+2 W5RrxAZUNhBy+c7gSSEJjy8+vBIwSQAliQLChHGOSejGCj94SxF5+zjUFvSX458I z0+ZQK+Fiw5NcpzEnBT0XPnLzap74a7TL0JcG1MLbj2QtHXhbfjIlkkPDX3kK0Gw xbelFy38X7KKbQsXXYSTCGqwRdJ3yqu21nEsjRuo2yT5H5rQbjCNggkMOL1DecDd ULGxit/z13Pt1Ad3oe+6FF17ggOiCG0F75DONASjFDthFYx6NQffkJS1n1VZauQj jU6LtWeD3HkgIYm6Udjq+LaSmkAmn5a+9tsElE/K+V1WG4rKyMeVmE3z/tCJG0l2 yhLKyFs+glXN/LiWHyX0mrQIIVZdRK237X1qXJuIvAuB7Drm5duXFAHR8pdJD0dg H23OhoO2FvLxb9GvnzGxqjdazzattctz31wU/1RgnPxumYkJ9PlBcXn9h1uXa8/K rDZFJADsQhEfRCjmLG3GIaFWqZdc4Cn+ZUk4iHkjPDFDL05Fq7JYHIuwteiN6/wP NHRvtKdJisu583NI9RN9300JykrEqjSRbMOWlc3vuFwbRLGioXvWhWlIZ3/t58jG R8s+f0nKSPr+fg== =ssit -----END PGP SIGNATURE----- Merge tag 'powerpc-6.5-1' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux Pull powerpc updates from Michael Ellerman: - Extend KCSAN support to 32-bit and BookE. Add some KCSAN annotations - Make ELFv2 ABI the default for 64-bit big-endian kernel builds, and use the -mprofile-kernel option (kernel specific ftrace ABI) for big endian ELFv2 kernels - Add initial Dynamic Execution Control Register (DEXCR) support, and allow the ROP protection instructions to be used on Power 10 - Various other small features and fixes Thanks to Aditya Gupta, Aneesh Kumar K.V, Benjamin Gray, Brian King, Christophe Leroy, Colin Ian King, Dmitry Torokhov, Gaurav Batra, Jean Delvare, Joel Stanley, Marco Elver, Masahiro Yamada, Nageswara R Sastry, Nathan Chancellor, Naveen N Rao, Nayna Jain, Nicholas Piggin, Paul Gortmaker, Randy Dunlap, Rob Herring, Rohan McLure, Russell Currey, Sachin Sant, Timothy Pearson, Tom Rix, and Uwe Kleine-König. * tag 'powerpc-6.5-1' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux: (76 commits) powerpc: remove checks for binutils older than 2.25 powerpc: Fail build if using recordmcount with binutils v2.37 powerpc/iommu: TCEs are incorrectly manipulated with DLPAR add/remove of memory powerpc/iommu: Only build sPAPR access functions on pSeries powerpc: powernv: Annotate data races in opal events powerpc: Mark writes registering ipi to host cpu through kvm and polling powerpc: Annotate accesses to ipi message flags powerpc: powernv: Fix KCSAN datarace warnings on idle_state contention powerpc: Mark [h]ssr_valid accesses in check_return_regs_valid powerpc: qspinlock: Enforce qnode writes prior to publishing to queue powerpc: qspinlock: Mark accesses to qnode lock checks powerpc/powernv/pci: Remove last IODA1 defines powerpc/powernv/pci: Remove MVE code powerpc/powernv/pci: Remove ioda1 support powerpc: 52xx: Make immr_id DT match tables static powerpc: mpc512x: Remove open coded "ranges" parsing powerpc: fsl_soc: Use of_range_to_resource() for "ranges" parsing powerpc: fsl: Use of_property_read_reg() to parse "reg" powerpc: fsl_rio: Use of_range_to_resource() for "ranges" parsing macintosh: Use of_property_read_reg() to parse "reg" ...
This commit is contained in:
commit
d8b0bd57c2
58
Documentation/powerpc/dexcr.rst
Normal file
58
Documentation/powerpc/dexcr.rst
Normal file
@ -0,0 +1,58 @@
|
||||
.. SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
==========================================
|
||||
DEXCR (Dynamic Execution Control Register)
|
||||
==========================================
|
||||
|
||||
Overview
|
||||
========
|
||||
|
||||
The DEXCR is a privileged special purpose register (SPR) introduced in
|
||||
PowerPC ISA 3.1B (Power10) that allows per-cpu control over several dynamic
|
||||
execution behaviours. These behaviours include speculation (e.g., indirect
|
||||
branch target prediction) and enabling return-oriented programming (ROP)
|
||||
protection instructions.
|
||||
|
||||
The execution control is exposed in hardware as up to 32 bits ('aspects') in
|
||||
the DEXCR. Each aspect controls a certain behaviour, and can be set or cleared
|
||||
to enable/disable the aspect. There are several variants of the DEXCR for
|
||||
different purposes:
|
||||
|
||||
DEXCR
|
||||
A privileged SPR that can control aspects for userspace and kernel space
|
||||
HDEXCR
|
||||
A hypervisor-privileged SPR that can control aspects for the hypervisor and
|
||||
enforce aspects for the kernel and userspace.
|
||||
UDEXCR
|
||||
An optional ultravisor-privileged SPR that can control aspects for the ultravisor.
|
||||
|
||||
Userspace can examine the current DEXCR state using a dedicated SPR that
|
||||
provides a non-privileged read-only view of the userspace DEXCR aspects.
|
||||
There is also an SPR that provides a read-only view of the hypervisor enforced
|
||||
aspects, which ORed with the userspace DEXCR view gives the effective DEXCR
|
||||
state for a process.
|
||||
|
||||
|
||||
Configuration
|
||||
=============
|
||||
|
||||
The DEXCR is currently unconfigurable. All threads are run with the
|
||||
NPHIE aspect enabled.
|
||||
|
||||
|
||||
coredump and ptrace
|
||||
===================
|
||||
|
||||
The userspace values of the DEXCR and HDEXCR (in this order) are exposed under
|
||||
``NT_PPC_DEXCR``. These are each 64 bits and readonly, and are intended to
|
||||
assist with core dumps. The DEXCR may be made writable in future. The top 32
|
||||
bits of both registers (corresponding to the non-userspace bits) are masked off.
|
||||
|
||||
If the kernel config ``CONFIG_CHECKPOINT_RESTORE`` is enabled, then
|
||||
``NT_PPC_HASHKEYR`` is available and exposes the HASHKEYR value of the process
|
||||
for reading and writing. This is a tradeoff between increased security and
|
||||
checkpoint/restore support: a process should normally have no need to know its
|
||||
secret key, but restoring a process requires setting its original key. The key
|
||||
therefore appears in core dumps, and an attacker may be able to retrieve it from
|
||||
a coredump and effectively bypass ROP protection on any threads that share this
|
||||
key (potentially all threads from the same parent that have not run ``exec()``).
|
@ -15,6 +15,7 @@ powerpc
|
||||
cxl
|
||||
cxlflash
|
||||
dawr-power9
|
||||
dexcr
|
||||
dscr
|
||||
eeh-pci-error-recovery
|
||||
elf_hwcaps
|
||||
|
11
MAINTAINERS
11
MAINTAINERS
@ -11448,7 +11448,13 @@ F: arch/mips/include/uapi/asm/kvm*
|
||||
F: arch/mips/kvm/
|
||||
|
||||
KERNEL VIRTUAL MACHINE FOR POWERPC (KVM/powerpc)
|
||||
M: Michael Ellerman <mpe@ellerman.id.au>
|
||||
R: Nicholas Piggin <npiggin@gmail.com>
|
||||
L: linuxppc-dev@lists.ozlabs.org
|
||||
L: kvm@vger.kernel.org
|
||||
S: Maintained (Book3S 64-bit HV)
|
||||
S: Odd fixes (Book3S 64-bit PR)
|
||||
S: Orphan (Book3E and 32-bit)
|
||||
T: git git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux.git topic/ppc-kvm
|
||||
F: arch/powerpc/include/asm/kvm*
|
||||
F: arch/powerpc/include/uapi/asm/kvm*
|
||||
@ -11981,11 +11987,12 @@ F: lib/linear_ranges.c
|
||||
F: lib/test_linear_ranges.c
|
||||
|
||||
LINUX FOR POWER MACINTOSH
|
||||
M: Benjamin Herrenschmidt <benh@kernel.crashing.org>
|
||||
L: linuxppc-dev@lists.ozlabs.org
|
||||
S: Odd Fixes
|
||||
S: Orphan
|
||||
F: arch/powerpc/platforms/powermac/
|
||||
F: drivers/macintosh/
|
||||
X: drivers/macintosh/adb-iop.c
|
||||
X: drivers/macintosh/via-macii.c
|
||||
|
||||
LINUX FOR POWERPC (32-BIT AND 64-BIT)
|
||||
M: Michael Ellerman <mpe@ellerman.id.au>
|
||||
|
@ -210,7 +210,7 @@ config PPC
|
||||
select HAVE_ARCH_KASAN if PPC_RADIX_MMU
|
||||
select HAVE_ARCH_KASAN if PPC_BOOK3E_64
|
||||
select HAVE_ARCH_KASAN_VMALLOC if HAVE_ARCH_KASAN
|
||||
select HAVE_ARCH_KCSAN if PPC_BOOK3S_64
|
||||
select HAVE_ARCH_KCSAN
|
||||
select HAVE_ARCH_KFENCE if ARCH_SUPPORTS_DEBUG_PAGEALLOC
|
||||
select HAVE_ARCH_RANDOMIZE_KSTACK_OFFSET
|
||||
select HAVE_ARCH_WITHIN_STACK_FRAMES
|
||||
@ -235,7 +235,7 @@ config PPC
|
||||
select HAVE_FUNCTION_DESCRIPTORS if PPC64_ELF_ABI_V1
|
||||
select HAVE_FUNCTION_ERROR_INJECTION
|
||||
select HAVE_FUNCTION_GRAPH_TRACER
|
||||
select HAVE_FUNCTION_TRACER
|
||||
select HAVE_FUNCTION_TRACER if PPC64 || (PPC32 && CC_IS_GCC)
|
||||
select HAVE_GCC_PLUGINS if GCC_VERSION >= 50200 # plugin support on gcc <= 5.1 is buggy on PPC
|
||||
select HAVE_GENERIC_VDSO
|
||||
select HAVE_HARDLOCKUP_DETECTOR_ARCH if PPC_BOOK3S_64 && SMP
|
||||
@ -547,8 +547,9 @@ config LD_HEAD_STUB_CATCH
|
||||
If unsure, say "N".
|
||||
|
||||
config MPROFILE_KERNEL
|
||||
depends on PPC64 && CPU_LITTLE_ENDIAN && FUNCTION_TRACER
|
||||
def_bool $(success,$(srctree)/arch/powerpc/tools/gcc-check-mprofile-kernel.sh $(CC) -I$(srctree)/include -D__KERNEL__)
|
||||
depends on PPC64_ELF_ABI_V2 && FUNCTION_TRACER
|
||||
def_bool $(success,$(srctree)/arch/powerpc/tools/gcc-check-mprofile-kernel.sh $(CC) -mlittle-endian) if CPU_LITTLE_ENDIAN
|
||||
def_bool $(success,$(srctree)/arch/powerpc/tools/gcc-check-mprofile-kernel.sh $(CC) -mbig-endian) if CPU_BIG_ENDIAN
|
||||
|
||||
config HOTPLUG_CPU
|
||||
bool "Support for enabling/disabling CPUs"
|
||||
@ -624,10 +625,12 @@ config ARCH_HAS_KEXEC_PURGATORY
|
||||
def_bool KEXEC_FILE
|
||||
|
||||
config PPC64_BIG_ENDIAN_ELF_ABI_V2
|
||||
bool "Build big-endian kernel using ELF ABI V2 (EXPERIMENTAL)"
|
||||
# Option is available to BFD, but LLD does not support ELFv1 so this is
|
||||
# always true there.
|
||||
prompt "Build big-endian kernel using ELF ABI V2" if LD_IS_BFD && EXPERT
|
||||
def_bool y
|
||||
depends on PPC64 && CPU_BIG_ENDIAN
|
||||
depends on CC_HAS_ELFV2
|
||||
depends on LD_VERSION >= 22400 || LLD_VERSION >= 150000
|
||||
help
|
||||
This builds the kernel image using the "Power Architecture 64-Bit ELF
|
||||
V2 ABI Specification", which has a reduced stack overhead and faster
|
||||
@ -638,8 +641,6 @@ config PPC64_BIG_ENDIAN_ELF_ABI_V2
|
||||
it is less well tested by kernel and toolchain. However some distros
|
||||
build userspace this way, and it can produce a functioning kernel.
|
||||
|
||||
This requires GCC and binutils 2.24 or newer.
|
||||
|
||||
config RELOCATABLE
|
||||
bool "Build a relocatable kernel"
|
||||
depends on PPC64 || (FLATMEM && (44x || PPC_85xx))
|
||||
|
@ -42,18 +42,13 @@ machine-$(CONFIG_PPC64) += 64
|
||||
machine-$(CONFIG_CPU_LITTLE_ENDIAN) += le
|
||||
UTS_MACHINE := $(subst $(space),,$(machine-y))
|
||||
|
||||
# XXX This needs to be before we override LD below
|
||||
ifdef CONFIG_PPC32
|
||||
KBUILD_LDFLAGS_MODULE += arch/powerpc/lib/crtsavres.o
|
||||
else
|
||||
ifeq ($(call ld-ifversion, -ge, 22500, y),y)
|
||||
ifeq ($(CONFIG_PPC64)$(CONFIG_LD_IS_BFD),yy)
|
||||
# Have the linker provide sfpr if possible.
|
||||
# There is a corresponding test in arch/powerpc/lib/Makefile
|
||||
KBUILD_LDFLAGS_MODULE += --save-restore-funcs
|
||||
else
|
||||
KBUILD_LDFLAGS_MODULE += arch/powerpc/lib/crtsavres.o
|
||||
endif
|
||||
endif
|
||||
|
||||
ifdef CONFIG_CPU_LITTLE_ENDIAN
|
||||
KBUILD_CFLAGS += -mlittle-endian
|
||||
@ -166,7 +161,7 @@ asinstr := $(call as-instr,lis 9$(comma)foo@high,-DHAVE_AS_ATHIGH=1)
|
||||
KBUILD_CPPFLAGS += -I $(srctree)/arch/$(ARCH) $(asinstr)
|
||||
KBUILD_AFLAGS += $(AFLAGS-y)
|
||||
KBUILD_CFLAGS += $(call cc-option,-msoft-float)
|
||||
KBUILD_CFLAGS += -pipe $(CFLAGS-y)
|
||||
KBUILD_CFLAGS += $(CFLAGS-y)
|
||||
CPP = $(CC) -E $(KBUILD_CFLAGS)
|
||||
|
||||
CHECKFLAGS += -m$(BITS) -D__powerpc__ -D__powerpc$(BITS)__
|
||||
@ -398,14 +393,12 @@ endif
|
||||
endif
|
||||
|
||||
PHONY += checkbin
|
||||
# Check toolchain versions:
|
||||
# - gcc-4.6 is the minimum kernel-wide version so nothing required.
|
||||
checkbin:
|
||||
@if test "x${CONFIG_LD_IS_LLD}" != "xy" -a \
|
||||
"x$(call ld-ifversion, -le, 22400, y)" = "xy" ; then \
|
||||
echo -n '*** binutils 2.24 miscompiles weak symbols ' ; \
|
||||
echo 'in some circumstances.' ; \
|
||||
echo '*** binutils 2.23 do not define the TOC symbol ' ; \
|
||||
echo -n '*** Please use a different binutils version.' ; \
|
||||
@if test "x${CONFIG_FTRACE_MCOUNT_USE_RECORDMCOUNT}" = "xy" -a \
|
||||
"x${CONFIG_LD_IS_BFD}" = "xy" -a \
|
||||
"${CONFIG_LD_VERSION}" = "23700" ; then \
|
||||
echo -n '*** binutils 2.37 drops unused section symbols, which recordmcount ' ; \
|
||||
echo 'is unable to handle.' ; \
|
||||
echo '*** Please use a different binutils version.' ; \
|
||||
false ; \
|
||||
fi
|
||||
|
@ -32,47 +32,59 @@ else
|
||||
BOOTAR := $(AR)
|
||||
endif
|
||||
|
||||
BOOTCFLAGS := -Wall -Wundef -Wstrict-prototypes -Wno-trigraphs \
|
||||
-fno-strict-aliasing -O2 -msoft-float -mno-altivec -mno-vsx \
|
||||
$(call cc-option,-mno-spe) $(call cc-option,-mspe=no) \
|
||||
-pipe -fomit-frame-pointer -fno-builtin -fPIC -nostdinc \
|
||||
$(LINUXINCLUDE)
|
||||
|
||||
ifdef CONFIG_PPC64_BOOT_WRAPPER
|
||||
BOOTCFLAGS += -m64
|
||||
BOOTTARGETFLAGS += -m64
|
||||
BOOTTARGETFLAGS += -mabi=elfv2
|
||||
ifdef CONFIG_PPC64_ELF_ABI_V2
|
||||
BOOTCFLAGS += $(call cc-option,-mabi=elfv2)
|
||||
BOOTTARGETFLAGS += $(call cc-option,-mabi=elfv2)
|
||||
endif
|
||||
else
|
||||
BOOTCFLAGS += -m32
|
||||
BOOTTARGETFLAGS := -m32
|
||||
endif
|
||||
|
||||
ifdef CONFIG_TARGET_CPU_BOOL
|
||||
BOOTCFLAGS += -mcpu=$(CONFIG_TARGET_CPU)
|
||||
BOOTTARGETFLAGS += -mcpu=$(CONFIG_TARGET_CPU)
|
||||
else ifdef CONFIG_PPC64_BOOT_WRAPPER
|
||||
ifdef CONFIG_CPU_LITTLE_ENDIAN
|
||||
BOOTCFLAGS += -mcpu=powerpc64le
|
||||
BOOTTARGETFLAGS += -mcpu=powerpc64le
|
||||
else
|
||||
BOOTCFLAGS += -mcpu=powerpc64
|
||||
BOOTTARGETFLAGS += -mcpu=powerpc64
|
||||
endif
|
||||
endif
|
||||
|
||||
BOOTCFLAGS += -isystem $(shell $(BOOTCC) -print-file-name=include)
|
||||
$(obj)/4xx.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
$(obj)/ebony.o: BOOTTARGETFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-hotfoot.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
$(obj)/cuboot-taishan.o: BOOTTARGETFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-katmai.o: BOOTTARGETFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-acadia.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-iss4xx.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-currituck.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-akebono.o: BOOTTARGETFLAGS += -mcpu=405
|
||||
|
||||
ifdef CONFIG_CPU_BIG_ENDIAN
|
||||
BOOTCFLAGS += -mbig-endian
|
||||
BOOTTARGETFLAGS += -mbig-endian
|
||||
else
|
||||
BOOTCFLAGS += -mlittle-endian
|
||||
BOOTTARGETFLAGS += -mlittle-endian
|
||||
endif
|
||||
|
||||
BOOTAFLAGS := -D__ASSEMBLY__ $(BOOTCFLAGS) -nostdinc
|
||||
BOOTCPPFLAGS := -nostdinc $(LINUXINCLUDE)
|
||||
BOOTCPPFLAGS += -isystem $(shell $(BOOTCC) -print-file-name=include)
|
||||
|
||||
BOOTCFLAGS := $(BOOTTARGETFLAGS) \
|
||||
-Wall -Wundef -Wstrict-prototypes -Wno-trigraphs \
|
||||
-fno-strict-aliasing -O2 \
|
||||
-msoft-float -mno-altivec -mno-vsx \
|
||||
$(call cc-option,-mno-prefixed) \
|
||||
$(call cc-option,-mno-pcrel) \
|
||||
$(call cc-option,-mno-mma) \
|
||||
$(call cc-option,-mno-spe) $(call cc-option,-mspe=no) \
|
||||
-fomit-frame-pointer -fno-builtin -fPIC
|
||||
|
||||
BOOTAFLAGS := $(BOOTTARGETFLAGS) -D__ASSEMBLY__
|
||||
|
||||
BOOTARFLAGS := -crD
|
||||
|
||||
BOOTCFLAGS += $(call cc-option,-mno-prefixed) \
|
||||
$(call cc-option,-mno-pcrel) \
|
||||
$(call cc-option,-mno-mma)
|
||||
|
||||
ifdef CONFIG_CC_IS_CLANG
|
||||
BOOTCFLAGS += $(CLANG_FLAGS)
|
||||
BOOTAFLAGS += $(CLANG_FLAGS)
|
||||
@ -91,16 +103,6 @@ BOOTCFLAGS += -I$(objtree)/$(obj) -I$(srctree)/$(obj)
|
||||
|
||||
DTC_FLAGS ?= -p 1024
|
||||
|
||||
$(obj)/4xx.o: BOOTCFLAGS += -mcpu=405
|
||||
$(obj)/ebony.o: BOOTCFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-hotfoot.o: BOOTCFLAGS += -mcpu=405
|
||||
$(obj)/cuboot-taishan.o: BOOTCFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-katmai.o: BOOTCFLAGS += -mcpu=440
|
||||
$(obj)/cuboot-acadia.o: BOOTCFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-iss4xx.o: BOOTCFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-currituck.o: BOOTCFLAGS += -mcpu=405
|
||||
$(obj)/treeboot-akebono.o: BOOTCFLAGS += -mcpu=405
|
||||
|
||||
# The pre-boot decompressors pull in a lot of kernel headers and other source
|
||||
# files. This creates a bit of a dependency headache since we need to copy
|
||||
# these files into the build dir, fix up any includes and ensure that dependent
|
||||
@ -224,10 +226,10 @@ clean-files := $(zlib-) $(zlibheader-) $(zliblinuxheader-) \
|
||||
empty.c zImage.coff.lds zImage.ps3.lds zImage.lds
|
||||
|
||||
quiet_cmd_bootcc = BOOTCC $@
|
||||
cmd_bootcc = $(BOOTCC) -Wp,-MD,$(depfile) $(BOOTCFLAGS) -c -o $@ $<
|
||||
cmd_bootcc = $(BOOTCC) -Wp,-MD,$(depfile) $(BOOTCPPFLAGS) $(BOOTCFLAGS) -c -o $@ $<
|
||||
|
||||
quiet_cmd_bootas = BOOTAS $@
|
||||
cmd_bootas = $(BOOTCC) -Wp,-MD,$(depfile) $(BOOTAFLAGS) -c -o $@ $<
|
||||
cmd_bootas = $(BOOTCC) -Wp,-MD,$(depfile) $(BOOTCPPFLAGS) $(BOOTAFLAGS) -c -o $@ $<
|
||||
|
||||
quiet_cmd_bootar = BOOTAR $@
|
||||
cmd_bootar = $(BOOTAR) $(BOOTARFLAGS) $@.$$$$ $(real-prereqs); mv $@.$$$$ $@
|
||||
@ -340,11 +342,6 @@ image-$(CONFIG_MPC834x_ITX) += cuImage.mpc8349emitx \
|
||||
image-$(CONFIG_ASP834x) += dtbImage.asp834x-redboot
|
||||
|
||||
# Board ports in arch/powerpc/platform/85xx/Kconfig
|
||||
image-$(CONFIG_MPC8540_ADS) += cuImage.mpc8540ads
|
||||
image-$(CONFIG_MPC8560_ADS) += cuImage.mpc8560ads
|
||||
image-$(CONFIG_MPC85xx_CDS) += cuImage.mpc8541cds \
|
||||
cuImage.mpc8548cds_32b \
|
||||
cuImage.mpc8555cds
|
||||
image-$(CONFIG_MPC85xx_MDS) += cuImage.mpc8568mds
|
||||
image-$(CONFIG_MPC85xx_DS) += cuImage.mpc8544ds \
|
||||
cuImage.mpc8572ds
|
||||
|
@ -1,355 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8540 ADS Device Tree Source
|
||||
*
|
||||
* Copyright 2006, 2008 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/dts-v1/;
|
||||
|
||||
/include/ "e500v1_power_isa.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8540ADS";
|
||||
compatible = "MPC8540ADS", "MPC85xxADS";
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
|
||||
aliases {
|
||||
ethernet0 = &enet0;
|
||||
ethernet1 = &enet1;
|
||||
ethernet2 = &enet2;
|
||||
serial0 = &serial0;
|
||||
serial1 = &serial1;
|
||||
pci0 = &pci0;
|
||||
};
|
||||
|
||||
cpus {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
PowerPC,8540@0 {
|
||||
device_type = "cpu";
|
||||
reg = <0x0>;
|
||||
d-cache-line-size = <32>; // 32 bytes
|
||||
i-cache-line-size = <32>; // 32 bytes
|
||||
d-cache-size = <0x8000>; // L1, 32K
|
||||
i-cache-size = <0x8000>; // L1, 32K
|
||||
timebase-frequency = <0>; // 33 MHz, from uboot
|
||||
bus-frequency = <0>; // 166 MHz
|
||||
clock-frequency = <0>; // 825 MHz, from uboot
|
||||
next-level-cache = <&L2>;
|
||||
};
|
||||
};
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0x0 0x8000000>; // 128M at 0x0
|
||||
};
|
||||
|
||||
soc8540@e0000000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
device_type = "soc";
|
||||
compatible = "simple-bus";
|
||||
ranges = <0x0 0xe0000000 0x100000>;
|
||||
bus-frequency = <0>;
|
||||
|
||||
ecm-law@0 {
|
||||
compatible = "fsl,ecm-law";
|
||||
reg = <0x0 0x1000>;
|
||||
fsl,num-laws = <8>;
|
||||
};
|
||||
|
||||
ecm@1000 {
|
||||
compatible = "fsl,mpc8540-ecm", "fsl,ecm";
|
||||
reg = <0x1000 0x1000>;
|
||||
interrupts = <17 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
memory-controller@2000 {
|
||||
compatible = "fsl,mpc8540-memory-controller";
|
||||
reg = <0x2000 0x1000>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <18 2>;
|
||||
};
|
||||
|
||||
L2: l2-cache-controller@20000 {
|
||||
compatible = "fsl,mpc8540-l2-cache-controller";
|
||||
reg = <0x20000 0x1000>;
|
||||
cache-line-size = <32>; // 32 bytes
|
||||
cache-size = <0x40000>; // L2, 256K
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <16 2>;
|
||||
};
|
||||
|
||||
i2c@3000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
cell-index = <0>;
|
||||
compatible = "fsl-i2c";
|
||||
reg = <0x3000 0x100>;
|
||||
interrupts = <43 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
dfsrr;
|
||||
};
|
||||
|
||||
dma@21300 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8540-dma", "fsl,eloplus-dma";
|
||||
reg = <0x21300 0x4>;
|
||||
ranges = <0x0 0x21100 0x200>;
|
||||
cell-index = <0>;
|
||||
dma-channel@0 {
|
||||
compatible = "fsl,mpc8540-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x0 0x80>;
|
||||
cell-index = <0>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <20 2>;
|
||||
};
|
||||
dma-channel@80 {
|
||||
compatible = "fsl,mpc8540-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x80 0x80>;
|
||||
cell-index = <1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <21 2>;
|
||||
};
|
||||
dma-channel@100 {
|
||||
compatible = "fsl,mpc8540-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x100 0x80>;
|
||||
cell-index = <2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <22 2>;
|
||||
};
|
||||
dma-channel@180 {
|
||||
compatible = "fsl,mpc8540-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x180 0x80>;
|
||||
cell-index = <3>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <23 2>;
|
||||
};
|
||||
};
|
||||
|
||||
enet0: ethernet@24000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <0>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x24000 0x1000>;
|
||||
ranges = <0x0 0x24000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <29 2 30 2 34 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi0>;
|
||||
phy-handle = <&phy0>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-mdio";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
phy0: ethernet-phy@0 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x0>;
|
||||
};
|
||||
phy1: ethernet-phy@1 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x1>;
|
||||
};
|
||||
phy3: ethernet-phy@3 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <7 1>;
|
||||
reg = <0x3>;
|
||||
};
|
||||
tbi0: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
enet1: ethernet@25000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <1>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x25000 0x1000>;
|
||||
ranges = <0x0 0x25000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <35 2 36 2 40 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi1>;
|
||||
phy-handle = <&phy1>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-tbi";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
tbi1: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
enet2: ethernet@26000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <2>;
|
||||
device_type = "network";
|
||||
model = "FEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x26000 0x1000>;
|
||||
ranges = <0x0 0x26000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <41 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi2>;
|
||||
phy-handle = <&phy3>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-tbi";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
tbi2: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
serial0: serial@4500 {
|
||||
cell-index = <0>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4500 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
serial1: serial@4600 {
|
||||
cell-index = <1>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4600 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
mpic: pic@40000 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
reg = <0x40000 0x40000>;
|
||||
compatible = "chrp,open-pic";
|
||||
device_type = "open-pic";
|
||||
};
|
||||
};
|
||||
|
||||
pci0: pci@e0008000 {
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x02 */
|
||||
0x1000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x1000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x1000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x1000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 0x03 */
|
||||
0x1800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x1800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x1800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x1800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x04 */
|
||||
0x2000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0x2000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0x2000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0x2000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 0x05 */
|
||||
0x2800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0x2800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0x2800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0x2800 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 0x0c */
|
||||
0x6000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x6000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x6000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x6000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 0x0d */
|
||||
0x6800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x6800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x6800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x6800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x0e */
|
||||
0x7000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0x7000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0x7000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0x7000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 0x0f */
|
||||
0x7800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0x7800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0x7800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0x7800 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 0x12 */
|
||||
0x9000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x9000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x9000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x9000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 0x13 */
|
||||
0x9800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x9800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x9800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x9800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x14 */
|
||||
0xa000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0xa000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0xa000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0xa000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 0x15 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x1 0x1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <24 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0x80000000 0x80000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe2000000 0x0 0x100000>;
|
||||
clock-frequency = <66666666>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
reg = <0xe0008000 0x1000>;
|
||||
compatible = "fsl,mpc8540-pcix", "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
};
|
||||
};
|
@ -1,375 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8541 CDS Device Tree Source
|
||||
*
|
||||
* Copyright 2006, 2008 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/dts-v1/;
|
||||
|
||||
/include/ "e500v1_power_isa.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8541CDS";
|
||||
compatible = "MPC8541CDS", "MPC85xxCDS";
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
|
||||
aliases {
|
||||
ethernet0 = &enet0;
|
||||
ethernet1 = &enet1;
|
||||
serial0 = &serial0;
|
||||
serial1 = &serial1;
|
||||
pci0 = &pci0;
|
||||
pci1 = &pci1;
|
||||
};
|
||||
|
||||
cpus {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
PowerPC,8541@0 {
|
||||
device_type = "cpu";
|
||||
reg = <0x0>;
|
||||
d-cache-line-size = <32>; // 32 bytes
|
||||
i-cache-line-size = <32>; // 32 bytes
|
||||
d-cache-size = <0x8000>; // L1, 32K
|
||||
i-cache-size = <0x8000>; // L1, 32K
|
||||
timebase-frequency = <0>; // 33 MHz, from uboot
|
||||
bus-frequency = <0>; // 166 MHz
|
||||
clock-frequency = <0>; // 825 MHz, from uboot
|
||||
next-level-cache = <&L2>;
|
||||
};
|
||||
};
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0x0 0x8000000>; // 128M at 0x0
|
||||
};
|
||||
|
||||
soc8541@e0000000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
device_type = "soc";
|
||||
compatible = "simple-bus";
|
||||
ranges = <0x0 0xe0000000 0x100000>;
|
||||
bus-frequency = <0>;
|
||||
|
||||
ecm-law@0 {
|
||||
compatible = "fsl,ecm-law";
|
||||
reg = <0x0 0x1000>;
|
||||
fsl,num-laws = <8>;
|
||||
};
|
||||
|
||||
ecm@1000 {
|
||||
compatible = "fsl,mpc8541-ecm", "fsl,ecm";
|
||||
reg = <0x1000 0x1000>;
|
||||
interrupts = <17 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
memory-controller@2000 {
|
||||
compatible = "fsl,mpc8541-memory-controller";
|
||||
reg = <0x2000 0x1000>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <18 2>;
|
||||
};
|
||||
|
||||
L2: l2-cache-controller@20000 {
|
||||
compatible = "fsl,mpc8541-l2-cache-controller";
|
||||
reg = <0x20000 0x1000>;
|
||||
cache-line-size = <32>; // 32 bytes
|
||||
cache-size = <0x40000>; // L2, 256K
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <16 2>;
|
||||
};
|
||||
|
||||
i2c@3000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
cell-index = <0>;
|
||||
compatible = "fsl-i2c";
|
||||
reg = <0x3000 0x100>;
|
||||
interrupts = <43 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
dfsrr;
|
||||
};
|
||||
|
||||
dma@21300 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8541-dma", "fsl,eloplus-dma";
|
||||
reg = <0x21300 0x4>;
|
||||
ranges = <0x0 0x21100 0x200>;
|
||||
cell-index = <0>;
|
||||
dma-channel@0 {
|
||||
compatible = "fsl,mpc8541-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x0 0x80>;
|
||||
cell-index = <0>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <20 2>;
|
||||
};
|
||||
dma-channel@80 {
|
||||
compatible = "fsl,mpc8541-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x80 0x80>;
|
||||
cell-index = <1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <21 2>;
|
||||
};
|
||||
dma-channel@100 {
|
||||
compatible = "fsl,mpc8541-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x100 0x80>;
|
||||
cell-index = <2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <22 2>;
|
||||
};
|
||||
dma-channel@180 {
|
||||
compatible = "fsl,mpc8541-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x180 0x80>;
|
||||
cell-index = <3>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <23 2>;
|
||||
};
|
||||
};
|
||||
|
||||
enet0: ethernet@24000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <0>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x24000 0x1000>;
|
||||
ranges = <0x0 0x24000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <29 2 30 2 34 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi0>;
|
||||
phy-handle = <&phy0>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-mdio";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
phy0: ethernet-phy@0 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x0>;
|
||||
};
|
||||
phy1: ethernet-phy@1 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x1>;
|
||||
};
|
||||
tbi0: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
enet1: ethernet@25000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <1>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x25000 0x1000>;
|
||||
ranges = <0x0 0x25000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <35 2 36 2 40 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi1>;
|
||||
phy-handle = <&phy1>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-tbi";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
tbi1: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
serial0: serial@4500 {
|
||||
cell-index = <0>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4500 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
serial1: serial@4600 {
|
||||
cell-index = <1>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4600 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
crypto@30000 {
|
||||
compatible = "fsl,sec2.0";
|
||||
reg = <0x30000 0x10000>;
|
||||
interrupts = <45 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
fsl,num-channels = <4>;
|
||||
fsl,channel-fifo-len = <24>;
|
||||
fsl,exec-units-mask = <0x7e>;
|
||||
fsl,descriptor-types-mask = <0x01010ebf>;
|
||||
};
|
||||
|
||||
mpic: pic@40000 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
reg = <0x40000 0x40000>;
|
||||
compatible = "chrp,open-pic";
|
||||
device_type = "open-pic";
|
||||
};
|
||||
|
||||
cpm@919c0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8541-cpm", "fsl,cpm2";
|
||||
reg = <0x919c0 0x30>;
|
||||
ranges;
|
||||
|
||||
muram@80000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
ranges = <0x0 0x80000 0x10000>;
|
||||
|
||||
data@0 {
|
||||
compatible = "fsl,cpm-muram-data";
|
||||
reg = <0x0 0x2000 0x9000 0x1000>;
|
||||
};
|
||||
};
|
||||
|
||||
brg@919f0 {
|
||||
compatible = "fsl,mpc8541-brg",
|
||||
"fsl,cpm2-brg",
|
||||
"fsl,cpm-brg";
|
||||
reg = <0x919f0 0x10 0x915f0 0x10>;
|
||||
};
|
||||
|
||||
cpmpic: pic@90c00 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
interrupts = <46 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
reg = <0x90c00 0x80>;
|
||||
compatible = "fsl,mpc8541-cpm-pic", "fsl,cpm2-pic";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
pci0: pci@e0008000 {
|
||||
interrupt-map-mask = <0x1f800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x10 */
|
||||
0x8000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x8000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x8000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x8000 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x11 */
|
||||
0x8800 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x8800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x8800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x8800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x12 (Slot 1) */
|
||||
0x9000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x9000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x9000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x9000 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x13 (Slot 2) */
|
||||
0x9800 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x9800 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x9800 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x9800 0x0 0x0 0x4 &mpic 0x0 0x1
|
||||
|
||||
/* IDSEL 0x14 (Slot 3) */
|
||||
0xa000 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0xa000 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0xa000 0x0 0x0 0x3 &mpic 0x0 0x1
|
||||
0xa000 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 0x15 (Slot 4) */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x0 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* Bus 1 (Tundra Bridge) */
|
||||
/* IDSEL 0x12 (ISA bridge) */
|
||||
0x19000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x19000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x19000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x19000 0x0 0x0 0x4 &mpic 0x3 0x1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <24 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0x80000000 0x80000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe2000000 0x0 0x100000>;
|
||||
clock-frequency = <66666666>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
reg = <0xe0008000 0x1000>;
|
||||
compatible = "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
|
||||
i8259@19000 {
|
||||
interrupt-controller;
|
||||
device_type = "interrupt-controller";
|
||||
reg = <0x19000 0x0 0x0 0x0 0x1>;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
compatible = "chrp,iic";
|
||||
interrupts = <1>;
|
||||
interrupt-parent = <&pci0>;
|
||||
};
|
||||
};
|
||||
|
||||
pci1: pci@e0009000 {
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x15 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0xb 0x1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <25 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0xa0000000 0xa0000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe3000000 0x0 0x100000>;
|
||||
clock-frequency = <66666666>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
reg = <0xe0009000 0x1000>;
|
||||
compatible = "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
};
|
||||
};
|
@ -1,302 +0,0 @@
|
||||
/*
|
||||
* MPC8548CDS Device Tree Source stub (no addresses or top-level ranges)
|
||||
*
|
||||
* Copyright 2012 Freescale Semiconductor Inc.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions are met:
|
||||
* * Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* * Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
* * Neither the name of Freescale Semiconductor nor the
|
||||
* names of its contributors may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
*
|
||||
* ALTERNATIVELY, this software may be distributed under the terms of the
|
||||
* GNU General Public License ("GPL") as published by the Free Software
|
||||
* Foundation, either version 2 of that License or (at your option) any
|
||||
* later version.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY Freescale Semiconductor ``AS IS'' AND ANY
|
||||
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
* DISCLAIMED. IN NO EVENT SHALL Freescale Semiconductor BE LIABLE FOR ANY
|
||||
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
&board_lbc {
|
||||
nor@0,0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "cfi-flash";
|
||||
reg = <0x0 0x0 0x01000000>;
|
||||
bank-width = <2>;
|
||||
device-width = <2>;
|
||||
|
||||
partition@0 {
|
||||
reg = <0x0 0x0b00000>;
|
||||
label = "ramdisk-nor";
|
||||
};
|
||||
|
||||
partition@300000 {
|
||||
reg = <0x0b00000 0x0400000>;
|
||||
label = "kernel-nor";
|
||||
};
|
||||
|
||||
partition@700000 {
|
||||
reg = <0x0f00000 0x060000>;
|
||||
label = "dtb-nor";
|
||||
};
|
||||
|
||||
partition@760000 {
|
||||
reg = <0x0f60000 0x020000>;
|
||||
label = "env-nor";
|
||||
read-only;
|
||||
};
|
||||
|
||||
partition@780000 {
|
||||
reg = <0x0f80000 0x080000>;
|
||||
label = "u-boot-nor";
|
||||
read-only;
|
||||
};
|
||||
};
|
||||
|
||||
board-control@1,0 {
|
||||
compatible = "fsl,mpc8548cds-fpga";
|
||||
reg = <0x1 0x0 0x1000>;
|
||||
};
|
||||
};
|
||||
|
||||
&board_soc {
|
||||
i2c@3000 {
|
||||
eeprom@50 {
|
||||
compatible = "atmel,24c64";
|
||||
reg = <0x50>;
|
||||
};
|
||||
|
||||
eeprom@56 {
|
||||
compatible = "atmel,24c64";
|
||||
reg = <0x56>;
|
||||
};
|
||||
|
||||
eeprom@57 {
|
||||
compatible = "atmel,24c64";
|
||||
reg = <0x57>;
|
||||
};
|
||||
};
|
||||
|
||||
i2c@3100 {
|
||||
eeprom@50 {
|
||||
compatible = "atmel,24c64";
|
||||
reg = <0x50>;
|
||||
};
|
||||
};
|
||||
|
||||
enet0: ethernet@24000 {
|
||||
tbi-handle = <&tbi0>;
|
||||
phy-handle = <&phy0>;
|
||||
};
|
||||
|
||||
mdio@24520 {
|
||||
phy0: ethernet-phy@0 {
|
||||
interrupts = <5 1 0 0>;
|
||||
reg = <0x0>;
|
||||
};
|
||||
phy1: ethernet-phy@1 {
|
||||
interrupts = <5 1 0 0>;
|
||||
reg = <0x1>;
|
||||
};
|
||||
phy2: ethernet-phy@2 {
|
||||
interrupts = <5 1 0 0>;
|
||||
reg = <0x2>;
|
||||
};
|
||||
phy3: ethernet-phy@3 {
|
||||
interrupts = <5 1 0 0>;
|
||||
reg = <0x3>;
|
||||
};
|
||||
tbi0: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
|
||||
enet1: ethernet@25000 {
|
||||
tbi-handle = <&tbi1>;
|
||||
phy-handle = <&phy1>;
|
||||
};
|
||||
|
||||
mdio@25520 {
|
||||
tbi1: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
|
||||
enet2: ethernet@26000 {
|
||||
tbi-handle = <&tbi2>;
|
||||
phy-handle = <&phy2>;
|
||||
};
|
||||
|
||||
mdio@26520 {
|
||||
tbi2: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
|
||||
enet3: ethernet@27000 {
|
||||
tbi-handle = <&tbi3>;
|
||||
phy-handle = <&phy3>;
|
||||
};
|
||||
|
||||
mdio@27520 {
|
||||
tbi3: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
&board_pci0 {
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
/* IDSEL 0x4 (PCIX Slot 2) */
|
||||
0x2000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0x2000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0x2000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0x2000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x5 (PCIX Slot 3) */
|
||||
0x2800 0x0 0x0 0x1 &mpic 0x1 0x1 0 0
|
||||
0x2800 0x0 0x0 0x2 &mpic 0x2 0x1 0 0
|
||||
0x2800 0x0 0x0 0x3 &mpic 0x3 0x1 0 0
|
||||
0x2800 0x0 0x0 0x4 &mpic 0x0 0x1 0 0
|
||||
|
||||
/* IDSEL 0x6 (PCIX Slot 4) */
|
||||
0x3000 0x0 0x0 0x1 &mpic 0x2 0x1 0 0
|
||||
0x3000 0x0 0x0 0x2 &mpic 0x3 0x1 0 0
|
||||
0x3000 0x0 0x0 0x3 &mpic 0x0 0x1 0 0
|
||||
0x3000 0x0 0x0 0x4 &mpic 0x1 0x1 0 0
|
||||
|
||||
/* IDSEL 0x8 (PCIX Slot 5) */
|
||||
0x4000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0x4000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0x4000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0x4000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0xC (Tsi310 bridge) */
|
||||
0x6000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0x6000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0x6000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0x6000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x14 (Slot 2) */
|
||||
0xa000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0xa000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0xa000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0xa000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x15 (Slot 3) */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0x1 0x1 0 0
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x2 0x1 0 0
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x3 0x1 0 0
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x0 0x1 0 0
|
||||
|
||||
/* IDSEL 0x16 (Slot 4) */
|
||||
0xb000 0x0 0x0 0x1 &mpic 0x2 0x1 0 0
|
||||
0xb000 0x0 0x0 0x2 &mpic 0x3 0x1 0 0
|
||||
0xb000 0x0 0x0 0x3 &mpic 0x0 0x1 0 0
|
||||
0xb000 0x0 0x0 0x4 &mpic 0x1 0x1 0 0
|
||||
|
||||
/* IDSEL 0x18 (Slot 5) */
|
||||
0xc000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0xc000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0xc000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0xc000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x1C (Tsi310 bridge PCI primary) */
|
||||
0xe000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0xe000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0xe000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0xe000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0>;
|
||||
|
||||
pci_bridge@1c {
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x00 (PrPMC Site) */
|
||||
0000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x04 (VIA chip) */
|
||||
0x2000 0x0 0x0 0x1 &mpic 0x0 0x1 0 0
|
||||
0x2000 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0x2000 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0x2000 0x0 0x0 0x4 &mpic 0x3 0x1 0 0
|
||||
|
||||
/* IDSEL 0x05 (8139) */
|
||||
0x2800 0x0 0x0 0x1 &mpic 0x1 0x1 0 0
|
||||
|
||||
/* IDSEL 0x06 (Slot 6) */
|
||||
0x3000 0x0 0x0 0x1 &mpic 0x2 0x1 0 0
|
||||
0x3000 0x0 0x0 0x2 &mpic 0x3 0x1 0 0
|
||||
0x3000 0x0 0x0 0x3 &mpic 0x0 0x1 0 0
|
||||
0x3000 0x0 0x0 0x4 &mpic 0x1 0x1 0 0
|
||||
|
||||
/* IDESL 0x07 (Slot 7) */
|
||||
0x3800 0x0 0x0 0x1 &mpic 0x3 0x1 0 0
|
||||
0x3800 0x0 0x0 0x2 &mpic 0x0 0x1 0 0
|
||||
0x3800 0x0 0x0 0x3 &mpic 0x1 0x1 0 0
|
||||
0x3800 0x0 0x0 0x4 &mpic 0x2 0x1 0 0>;
|
||||
|
||||
reg = <0xe000 0x0 0x0 0x0 0x0>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
ranges = <0x2000000 0x0 0x80000000
|
||||
0x2000000 0x0 0x80000000
|
||||
0x0 0x20000000
|
||||
0x1000000 0x0 0x0
|
||||
0x1000000 0x0 0x0
|
||||
0x0 0x80000>;
|
||||
clock-frequency = <33333333>;
|
||||
|
||||
isa@4 {
|
||||
device_type = "isa";
|
||||
#interrupt-cells = <2>;
|
||||
#size-cells = <1>;
|
||||
#address-cells = <2>;
|
||||
reg = <0x2000 0x0 0x0 0x0 0x0>;
|
||||
ranges = <0x1 0x0 0x1000000 0x0 0x0 0x1000>;
|
||||
interrupt-parent = <&i8259>;
|
||||
|
||||
i8259: interrupt-controller@20 {
|
||||
interrupt-controller;
|
||||
device_type = "interrupt-controller";
|
||||
reg = <0x1 0x20 0x2
|
||||
0x1 0xa0 0x2
|
||||
0x1 0x4d0 0x2>;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
compatible = "chrp,iic";
|
||||
interrupts = <0 1 0 0>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
rtc@70 {
|
||||
compatible = "pnpPNP,b00";
|
||||
reg = <0x1 0x70 0x2>;
|
||||
};
|
||||
};
|
||||
};
|
||||
};
|
@ -1,82 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8548 CDS Device Tree Source (32-bit address map)
|
||||
*
|
||||
* Copyright 2006, 2008, 2011-2012 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/include/ "mpc8548si-pre.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8548CDS";
|
||||
compatible = "MPC8548CDS", "MPC85xxCDS";
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0 0 0x0 0x8000000>; // 128M at 0x0
|
||||
};
|
||||
|
||||
board_lbc: lbc: localbus@e0005000 {
|
||||
reg = <0 0xe0005000 0 0x1000>;
|
||||
|
||||
ranges = <0x0 0x0 0x0 0xff000000 0x01000000
|
||||
0x1 0x0 0x0 0xf8004000 0x00001000>;
|
||||
|
||||
};
|
||||
|
||||
board_soc: soc: soc8548@e0000000 {
|
||||
ranges = <0 0x0 0xe0000000 0x100000>;
|
||||
};
|
||||
|
||||
board_pci0: pci0: pci@e0008000 {
|
||||
reg = <0 0xe0008000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0x80000000 0 0x80000000 0x0 0x10000000
|
||||
0x1000000 0x0 0x00000000 0 0xe2000000 0x0 0x800000>;
|
||||
clock-frequency = <66666666>;
|
||||
};
|
||||
|
||||
pci1: pci@e0009000 {
|
||||
reg = <0 0xe0009000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0x90000000 0 0x90000000 0x0 0x10000000
|
||||
0x1000000 0x0 0x00000000 0 0xe2800000 0x0 0x800000>;
|
||||
clock-frequency = <66666666>;
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x15 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0xb 0x1 0 0
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x3 0x1 0 0>;
|
||||
};
|
||||
|
||||
pci2: pcie@e000a000 {
|
||||
reg = <0 0xe000a000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0xa0000000 0 0xa0000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x00000000 0 0xe3000000 0x0 0x100000>;
|
||||
pcie@0 {
|
||||
ranges = <0x2000000 0x0 0xa0000000
|
||||
0x2000000 0x0 0xa0000000
|
||||
0x0 0x20000000
|
||||
|
||||
0x1000000 0x0 0x0
|
||||
0x1000000 0x0 0x0
|
||||
0x0 0x100000>;
|
||||
};
|
||||
};
|
||||
|
||||
rio: rapidio@e00c0000 {
|
||||
reg = <0x0 0xe00c0000 0x0 0x20000>;
|
||||
port1 {
|
||||
ranges = <0x0 0x0 0x0 0xc0000000 0x0 0x20000000>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
/*
|
||||
* mpc8548cds.dtsi must be last to ensure board_pci0 overrides pci0 settings
|
||||
* for interrupt-map & interrupt-map-mask.
|
||||
*/
|
||||
|
||||
/include/ "mpc8548si-post.dtsi"
|
||||
/include/ "mpc8548cds.dtsi"
|
@ -1,82 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8548 CDS Device Tree Source (36-bit address map)
|
||||
*
|
||||
* Copyright 2012 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/include/ "mpc8548si-pre.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8548CDS";
|
||||
compatible = "MPC8548CDS", "MPC85xxCDS";
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0 0 0x0 0x8000000>; // 128M at 0x0
|
||||
};
|
||||
|
||||
board_lbc: lbc: localbus@fe0005000 {
|
||||
reg = <0xf 0xe0005000 0 0x1000>;
|
||||
|
||||
ranges = <0x0 0x0 0xf 0xff000000 0x01000000
|
||||
0x1 0x0 0xf 0xf8004000 0x00001000>;
|
||||
|
||||
};
|
||||
|
||||
board_soc: soc: soc8548@fe0000000 {
|
||||
ranges = <0 0xf 0xe0000000 0x100000>;
|
||||
};
|
||||
|
||||
board_pci0: pci0: pci@fe0008000 {
|
||||
reg = <0xf 0xe0008000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0xe0000000 0xc 0x00000000 0x0 0x10000000
|
||||
0x1000000 0x0 0x00000000 0xf 0xe2000000 0x0 0x800000>;
|
||||
clock-frequency = <66666666>;
|
||||
};
|
||||
|
||||
pci1: pci@fe0009000 {
|
||||
reg = <0xf 0xe0009000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0xe0000000 0xc 0x10000000 0x0 0x10000000
|
||||
0x1000000 0x0 0x00000000 0xf 0xe2800000 0x0 0x800000>;
|
||||
clock-frequency = <66666666>;
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x15 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0xb 0x1 0 0
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x1 0x1 0 0
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x2 0x1 0 0
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x3 0x1 0 0>;
|
||||
};
|
||||
|
||||
pci2: pcie@fe000a000 {
|
||||
reg = <0xf 0xe000a000 0 0x1000>;
|
||||
ranges = <0x2000000 0x0 0xe0000000 0xc 0x20000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x00000000 0xf 0xe3000000 0x0 0x100000>;
|
||||
pcie@0 {
|
||||
ranges = <0x2000000 0x0 0xa0000000
|
||||
0x2000000 0x0 0xa0000000
|
||||
0x0 0x20000000
|
||||
|
||||
0x1000000 0x0 0x0
|
||||
0x1000000 0x0 0x0
|
||||
0x0 0x100000>;
|
||||
};
|
||||
};
|
||||
|
||||
rio: rapidio@fe00c0000 {
|
||||
reg = <0xf 0xe00c0000 0x0 0x20000>;
|
||||
port1 {
|
||||
ranges = <0x0 0x0 0xc 0x40000000 0x0 0x20000000>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
/*
|
||||
* mpc8548cds.dtsi must be last to ensure board_pci0 overrides pci0 settings
|
||||
* for interrupt-map & interrupt-map-mask.
|
||||
*/
|
||||
|
||||
/include/ "mpc8548si-post.dtsi"
|
||||
/include/ "mpc8548cds.dtsi"
|
@ -1,375 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8555 CDS Device Tree Source
|
||||
*
|
||||
* Copyright 2006, 2008 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/dts-v1/;
|
||||
|
||||
/include/ "e500v1_power_isa.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8555CDS";
|
||||
compatible = "MPC8555CDS", "MPC85xxCDS";
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
|
||||
aliases {
|
||||
ethernet0 = &enet0;
|
||||
ethernet1 = &enet1;
|
||||
serial0 = &serial0;
|
||||
serial1 = &serial1;
|
||||
pci0 = &pci0;
|
||||
pci1 = &pci1;
|
||||
};
|
||||
|
||||
cpus {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
PowerPC,8555@0 {
|
||||
device_type = "cpu";
|
||||
reg = <0x0>;
|
||||
d-cache-line-size = <32>; // 32 bytes
|
||||
i-cache-line-size = <32>; // 32 bytes
|
||||
d-cache-size = <0x8000>; // L1, 32K
|
||||
i-cache-size = <0x8000>; // L1, 32K
|
||||
timebase-frequency = <0>; // 33 MHz, from uboot
|
||||
bus-frequency = <0>; // 166 MHz
|
||||
clock-frequency = <0>; // 825 MHz, from uboot
|
||||
next-level-cache = <&L2>;
|
||||
};
|
||||
};
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0x0 0x8000000>; // 128M at 0x0
|
||||
};
|
||||
|
||||
soc8555@e0000000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
device_type = "soc";
|
||||
compatible = "simple-bus";
|
||||
ranges = <0x0 0xe0000000 0x100000>;
|
||||
bus-frequency = <0>;
|
||||
|
||||
ecm-law@0 {
|
||||
compatible = "fsl,ecm-law";
|
||||
reg = <0x0 0x1000>;
|
||||
fsl,num-laws = <8>;
|
||||
};
|
||||
|
||||
ecm@1000 {
|
||||
compatible = "fsl,mpc8555-ecm", "fsl,ecm";
|
||||
reg = <0x1000 0x1000>;
|
||||
interrupts = <17 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
memory-controller@2000 {
|
||||
compatible = "fsl,mpc8555-memory-controller";
|
||||
reg = <0x2000 0x1000>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <18 2>;
|
||||
};
|
||||
|
||||
L2: l2-cache-controller@20000 {
|
||||
compatible = "fsl,mpc8555-l2-cache-controller";
|
||||
reg = <0x20000 0x1000>;
|
||||
cache-line-size = <32>; // 32 bytes
|
||||
cache-size = <0x40000>; // L2, 256K
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <16 2>;
|
||||
};
|
||||
|
||||
i2c@3000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
cell-index = <0>;
|
||||
compatible = "fsl-i2c";
|
||||
reg = <0x3000 0x100>;
|
||||
interrupts = <43 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
dfsrr;
|
||||
};
|
||||
|
||||
dma@21300 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8555-dma", "fsl,eloplus-dma";
|
||||
reg = <0x21300 0x4>;
|
||||
ranges = <0x0 0x21100 0x200>;
|
||||
cell-index = <0>;
|
||||
dma-channel@0 {
|
||||
compatible = "fsl,mpc8555-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x0 0x80>;
|
||||
cell-index = <0>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <20 2>;
|
||||
};
|
||||
dma-channel@80 {
|
||||
compatible = "fsl,mpc8555-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x80 0x80>;
|
||||
cell-index = <1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <21 2>;
|
||||
};
|
||||
dma-channel@100 {
|
||||
compatible = "fsl,mpc8555-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x100 0x80>;
|
||||
cell-index = <2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <22 2>;
|
||||
};
|
||||
dma-channel@180 {
|
||||
compatible = "fsl,mpc8555-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x180 0x80>;
|
||||
cell-index = <3>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <23 2>;
|
||||
};
|
||||
};
|
||||
|
||||
enet0: ethernet@24000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <0>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x24000 0x1000>;
|
||||
ranges = <0x0 0x24000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <29 2 30 2 34 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi0>;
|
||||
phy-handle = <&phy0>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-mdio";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
phy0: ethernet-phy@0 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x0>;
|
||||
};
|
||||
phy1: ethernet-phy@1 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x1>;
|
||||
};
|
||||
tbi0: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
enet1: ethernet@25000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <1>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x25000 0x1000>;
|
||||
ranges = <0x0 0x25000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <35 2 36 2 40 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi1>;
|
||||
phy-handle = <&phy1>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-tbi";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
tbi1: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
serial0: serial@4500 {
|
||||
cell-index = <0>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4500 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
serial1: serial@4600 {
|
||||
cell-index = <1>;
|
||||
device_type = "serial";
|
||||
compatible = "fsl,ns16550", "ns16550";
|
||||
reg = <0x4600 0x100>; // reg base, size
|
||||
clock-frequency = <0>; // should we fill in in uboot?
|
||||
interrupts = <42 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
crypto@30000 {
|
||||
compatible = "fsl,sec2.0";
|
||||
reg = <0x30000 0x10000>;
|
||||
interrupts = <45 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
fsl,num-channels = <4>;
|
||||
fsl,channel-fifo-len = <24>;
|
||||
fsl,exec-units-mask = <0x7e>;
|
||||
fsl,descriptor-types-mask = <0x01010ebf>;
|
||||
};
|
||||
|
||||
mpic: pic@40000 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
reg = <0x40000 0x40000>;
|
||||
compatible = "chrp,open-pic";
|
||||
device_type = "open-pic";
|
||||
};
|
||||
|
||||
cpm@919c0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8555-cpm", "fsl,cpm2";
|
||||
reg = <0x919c0 0x30>;
|
||||
ranges;
|
||||
|
||||
muram@80000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
ranges = <0x0 0x80000 0x10000>;
|
||||
|
||||
data@0 {
|
||||
compatible = "fsl,cpm-muram-data";
|
||||
reg = <0x0 0x2000 0x9000 0x1000>;
|
||||
};
|
||||
};
|
||||
|
||||
brg@919f0 {
|
||||
compatible = "fsl,mpc8555-brg",
|
||||
"fsl,cpm2-brg",
|
||||
"fsl,cpm-brg";
|
||||
reg = <0x919f0 0x10 0x915f0 0x10>;
|
||||
};
|
||||
|
||||
cpmpic: pic@90c00 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
interrupts = <46 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
reg = <0x90c00 0x80>;
|
||||
compatible = "fsl,mpc8555-cpm-pic", "fsl,cpm2-pic";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
pci0: pci@e0008000 {
|
||||
interrupt-map-mask = <0x1f800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x10 */
|
||||
0x8000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x8000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x8000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x8000 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x11 */
|
||||
0x8800 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x8800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x8800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x8800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x12 (Slot 1) */
|
||||
0x9000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x9000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x9000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x9000 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x13 (Slot 2) */
|
||||
0x9800 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x9800 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x9800 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x9800 0x0 0x0 0x4 &mpic 0x0 0x1
|
||||
|
||||
/* IDSEL 0x14 (Slot 3) */
|
||||
0xa000 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0xa000 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0xa000 0x0 0x0 0x3 &mpic 0x0 0x1
|
||||
0xa000 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 0x15 (Slot 4) */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x0 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* Bus 1 (Tundra Bridge) */
|
||||
/* IDSEL 0x12 (ISA bridge) */
|
||||
0x19000 0x0 0x0 0x1 &mpic 0x0 0x1
|
||||
0x19000 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x19000 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x19000 0x0 0x0 0x4 &mpic 0x3 0x1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <24 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0x80000000 0x80000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe2000000 0x0 0x100000>;
|
||||
clock-frequency = <66666666>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
reg = <0xe0008000 0x1000>;
|
||||
compatible = "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
|
||||
i8259@19000 {
|
||||
interrupt-controller;
|
||||
device_type = "interrupt-controller";
|
||||
reg = <0x19000 0x0 0x0 0x0 0x1>;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
compatible = "chrp,iic";
|
||||
interrupts = <1>;
|
||||
interrupt-parent = <&pci0>;
|
||||
};
|
||||
};
|
||||
|
||||
pci1: pci@e0009000 {
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x15 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0xb 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0xb 0x1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <25 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0xa0000000 0xa0000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe3000000 0x0 0x100000>;
|
||||
clock-frequency = <66666666>;
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
reg = <0xe0009000 0x1000>;
|
||||
compatible = "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
};
|
||||
};
|
@ -1,388 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC8560 ADS Device Tree Source
|
||||
*
|
||||
* Copyright 2006, 2008 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
/dts-v1/;
|
||||
|
||||
/include/ "e500v1_power_isa.dtsi"
|
||||
|
||||
/ {
|
||||
model = "MPC8560ADS";
|
||||
compatible = "MPC8560ADS", "MPC85xxADS";
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
|
||||
aliases {
|
||||
ethernet0 = &enet0;
|
||||
ethernet1 = &enet1;
|
||||
ethernet2 = &enet2;
|
||||
ethernet3 = &enet3;
|
||||
serial0 = &serial0;
|
||||
serial1 = &serial1;
|
||||
pci0 = &pci0;
|
||||
};
|
||||
|
||||
cpus {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
|
||||
PowerPC,8560@0 {
|
||||
device_type = "cpu";
|
||||
reg = <0x0>;
|
||||
d-cache-line-size = <32>; // 32 bytes
|
||||
i-cache-line-size = <32>; // 32 bytes
|
||||
d-cache-size = <0x8000>; // L1, 32K
|
||||
i-cache-size = <0x8000>; // L1, 32K
|
||||
timebase-frequency = <82500000>;
|
||||
bus-frequency = <330000000>;
|
||||
clock-frequency = <825000000>;
|
||||
};
|
||||
};
|
||||
|
||||
memory {
|
||||
device_type = "memory";
|
||||
reg = <0x0 0x10000000>;
|
||||
};
|
||||
|
||||
soc8560@e0000000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
device_type = "soc";
|
||||
compatible = "simple-bus";
|
||||
ranges = <0x0 0xe0000000 0x100000>;
|
||||
bus-frequency = <330000000>;
|
||||
|
||||
ecm-law@0 {
|
||||
compatible = "fsl,ecm-law";
|
||||
reg = <0x0 0x1000>;
|
||||
fsl,num-laws = <8>;
|
||||
};
|
||||
|
||||
ecm@1000 {
|
||||
compatible = "fsl,mpc8560-ecm", "fsl,ecm";
|
||||
reg = <0x1000 0x1000>;
|
||||
interrupts = <17 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
};
|
||||
|
||||
memory-controller@2000 {
|
||||
compatible = "fsl,mpc8540-memory-controller";
|
||||
reg = <0x2000 0x1000>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <18 2>;
|
||||
};
|
||||
|
||||
L2: l2-cache-controller@20000 {
|
||||
compatible = "fsl,mpc8540-l2-cache-controller";
|
||||
reg = <0x20000 0x1000>;
|
||||
cache-line-size = <32>; // 32 bytes
|
||||
cache-size = <0x40000>; // L2, 256K
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <16 2>;
|
||||
};
|
||||
|
||||
dma@21300 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8560-dma", "fsl,eloplus-dma";
|
||||
reg = <0x21300 0x4>;
|
||||
ranges = <0x0 0x21100 0x200>;
|
||||
cell-index = <0>;
|
||||
dma-channel@0 {
|
||||
compatible = "fsl,mpc8560-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x0 0x80>;
|
||||
cell-index = <0>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <20 2>;
|
||||
};
|
||||
dma-channel@80 {
|
||||
compatible = "fsl,mpc8560-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x80 0x80>;
|
||||
cell-index = <1>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <21 2>;
|
||||
};
|
||||
dma-channel@100 {
|
||||
compatible = "fsl,mpc8560-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x100 0x80>;
|
||||
cell-index = <2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <22 2>;
|
||||
};
|
||||
dma-channel@180 {
|
||||
compatible = "fsl,mpc8560-dma-channel",
|
||||
"fsl,eloplus-dma-channel";
|
||||
reg = <0x180 0x80>;
|
||||
cell-index = <3>;
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <23 2>;
|
||||
};
|
||||
};
|
||||
|
||||
enet0: ethernet@24000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <0>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x24000 0x1000>;
|
||||
ranges = <0x0 0x24000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <29 2 30 2 34 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi0>;
|
||||
phy-handle = <&phy0>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-mdio";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
phy0: ethernet-phy@0 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x0>;
|
||||
};
|
||||
phy1: ethernet-phy@1 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <5 1>;
|
||||
reg = <0x1>;
|
||||
};
|
||||
phy2: ethernet-phy@2 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <7 1>;
|
||||
reg = <0x2>;
|
||||
};
|
||||
phy3: ethernet-phy@3 {
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <7 1>;
|
||||
reg = <0x3>;
|
||||
};
|
||||
tbi0: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
enet1: ethernet@25000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
cell-index = <1>;
|
||||
device_type = "network";
|
||||
model = "TSEC";
|
||||
compatible = "gianfar";
|
||||
reg = <0x25000 0x1000>;
|
||||
ranges = <0x0 0x25000 0x1000>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
interrupts = <35 2 36 2 40 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
tbi-handle = <&tbi1>;
|
||||
phy-handle = <&phy1>;
|
||||
|
||||
mdio@520 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <0>;
|
||||
compatible = "fsl,gianfar-tbi";
|
||||
reg = <0x520 0x20>;
|
||||
|
||||
tbi1: tbi-phy@11 {
|
||||
reg = <0x11>;
|
||||
device_type = "tbi-phy";
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
mpic: pic@40000 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
reg = <0x40000 0x40000>;
|
||||
compatible = "chrp,open-pic";
|
||||
device_type = "open-pic";
|
||||
};
|
||||
|
||||
cpm@919c0 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
compatible = "fsl,mpc8560-cpm", "fsl,cpm2";
|
||||
reg = <0x919c0 0x30>;
|
||||
ranges;
|
||||
|
||||
muram@80000 {
|
||||
#address-cells = <1>;
|
||||
#size-cells = <1>;
|
||||
ranges = <0x0 0x80000 0x10000>;
|
||||
|
||||
data@0 {
|
||||
compatible = "fsl,cpm-muram-data";
|
||||
reg = <0x0 0x4000 0x9000 0x2000>;
|
||||
};
|
||||
};
|
||||
|
||||
brg@919f0 {
|
||||
compatible = "fsl,mpc8560-brg",
|
||||
"fsl,cpm2-brg",
|
||||
"fsl,cpm-brg";
|
||||
reg = <0x919f0 0x10 0x915f0 0x10>;
|
||||
clock-frequency = <165000000>;
|
||||
};
|
||||
|
||||
cpmpic: pic@90c00 {
|
||||
interrupt-controller;
|
||||
#address-cells = <0>;
|
||||
#interrupt-cells = <2>;
|
||||
interrupts = <46 2>;
|
||||
interrupt-parent = <&mpic>;
|
||||
reg = <0x90c00 0x80>;
|
||||
compatible = "fsl,mpc8560-cpm-pic", "fsl,cpm2-pic";
|
||||
};
|
||||
|
||||
serial0: serial@91a00 {
|
||||
device_type = "serial";
|
||||
compatible = "fsl,mpc8560-scc-uart",
|
||||
"fsl,cpm2-scc-uart";
|
||||
reg = <0x91a00 0x20 0x88000 0x100>;
|
||||
fsl,cpm-brg = <1>;
|
||||
fsl,cpm-command = <0x800000>;
|
||||
current-speed = <115200>;
|
||||
interrupts = <40 8>;
|
||||
interrupt-parent = <&cpmpic>;
|
||||
};
|
||||
|
||||
serial1: serial@91a20 {
|
||||
device_type = "serial";
|
||||
compatible = "fsl,mpc8560-scc-uart",
|
||||
"fsl,cpm2-scc-uart";
|
||||
reg = <0x91a20 0x20 0x88100 0x100>;
|
||||
fsl,cpm-brg = <2>;
|
||||
fsl,cpm-command = <0x4a00000>;
|
||||
current-speed = <115200>;
|
||||
interrupts = <41 8>;
|
||||
interrupt-parent = <&cpmpic>;
|
||||
};
|
||||
|
||||
enet2: ethernet@91320 {
|
||||
device_type = "network";
|
||||
compatible = "fsl,mpc8560-fcc-enet",
|
||||
"fsl,cpm2-fcc-enet";
|
||||
reg = <0x91320 0x20 0x88500 0x100 0x913b0 0x1>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
fsl,cpm-command = <0x16200300>;
|
||||
interrupts = <33 8>;
|
||||
interrupt-parent = <&cpmpic>;
|
||||
phy-handle = <&phy2>;
|
||||
};
|
||||
|
||||
enet3: ethernet@91340 {
|
||||
device_type = "network";
|
||||
compatible = "fsl,mpc8560-fcc-enet",
|
||||
"fsl,cpm2-fcc-enet";
|
||||
reg = <0x91340 0x20 0x88600 0x100 0x913d0 0x1>;
|
||||
local-mac-address = [ 00 00 00 00 00 00 ];
|
||||
fsl,cpm-command = <0x1a400300>;
|
||||
interrupts = <34 8>;
|
||||
interrupt-parent = <&cpmpic>;
|
||||
phy-handle = <&phy3>;
|
||||
};
|
||||
};
|
||||
};
|
||||
|
||||
pci0: pci@e0008000 {
|
||||
#interrupt-cells = <1>;
|
||||
#size-cells = <2>;
|
||||
#address-cells = <3>;
|
||||
compatible = "fsl,mpc8540-pcix", "fsl,mpc8540-pci";
|
||||
device_type = "pci";
|
||||
reg = <0xe0008000 0x1000>;
|
||||
clock-frequency = <66666666>;
|
||||
interrupt-map-mask = <0xf800 0x0 0x0 0x7>;
|
||||
interrupt-map = <
|
||||
|
||||
/* IDSEL 0x2 */
|
||||
0x1000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x1000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x1000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x1000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 0x3 */
|
||||
0x1800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x1800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x1800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x1800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 0x4 */
|
||||
0x2000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0x2000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0x2000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0x2000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 0x5 */
|
||||
0x2800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0x2800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0x2800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0x2800 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 12 */
|
||||
0x6000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x6000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x6000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x6000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 13 */
|
||||
0x6800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x6800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x6800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x6800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 14*/
|
||||
0x7000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0x7000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0x7000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0x7000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 15 */
|
||||
0x7800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0x7800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0x7800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0x7800 0x0 0x0 0x4 &mpic 0x1 0x1
|
||||
|
||||
/* IDSEL 18 */
|
||||
0x9000 0x0 0x0 0x1 &mpic 0x1 0x1
|
||||
0x9000 0x0 0x0 0x2 &mpic 0x2 0x1
|
||||
0x9000 0x0 0x0 0x3 &mpic 0x3 0x1
|
||||
0x9000 0x0 0x0 0x4 &mpic 0x4 0x1
|
||||
|
||||
/* IDSEL 19 */
|
||||
0x9800 0x0 0x0 0x1 &mpic 0x4 0x1
|
||||
0x9800 0x0 0x0 0x2 &mpic 0x1 0x1
|
||||
0x9800 0x0 0x0 0x3 &mpic 0x2 0x1
|
||||
0x9800 0x0 0x0 0x4 &mpic 0x3 0x1
|
||||
|
||||
/* IDSEL 20 */
|
||||
0xa000 0x0 0x0 0x1 &mpic 0x3 0x1
|
||||
0xa000 0x0 0x0 0x2 &mpic 0x4 0x1
|
||||
0xa000 0x0 0x0 0x3 &mpic 0x1 0x1
|
||||
0xa000 0x0 0x0 0x4 &mpic 0x2 0x1
|
||||
|
||||
/* IDSEL 21 */
|
||||
0xa800 0x0 0x0 0x1 &mpic 0x2 0x1
|
||||
0xa800 0x0 0x0 0x2 &mpic 0x3 0x1
|
||||
0xa800 0x0 0x0 0x3 &mpic 0x4 0x1
|
||||
0xa800 0x0 0x0 0x4 &mpic 0x1 0x1>;
|
||||
|
||||
interrupt-parent = <&mpic>;
|
||||
interrupts = <24 2>;
|
||||
bus-range = <0 0>;
|
||||
ranges = <0x2000000 0x0 0x80000000 0x80000000 0x0 0x20000000
|
||||
0x1000000 0x0 0x0 0xe2000000 0x0 0x1000000>;
|
||||
};
|
||||
};
|
@ -1,47 +0,0 @@
|
||||
CONFIG_PPC_85xx=y
|
||||
CONFIG_SYSVIPC=y
|
||||
CONFIG_NO_HZ=y
|
||||
CONFIG_HIGH_RES_TIMERS=y
|
||||
CONFIG_LOG_BUF_SHIFT=14
|
||||
CONFIG_BLK_DEV_INITRD=y
|
||||
CONFIG_EXPERT=y
|
||||
# CONFIG_BLK_DEV_BSG is not set
|
||||
CONFIG_PARTITION_ADVANCED=y
|
||||
# CONFIG_MSDOS_PARTITION is not set
|
||||
CONFIG_MPC8540_ADS=y
|
||||
CONFIG_GEN_RTC=y
|
||||
CONFIG_BINFMT_MISC=y
|
||||
CONFIG_MATH_EMULATION=y
|
||||
# CONFIG_SECCOMP is not set
|
||||
CONFIG_NET=y
|
||||
CONFIG_PACKET=y
|
||||
CONFIG_UNIX=y
|
||||
CONFIG_XFRM_USER=y
|
||||
CONFIG_INET=y
|
||||
CONFIG_IP_MULTICAST=y
|
||||
CONFIG_IP_PNP=y
|
||||
CONFIG_IP_PNP_DHCP=y
|
||||
CONFIG_IP_PNP_BOOTP=y
|
||||
CONFIG_SYN_COOKIES=y
|
||||
# CONFIG_IPV6 is not set
|
||||
# CONFIG_FW_LOADER is not set
|
||||
CONFIG_BLK_DEV_LOOP=y
|
||||
CONFIG_BLK_DEV_RAM=y
|
||||
CONFIG_BLK_DEV_RAM_SIZE=32768
|
||||
CONFIG_NETDEVICES=y
|
||||
CONFIG_GIANFAR=y
|
||||
# CONFIG_INPUT_KEYBOARD is not set
|
||||
# CONFIG_INPUT_MOUSE is not set
|
||||
# CONFIG_SERIO is not set
|
||||
# CONFIG_VT is not set
|
||||
CONFIG_SERIAL_8250=y
|
||||
CONFIG_SERIAL_8250_CONSOLE=y
|
||||
# CONFIG_HW_RANDOM is not set
|
||||
CONFIG_EXT2_FS=y
|
||||
CONFIG_EXT4_FS=y
|
||||
CONFIG_PROC_KCORE=y
|
||||
CONFIG_TMPFS=y
|
||||
CONFIG_NFS_FS=y
|
||||
CONFIG_ROOT_NFS=y
|
||||
CONFIG_DETECT_HUNG_TASK=y
|
||||
CONFIG_DEBUG_MUTEXES=y
|
@ -1,50 +0,0 @@
|
||||
CONFIG_PPC_85xx=y
|
||||
CONFIG_SYSVIPC=y
|
||||
CONFIG_LOG_BUF_SHIFT=14
|
||||
CONFIG_BLK_DEV_INITRD=y
|
||||
CONFIG_EXPERT=y
|
||||
# CONFIG_BLK_DEV_BSG is not set
|
||||
CONFIG_PARTITION_ADVANCED=y
|
||||
# CONFIG_MSDOS_PARTITION is not set
|
||||
CONFIG_MPC8560_ADS=y
|
||||
CONFIG_GEN_RTC=y
|
||||
CONFIG_BINFMT_MISC=y
|
||||
CONFIG_MATH_EMULATION=y
|
||||
# CONFIG_SECCOMP is not set
|
||||
CONFIG_PCI=y
|
||||
CONFIG_PCI_DEBUG=y
|
||||
CONFIG_NET=y
|
||||
CONFIG_PACKET=y
|
||||
CONFIG_UNIX=y
|
||||
CONFIG_INET=y
|
||||
CONFIG_IP_MULTICAST=y
|
||||
CONFIG_IP_PNP=y
|
||||
CONFIG_IP_PNP_DHCP=y
|
||||
CONFIG_IP_PNP_BOOTP=y
|
||||
CONFIG_SYN_COOKIES=y
|
||||
# CONFIG_IPV6 is not set
|
||||
# CONFIG_FW_LOADER is not set
|
||||
CONFIG_BLK_DEV_LOOP=y
|
||||
CONFIG_BLK_DEV_RAM=y
|
||||
CONFIG_BLK_DEV_RAM_SIZE=32768
|
||||
CONFIG_NETDEVICES=y
|
||||
CONFIG_FS_ENET=y
|
||||
# CONFIG_FS_ENET_HAS_SCC is not set
|
||||
CONFIG_GIANFAR=y
|
||||
CONFIG_E1000=y
|
||||
CONFIG_DAVICOM_PHY=y
|
||||
CONFIG_MARVELL_PHY=y
|
||||
# CONFIG_INPUT_KEYBOARD is not set
|
||||
# CONFIG_INPUT_MOUSE is not set
|
||||
# CONFIG_SERIO is not set
|
||||
# CONFIG_VT is not set
|
||||
CONFIG_SERIAL_CPM=y
|
||||
CONFIG_SERIAL_CPM_CONSOLE=y
|
||||
CONFIG_EXT2_FS=y
|
||||
CONFIG_EXT4_FS=y
|
||||
CONFIG_PROC_KCORE=y
|
||||
CONFIG_TMPFS=y
|
||||
CONFIG_NFS_FS=y
|
||||
CONFIG_ROOT_NFS=y
|
||||
CONFIG_DETECT_HUNG_TASK=y
|
||||
CONFIG_DEBUG_MUTEXES=y
|
@ -1,52 +0,0 @@
|
||||
CONFIG_PPC_85xx=y
|
||||
CONFIG_SYSVIPC=y
|
||||
CONFIG_NO_HZ=y
|
||||
CONFIG_HIGH_RES_TIMERS=y
|
||||
CONFIG_LOG_BUF_SHIFT=14
|
||||
CONFIG_BLK_DEV_INITRD=y
|
||||
CONFIG_EXPERT=y
|
||||
# CONFIG_BLK_DEV_BSG is not set
|
||||
CONFIG_PARTITION_ADVANCED=y
|
||||
# CONFIG_MSDOS_PARTITION is not set
|
||||
CONFIG_MPC85xx_CDS=y
|
||||
CONFIG_GEN_RTC=y
|
||||
CONFIG_BINFMT_MISC=y
|
||||
CONFIG_MATH_EMULATION=y
|
||||
# CONFIG_SECCOMP is not set
|
||||
CONFIG_PCI=y
|
||||
CONFIG_NET=y
|
||||
CONFIG_PACKET=y
|
||||
CONFIG_UNIX=y
|
||||
CONFIG_XFRM_USER=y
|
||||
CONFIG_INET=y
|
||||
CONFIG_IP_MULTICAST=y
|
||||
CONFIG_IP_PNP=y
|
||||
CONFIG_IP_PNP_DHCP=y
|
||||
CONFIG_IP_PNP_BOOTP=y
|
||||
CONFIG_SYN_COOKIES=y
|
||||
# CONFIG_IPV6 is not set
|
||||
# CONFIG_FW_LOADER is not set
|
||||
CONFIG_BLK_DEV_LOOP=y
|
||||
CONFIG_BLK_DEV_RAM=y
|
||||
CONFIG_BLK_DEV_RAM_SIZE=32768
|
||||
CONFIG_ATA=y
|
||||
CONFIG_ATA_GENERIC=y
|
||||
CONFIG_PATA_VIA=y
|
||||
CONFIG_NETDEVICES=y
|
||||
CONFIG_GIANFAR=y
|
||||
CONFIG_E1000=y
|
||||
# CONFIG_INPUT_KEYBOARD is not set
|
||||
# CONFIG_INPUT_MOUSE is not set
|
||||
# CONFIG_SERIO is not set
|
||||
# CONFIG_VT is not set
|
||||
CONFIG_SERIAL_8250=y
|
||||
CONFIG_SERIAL_8250_CONSOLE=y
|
||||
# CONFIG_HW_RANDOM is not set
|
||||
CONFIG_EXT2_FS=y
|
||||
CONFIG_EXT4_FS=y
|
||||
CONFIG_PROC_KCORE=y
|
||||
CONFIG_TMPFS=y
|
||||
CONFIG_NFS_FS=y
|
||||
CONFIG_ROOT_NFS=y
|
||||
CONFIG_DETECT_HUNG_TASK=y
|
||||
CONFIG_DEBUG_MUTEXES=y
|
@ -1,8 +1,5 @@
|
||||
CONFIG_MATH_EMULATION=y
|
||||
CONFIG_MPC8536_DS=y
|
||||
CONFIG_MPC8540_ADS=y
|
||||
CONFIG_MPC8560_ADS=y
|
||||
CONFIG_MPC85xx_CDS=y
|
||||
CONFIG_MPC85xx_DS=y
|
||||
CONFIG_MPC85xx_MDS=y
|
||||
CONFIG_MPC85xx_RDB=y
|
||||
|
@ -21,6 +21,11 @@ static inline void reset_sprs(void)
|
||||
plpar_set_ciabr(0);
|
||||
}
|
||||
|
||||
if (cpu_has_feature(CPU_FTR_ARCH_31)) {
|
||||
mtspr(SPRN_DEXCR, 0);
|
||||
mtspr(SPRN_HASHKEYR, 0);
|
||||
}
|
||||
|
||||
/* Do we need isync()? We are going via a kexec reset */
|
||||
isync();
|
||||
}
|
||||
|
@ -194,6 +194,7 @@
|
||||
#else /* !__ASSEMBLY__ */
|
||||
|
||||
#include <linux/jump_label.h>
|
||||
#include <linux/sched.h>
|
||||
|
||||
DECLARE_STATIC_KEY_FALSE(uaccess_flush_key);
|
||||
|
||||
|
@ -192,6 +192,7 @@ static inline void cpu_feature_keys_init(void) { }
|
||||
#define CPU_FTR_P9_RADIX_PREFETCH_BUG LONG_ASM_CONST(0x0002000000000000)
|
||||
#define CPU_FTR_ARCH_31 LONG_ASM_CONST(0x0004000000000000)
|
||||
#define CPU_FTR_DAWR1 LONG_ASM_CONST(0x0008000000000000)
|
||||
#define CPU_FTR_DEXCR_NPHIE LONG_ASM_CONST(0x0010000000000000)
|
||||
|
||||
#ifndef __ASSEMBLY__
|
||||
|
||||
@ -451,7 +452,8 @@ static inline void cpu_feature_keys_init(void) { }
|
||||
CPU_FTR_CFAR | CPU_FTR_HVMODE | CPU_FTR_VMX_COPY | \
|
||||
CPU_FTR_DBELL | CPU_FTR_HAS_PPR | CPU_FTR_ARCH_207S | \
|
||||
CPU_FTR_ARCH_300 | CPU_FTR_ARCH_31 | \
|
||||
CPU_FTR_DAWR | CPU_FTR_DAWR1)
|
||||
CPU_FTR_DAWR | CPU_FTR_DAWR1 | \
|
||||
CPU_FTR_DEXCR_NPHIE)
|
||||
#define CPU_FTRS_CELL (CPU_FTR_LWSYNC | \
|
||||
CPU_FTR_PPCAS_ARCH_V2 | CPU_FTR_CTRL | \
|
||||
CPU_FTR_ALTIVEC_COMP | CPU_FTR_MMCRA | CPU_FTR_SMT | \
|
||||
|
@ -12,8 +12,14 @@
|
||||
|
||||
/*
|
||||
* This is used to ensure we don't load something for the wrong architecture.
|
||||
* 64le only supports ELFv2 64-bit binaries (64be supports v1 and v2).
|
||||
*/
|
||||
#if defined(CONFIG_PPC64) && defined(CONFIG_CPU_LITTLE_ENDIAN)
|
||||
#define elf_check_arch(x) (((x)->e_machine == ELF_ARCH) && \
|
||||
(((x)->e_flags & 0x3) == 0x2))
|
||||
#else
|
||||
#define elf_check_arch(x) ((x)->e_machine == ELF_ARCH)
|
||||
#endif
|
||||
#define compat_elf_check_arch(x) ((x)->e_machine == EM_PPC)
|
||||
|
||||
#define CORE_DUMP_USE_REGSET
|
||||
|
@ -548,12 +548,12 @@ static inline void kvmppc_set_host_ipi(int cpu)
|
||||
* pairs with the barrier in kvmppc_clear_host_ipi()
|
||||
*/
|
||||
smp_mb();
|
||||
paca_ptrs[cpu]->kvm_hstate.host_ipi = 1;
|
||||
WRITE_ONCE(paca_ptrs[cpu]->kvm_hstate.host_ipi, 1);
|
||||
}
|
||||
|
||||
static inline void kvmppc_clear_host_ipi(int cpu)
|
||||
{
|
||||
paca_ptrs[cpu]->kvm_hstate.host_ipi = 0;
|
||||
WRITE_ONCE(paca_ptrs[cpu]->kvm_hstate.host_ipi, 0);
|
||||
/*
|
||||
* order clearing of host_ipi flag vs. processing of IPI messages
|
||||
*
|
||||
|
@ -285,47 +285,6 @@ extern int mpc52xx_gpt_start_timer(struct mpc52xx_gpt_priv *gpt, u64 period,
|
||||
extern u64 mpc52xx_gpt_timer_period(struct mpc52xx_gpt_priv *gpt);
|
||||
extern int mpc52xx_gpt_stop_timer(struct mpc52xx_gpt_priv *gpt);
|
||||
|
||||
/* mpc52xx_lpbfifo.c */
|
||||
#define MPC52XX_LPBFIFO_FLAG_READ (0)
|
||||
#define MPC52XX_LPBFIFO_FLAG_WRITE (1<<0)
|
||||
#define MPC52XX_LPBFIFO_FLAG_NO_INCREMENT (1<<1)
|
||||
#define MPC52XX_LPBFIFO_FLAG_NO_DMA (1<<2)
|
||||
#define MPC52XX_LPBFIFO_FLAG_POLL_DMA (1<<3)
|
||||
|
||||
struct mpc52xx_lpbfifo_request {
|
||||
struct list_head list;
|
||||
|
||||
/* localplus bus address */
|
||||
unsigned int cs;
|
||||
size_t offset;
|
||||
|
||||
/* Memory address */
|
||||
void *data;
|
||||
phys_addr_t data_phys;
|
||||
|
||||
/* Details of transfer */
|
||||
size_t size;
|
||||
size_t pos; /* current position of transfer */
|
||||
int flags;
|
||||
int defer_xfer_start;
|
||||
|
||||
/* What to do when finished */
|
||||
void (*callback)(struct mpc52xx_lpbfifo_request *);
|
||||
|
||||
void *priv; /* Driver private data */
|
||||
|
||||
/* statistics */
|
||||
int irq_count;
|
||||
int irq_ticks;
|
||||
u8 last_byte;
|
||||
int buffer_not_done_cnt;
|
||||
};
|
||||
|
||||
extern int mpc52xx_lpbfifo_submit(struct mpc52xx_lpbfifo_request *req);
|
||||
extern void mpc52xx_lpbfifo_abort(struct mpc52xx_lpbfifo_request *req);
|
||||
extern void mpc52xx_lpbfifo_poll(void);
|
||||
extern int mpc52xx_lpbfifo_start_xfer(struct mpc52xx_lpbfifo_request *req);
|
||||
|
||||
/* mpc52xx_pic.c */
|
||||
extern void mpc52xx_init_irq(void);
|
||||
extern unsigned int mpc52xx_get_irq(void);
|
||||
|
@ -112,9 +112,6 @@ int64_t opal_pci_set_pe(uint64_t phb_id, uint64_t pe_number, uint64_t bus_dev_fu
|
||||
uint8_t pe_action);
|
||||
int64_t opal_pci_set_peltv(uint64_t phb_id, uint32_t parent_pe, uint32_t child_pe,
|
||||
uint8_t state);
|
||||
int64_t opal_pci_set_mve(uint64_t phb_id, uint32_t mve_number, uint32_t pe_number);
|
||||
int64_t opal_pci_set_mve_enable(uint64_t phb_id, uint32_t mve_number,
|
||||
uint32_t state);
|
||||
int64_t opal_pci_get_xive_reissue(uint64_t phb_id, uint32_t xive_number,
|
||||
uint8_t *p_bit, uint8_t *q_bit);
|
||||
int64_t opal_pci_set_xive_reissue(uint64_t phb_id, uint32_t xive_number,
|
||||
|
@ -191,6 +191,7 @@ struct paca_struct {
|
||||
#ifdef CONFIG_PPC_POWERNV
|
||||
/* PowerNV idle fields */
|
||||
/* PNV_CORE_IDLE_* bits, all siblings work on thread 0 paca */
|
||||
unsigned long idle_lock; /* A value of 1 means acquired */
|
||||
unsigned long idle_state;
|
||||
union {
|
||||
/* P7/P8 specific fields */
|
||||
|
@ -222,6 +222,7 @@
|
||||
#define OP_31_XOP_STFSX 663
|
||||
#define OP_31_XOP_STFSUX 695
|
||||
#define OP_31_XOP_STFDX 727
|
||||
#define OP_31_XOP_HASHCHK 754
|
||||
#define OP_31_XOP_STFDUX 759
|
||||
#define OP_31_XOP_LHBRX 790
|
||||
#define OP_31_XOP_LFIWAX 855
|
||||
|
@ -406,6 +406,15 @@ n:
|
||||
/* offsets for stack frame layout */
|
||||
#define LRSAVE 16
|
||||
|
||||
/*
|
||||
* GCC stack frames follow a different pattern on 32 vs 64. This can be used
|
||||
* to make asm frames be consistent with C.
|
||||
*/
|
||||
#define PPC_CREATE_STACK_FRAME(size) \
|
||||
mflr r0; \
|
||||
std r0,16(r1); \
|
||||
stdu r1,-(size)(r1)
|
||||
|
||||
#else /* 32-bit */
|
||||
|
||||
#define LOAD_REG_IMMEDIATE(reg, expr) __LOAD_REG_IMMEDIATE_32 reg, expr
|
||||
@ -422,6 +431,11 @@ n:
|
||||
/* offsets for stack frame layout */
|
||||
#define LRSAVE 4
|
||||
|
||||
#define PPC_CREATE_STACK_FRAME(size) \
|
||||
stwu r1,-(size)(r1); \
|
||||
mflr r0; \
|
||||
stw r0,(size+4)(r1)
|
||||
|
||||
#endif
|
||||
|
||||
/* various errata or part fixups */
|
||||
|
@ -264,6 +264,7 @@ struct thread_struct {
|
||||
unsigned long mmcr3;
|
||||
unsigned long sier2;
|
||||
unsigned long sier3;
|
||||
unsigned long hashkeyr;
|
||||
|
||||
#endif
|
||||
};
|
||||
|
@ -180,8 +180,8 @@ void do_syscall_trace_leave(struct pt_regs *regs);
|
||||
static inline void set_return_regs_changed(void)
|
||||
{
|
||||
#ifdef CONFIG_PPC_BOOK3S_64
|
||||
local_paca->hsrr_valid = 0;
|
||||
local_paca->srr_valid = 0;
|
||||
WRITE_ONCE(local_paca->hsrr_valid, 0);
|
||||
WRITE_ONCE(local_paca->srr_valid, 0);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
@ -382,7 +382,17 @@
|
||||
#define SPRN_HIOR 0x137 /* 970 Hypervisor interrupt offset */
|
||||
#define SPRN_RMOR 0x138 /* Real mode offset register */
|
||||
#define SPRN_HRMOR 0x139 /* Real mode offset register */
|
||||
#define SPRN_HDEXCR_RO 0x1C7 /* Hypervisor DEXCR (non-privileged, readonly) */
|
||||
#define SPRN_HASHKEYR 0x1D4 /* Non-privileged hashst/hashchk key register */
|
||||
#define SPRN_HDEXCR 0x1D7 /* Hypervisor dynamic execution control register */
|
||||
#define SPRN_DEXCR_RO 0x32C /* DEXCR (non-privileged, readonly) */
|
||||
#define SPRN_ASDR 0x330 /* Access segment descriptor register */
|
||||
#define SPRN_DEXCR 0x33C /* Dynamic execution control register */
|
||||
#define DEXCR_PR_SBHE 0x80000000UL /* 0: Speculative Branch Hint Enable */
|
||||
#define DEXCR_PR_IBRTPD 0x10000000UL /* 3: Indirect Branch Recurrent Target Prediction Disable */
|
||||
#define DEXCR_PR_SRAPD 0x08000000UL /* 4: Subroutine Return Address Prediction Disable */
|
||||
#define DEXCR_PR_NPHIE 0x04000000UL /* 5: Non-Privileged Hash Instruction Enable */
|
||||
#define DEXCR_INIT DEXCR_PR_NPHIE /* Fixed DEXCR value to initialise all CPUs with */
|
||||
#define SPRN_IC 0x350 /* Virtual Instruction Count */
|
||||
#define SPRN_VTB 0x351 /* Virtual Time Base */
|
||||
#define SPRN_LDBAR 0x352 /* LD Base Address Register */
|
||||
|
@ -15,6 +15,7 @@
|
||||
* (the type definitions are in asm/simple_spinlock_types.h)
|
||||
*/
|
||||
#include <linux/irqflags.h>
|
||||
#include <linux/kcsan-checks.h>
|
||||
#include <asm/paravirt.h>
|
||||
#include <asm/paca.h>
|
||||
#include <asm/synch.h>
|
||||
@ -126,6 +127,7 @@ static inline void arch_spin_lock(arch_spinlock_t *lock)
|
||||
|
||||
static inline void arch_spin_unlock(arch_spinlock_t *lock)
|
||||
{
|
||||
kcsan_mb();
|
||||
__asm__ __volatile__("# arch_spin_unlock\n\t"
|
||||
PPC_RELEASE_BARRIER: : :"memory");
|
||||
lock->slock = 0;
|
||||
|
@ -183,9 +183,13 @@ static inline bool test_thread_local_flags(unsigned int flags)
|
||||
#define clear_tsk_compat_task(tsk) do { } while (0)
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_PPC64)
|
||||
#ifdef CONFIG_PPC64
|
||||
#ifdef CONFIG_CPU_BIG_ENDIAN
|
||||
#define is_elf2_task() (test_thread_flag(TIF_ELF2ABI))
|
||||
#else
|
||||
#define is_elf2_task() (1)
|
||||
#endif
|
||||
#else
|
||||
#define is_elf2_task() (0)
|
||||
#endif
|
||||
|
||||
|
@ -98,6 +98,8 @@
|
||||
#define ELF_NEBB 3 /* includes ebbrr, ebbhr, bescr */
|
||||
#define ELF_NPMU 5 /* includes siar, sdar, sier, mmcr2, mmcr0 */
|
||||
#define ELF_NPKEY 3 /* includes amr, iamr, uamor */
|
||||
#define ELF_NDEXCR 2 /* includes dexcr, hdexcr */
|
||||
#define ELF_NHASHKEYR 1 /* includes hashkeyr */
|
||||
|
||||
typedef unsigned long elf_greg_t64;
|
||||
typedef elf_greg_t64 elf_gregset_t64[ELF_NGREG];
|
||||
|
@ -68,7 +68,7 @@ CFLAGS_REMOVE_syscall.o = -fstack-protector -fstack-protector-strong
|
||||
CFLAGS_syscall.o += -fno-stack-protector
|
||||
#endif
|
||||
|
||||
obj-y := cputable.o syscalls.o \
|
||||
obj-y := cputable.o syscalls.o switch.o \
|
||||
irq.o align.o signal_$(BITS).o pmc.o vdso.o \
|
||||
process.o systbl.o idle.o \
|
||||
signal.o sysfs.o cacheinfo.o time.o \
|
||||
@ -165,9 +165,6 @@ endif
|
||||
|
||||
obj64-$(CONFIG_PPC_TRANSACTIONAL_MEM) += tm.o
|
||||
|
||||
obj-$(CONFIG_PPC64) += $(obj64-y)
|
||||
obj-$(CONFIG_PPC32) += $(obj32-y)
|
||||
|
||||
ifneq ($(CONFIG_XMON)$(CONFIG_KEXEC_CORE)(CONFIG_PPC_BOOK3S),)
|
||||
obj-y += ppc_save_regs.o
|
||||
endif
|
||||
@ -209,11 +206,14 @@ CFLAGS_paca.o += -fno-stack-protector
|
||||
|
||||
obj-$(CONFIG_PPC_FPU) += fpu.o
|
||||
obj-$(CONFIG_ALTIVEC) += vector.o
|
||||
obj-$(CONFIG_PPC64) += entry_64.o
|
||||
obj-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_init.o
|
||||
|
||||
obj-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_init.o
|
||||
obj64-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_entry_64.o
|
||||
extra-$(CONFIG_PPC_OF_BOOT_TRAMPOLINE) += prom_init_check
|
||||
|
||||
obj-$(CONFIG_PPC64) += $(obj64-y)
|
||||
obj-$(CONFIG_PPC32) += $(obj32-y)
|
||||
|
||||
quiet_cmd_prom_init_check = PROMCHK $@
|
||||
cmd_prom_init_check = $(CONFIG_SHELL) $< "$(NM)" $(obj)/prom_init.o; touch $@
|
||||
|
||||
|
@ -126,6 +126,12 @@ static void init_PMU_ISA31(void)
|
||||
mtspr(SPRN_MMCR0, MMCR0_FC | MMCR0_PMCCEXT);
|
||||
}
|
||||
|
||||
static void init_DEXCR(void)
|
||||
{
|
||||
mtspr(SPRN_DEXCR, DEXCR_INIT);
|
||||
mtspr(SPRN_HASHKEYR, 0);
|
||||
}
|
||||
|
||||
/*
|
||||
* Note that we can be called twice of pseudo-PVRs.
|
||||
* The parameter offset is not used.
|
||||
@ -241,6 +247,7 @@ void __setup_cpu_power10(unsigned long offset, struct cpu_spec *t)
|
||||
init_FSCR_power10();
|
||||
init_PMU();
|
||||
init_PMU_ISA31();
|
||||
init_DEXCR();
|
||||
|
||||
if (!init_hvmode_206(t))
|
||||
return;
|
||||
@ -263,6 +270,7 @@ void __restore_cpu_power10(void)
|
||||
init_FSCR_power10();
|
||||
init_PMU();
|
||||
init_PMU_ISA31();
|
||||
init_DEXCR();
|
||||
|
||||
msr = mfmsr();
|
||||
if (!(msr & MSR_HV))
|
||||
|
@ -211,64 +211,6 @@ start_kernel_thread:
|
||||
100: trap
|
||||
EMIT_BUG_ENTRY 100b,__FILE__,__LINE__,0
|
||||
|
||||
|
||||
/*
|
||||
* This routine switches between two different tasks. The process
|
||||
* state of one is saved on its kernel stack. Then the state
|
||||
* of the other is restored from its kernel stack. The memory
|
||||
* management hardware is updated to the second process's state.
|
||||
* Finally, we can return to the second process.
|
||||
* On entry, r3 points to the THREAD for the current task, r4
|
||||
* points to the THREAD for the new task.
|
||||
*
|
||||
* This routine is always called with interrupts disabled.
|
||||
*
|
||||
* Note: there are two ways to get to the "going out" portion
|
||||
* of this code; either by coming in via the entry (_switch)
|
||||
* or via "fork" which must set up an environment equivalent
|
||||
* to the "_switch" path. If you change this , you'll have to
|
||||
* change the fork code also.
|
||||
*
|
||||
* The code which creates the new task context is in 'copy_thread'
|
||||
* in arch/ppc/kernel/process.c
|
||||
*/
|
||||
_GLOBAL(_switch)
|
||||
stwu r1,-SWITCH_FRAME_SIZE(r1)
|
||||
mflr r0
|
||||
stw r0,SWITCH_FRAME_SIZE+4(r1)
|
||||
/* r3-r12 are caller saved -- Cort */
|
||||
SAVE_NVGPRS(r1)
|
||||
stw r0,_NIP(r1) /* Return to switch caller */
|
||||
mfcr r10
|
||||
stw r10,_CCR(r1)
|
||||
stw r1,KSP(r3) /* Set old stack pointer */
|
||||
|
||||
#ifdef CONFIG_SMP
|
||||
/* We need a sync somewhere here to make sure that if the
|
||||
* previous task gets rescheduled on another CPU, it sees all
|
||||
* stores it has performed on this one.
|
||||
*/
|
||||
sync
|
||||
#endif /* CONFIG_SMP */
|
||||
|
||||
tophys(r0,r4)
|
||||
mtspr SPRN_SPRG_THREAD,r0 /* Update current THREAD phys addr */
|
||||
lwz r1,KSP(r4) /* Load new stack pointer */
|
||||
|
||||
/* save the old current 'last' for return value */
|
||||
mr r3,r2
|
||||
addi r2,r4,-THREAD /* Update current */
|
||||
|
||||
lwz r0,_CCR(r1)
|
||||
mtcrf 0xFF,r0
|
||||
/* r3-r12 are destroyed -- Cort */
|
||||
REST_NVGPRS(r1)
|
||||
|
||||
lwz r4,_NIP(r1) /* Return to _switch caller in new task */
|
||||
mtlr r4
|
||||
addi r1,r1,SWITCH_FRAME_SIZE
|
||||
blr
|
||||
|
||||
.globl fast_exception_return
|
||||
fast_exception_return:
|
||||
#if !(defined(CONFIG_4xx) || defined(CONFIG_BOOKE))
|
||||
|
@ -125,7 +125,7 @@ static notrace void check_return_regs_valid(struct pt_regs *regs)
|
||||
case 0x1600:
|
||||
case 0x1800:
|
||||
validp = &local_paca->hsrr_valid;
|
||||
if (!*validp)
|
||||
if (!READ_ONCE(*validp))
|
||||
return;
|
||||
|
||||
srr0 = mfspr(SPRN_HSRR0);
|
||||
@ -135,7 +135,7 @@ static notrace void check_return_regs_valid(struct pt_regs *regs)
|
||||
break;
|
||||
default:
|
||||
validp = &local_paca->srr_valid;
|
||||
if (!*validp)
|
||||
if (!READ_ONCE(*validp))
|
||||
return;
|
||||
|
||||
srr0 = mfspr(SPRN_SRR0);
|
||||
@ -161,19 +161,17 @@ static notrace void check_return_regs_valid(struct pt_regs *regs)
|
||||
* such things will get caught most of the time, statistically
|
||||
* enough to be able to get a warning out.
|
||||
*/
|
||||
barrier();
|
||||
|
||||
if (!*validp)
|
||||
if (!READ_ONCE(*validp))
|
||||
return;
|
||||
|
||||
if (!warned) {
|
||||
warned = true;
|
||||
if (!data_race(warned)) {
|
||||
data_race(warned = true);
|
||||
printk("%sSRR0 was: %lx should be: %lx\n", h, srr0, regs->nip);
|
||||
printk("%sSRR1 was: %lx should be: %lx\n", h, srr1, regs->msr);
|
||||
show_regs(regs);
|
||||
}
|
||||
|
||||
*validp = 0; /* fixup */
|
||||
WRITE_ONCE(*validp, 0); /* fixup */
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -368,7 +366,6 @@ void preempt_schedule_irq(void);
|
||||
|
||||
notrace unsigned long interrupt_exit_kernel_prepare(struct pt_regs *regs)
|
||||
{
|
||||
unsigned long flags;
|
||||
unsigned long ret = 0;
|
||||
unsigned long kuap;
|
||||
bool stack_store = read_thread_flags() & _TIF_EMULATE_STACK_STORE;
|
||||
@ -392,7 +389,7 @@ notrace unsigned long interrupt_exit_kernel_prepare(struct pt_regs *regs)
|
||||
|
||||
kuap = kuap_get_and_assert_locked();
|
||||
|
||||
local_irq_save(flags);
|
||||
local_irq_disable();
|
||||
|
||||
if (!arch_irq_disabled_regs(regs)) {
|
||||
/* Returning to a kernel context with local irqs enabled. */
|
||||
|
@ -1090,6 +1090,7 @@ void iommu_tce_kill(struct iommu_table *tbl,
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iommu_tce_kill);
|
||||
|
||||
#if defined(CONFIG_PPC_PSERIES) || defined(CONFIG_PPC_POWERNV)
|
||||
static int iommu_take_ownership(struct iommu_table *tbl)
|
||||
{
|
||||
unsigned long flags, i, sz = (tbl->it_size + 7) >> 3;
|
||||
@ -1140,6 +1141,7 @@ static void iommu_release_ownership(struct iommu_table *tbl)
|
||||
spin_unlock(&tbl->pools[i].lock);
|
||||
spin_unlock_irqrestore(&tbl->large_pool.lock, flags);
|
||||
}
|
||||
#endif
|
||||
|
||||
int iommu_add_device(struct iommu_table_group *table_group, struct device *dev)
|
||||
{
|
||||
@ -1171,6 +1173,7 @@ int iommu_add_device(struct iommu_table_group *table_group, struct device *dev)
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(iommu_add_device);
|
||||
|
||||
#if defined(CONFIG_PPC_PSERIES) || defined(CONFIG_PPC_POWERNV)
|
||||
/*
|
||||
* A simple iommu_table_group_ops which only allows reusing the existing
|
||||
* iommu_table. This handles VFIO for POWER7 or the nested KVM.
|
||||
@ -1398,5 +1401,6 @@ static int __init spapr_tce_setup_phb_iommus_initcall(void)
|
||||
return 0;
|
||||
}
|
||||
postcore_initcall_sync(spapr_tce_setup_phb_iommus_initcall);
|
||||
#endif
|
||||
|
||||
#endif /* CONFIG_IOMMU_API */
|
||||
|
@ -31,10 +31,10 @@ _GLOBAL(ppc_save_regs)
|
||||
lbz r0,PACAIRQSOFTMASK(r13)
|
||||
PPC_STL r0,SOFTE(r3)
|
||||
#endif
|
||||
/* go up one stack frame for SP */
|
||||
PPC_LL r4,0(r1)
|
||||
PPC_STL r4,GPR1(r3)
|
||||
/* store current SP */
|
||||
PPC_STL r1,GPR1(r3)
|
||||
/* get caller's LR */
|
||||
PPC_LL r4,0(r1)
|
||||
PPC_LL r0,LRSAVE(r4)
|
||||
PPC_STL r0,_LINK(r3)
|
||||
mflr r0
|
||||
|
@ -1182,6 +1182,9 @@ static inline void save_sprs(struct thread_struct *t)
|
||||
*/
|
||||
t->tar = mfspr(SPRN_TAR);
|
||||
}
|
||||
|
||||
if (cpu_has_feature(CPU_FTR_DEXCR_NPHIE))
|
||||
t->hashkeyr = mfspr(SPRN_HASHKEYR);
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -1260,6 +1263,10 @@ static inline void restore_sprs(struct thread_struct *old_thread,
|
||||
if (cpu_has_feature(CPU_FTR_P9_TIDR) &&
|
||||
old_thread->tidr != new_thread->tidr)
|
||||
mtspr(SPRN_TIDR, new_thread->tidr);
|
||||
|
||||
if (cpu_has_feature(CPU_FTR_DEXCR_NPHIE) &&
|
||||
old_thread->hashkeyr != new_thread->hashkeyr)
|
||||
mtspr(SPRN_HASHKEYR, new_thread->hashkeyr);
|
||||
#endif
|
||||
|
||||
}
|
||||
@ -1867,6 +1874,10 @@ int copy_thread(struct task_struct *p, const struct kernel_clone_args *args)
|
||||
}
|
||||
|
||||
p->thread.tidr = 0;
|
||||
#endif
|
||||
#ifdef CONFIG_PPC_BOOK3S_64
|
||||
if (cpu_has_feature(CPU_FTR_DEXCR_NPHIE))
|
||||
p->thread.hashkeyr = current->thread.hashkeyr;
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
@ -1984,6 +1995,12 @@ void start_thread(struct pt_regs *regs, unsigned long start, unsigned long sp)
|
||||
current->thread.tm_tfiar = 0;
|
||||
current->thread.load_tm = 0;
|
||||
#endif /* CONFIG_PPC_TRANSACTIONAL_MEM */
|
||||
#ifdef CONFIG_PPC_BOOK3S_64
|
||||
if (cpu_has_feature(CPU_FTR_DEXCR_NPHIE)) {
|
||||
current->thread.hashkeyr = get_random_long();
|
||||
mtspr(SPRN_HASHKEYR, current->thread.hashkeyr);
|
||||
}
|
||||
#endif /* CONFIG_PPC_BOOK3S_64 */
|
||||
}
|
||||
EXPORT_SYMBOL(start_thread);
|
||||
|
||||
|
@ -182,6 +182,7 @@ static struct ibm_feature ibm_pa_features[] __initdata = {
|
||||
.cpu_user_ftrs2 = PPC_FEATURE2_HTM_COMP | PPC_FEATURE2_HTM_NOSC_COMP },
|
||||
|
||||
{ .pabyte = 64, .pabit = 0, .cpu_features = CPU_FTR_DAWR1 },
|
||||
{ .pabyte = 68, .pabit = 5, .cpu_features = CPU_FTR_DEXCR_NPHIE },
|
||||
};
|
||||
|
||||
/*
|
||||
|
87
arch/powerpc/kernel/prom_entry_64.S
Normal file
87
arch/powerpc/kernel/prom_entry_64.S
Normal file
@ -0,0 +1,87 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* PowerPC version
|
||||
* Copyright (C) 1995-1996 Gary Thomas (gdt@linuxppc.org)
|
||||
* Rewritten by Cort Dougan (cort@cs.nmt.edu) for PReP
|
||||
* Copyright (C) 1996 Cort Dougan <cort@cs.nmt.edu>
|
||||
* Adapted for Power Macintosh by Paul Mackerras.
|
||||
* Low-level exception handlers and MMU support
|
||||
* rewritten by Paul Mackerras.
|
||||
* Copyright (C) 1996 Paul Mackerras.
|
||||
* MPC8xx modifications Copyright (C) 1997 Dan Malek (dmalek@jlc.net).
|
||||
*
|
||||
* This file contains the 64-bit prom entry code.
|
||||
*/
|
||||
#include <asm/asm-offsets.h>
|
||||
#ifdef CONFIG_PPC_BOOK3S
|
||||
#include <asm/exception-64s.h>
|
||||
#else
|
||||
#include <asm/exception-64e.h>
|
||||
#endif
|
||||
#include <asm/ppc_asm.h>
|
||||
|
||||
.section ".text","ax",@progbits
|
||||
|
||||
_GLOBAL(enter_prom)
|
||||
mflr r0
|
||||
std r0,16(r1)
|
||||
stdu r1,-SWITCH_FRAME_SIZE(r1) /* Save SP and create stack space */
|
||||
|
||||
/* Because PROM is running in 32b mode, it clobbers the high order half
|
||||
* of all registers that it saves. We therefore save those registers
|
||||
* PROM might touch to the stack. (r0, r3-r13 are caller saved)
|
||||
*/
|
||||
SAVE_GPR(2, r1)
|
||||
SAVE_GPR(13, r1)
|
||||
SAVE_NVGPRS(r1)
|
||||
mfcr r10
|
||||
mfmsr r11
|
||||
std r10,_CCR(r1)
|
||||
std r11,_MSR(r1)
|
||||
|
||||
/* Put PROM address in SRR0 */
|
||||
mtsrr0 r4
|
||||
|
||||
/* Setup our trampoline return addr in LR */
|
||||
bcl 20,31,$+4
|
||||
0: mflr r4
|
||||
addi r4,r4,(1f - 0b)
|
||||
mtlr r4
|
||||
|
||||
/* Prepare a 32-bit mode big endian MSR
|
||||
*/
|
||||
#ifdef CONFIG_PPC_BOOK3E_64
|
||||
rlwinm r11,r11,0,1,31
|
||||
mtsrr1 r11
|
||||
rfi
|
||||
#else /* CONFIG_PPC_BOOK3E_64 */
|
||||
LOAD_REG_IMMEDIATE(r12, MSR_SF | MSR_LE)
|
||||
andc r11,r11,r12
|
||||
mtsrr1 r11
|
||||
RFI_TO_KERNEL
|
||||
#endif /* CONFIG_PPC_BOOK3E_64 */
|
||||
|
||||
1: /* Return from OF */
|
||||
FIXUP_ENDIAN
|
||||
|
||||
/* Just make sure that r1 top 32 bits didn't get
|
||||
* corrupt by OF
|
||||
*/
|
||||
rldicl r1,r1,0,32
|
||||
|
||||
/* Restore the MSR (back to 64 bits) */
|
||||
ld r0,_MSR(r1)
|
||||
MTMSRD(r0)
|
||||
isync
|
||||
|
||||
/* Restore other registers */
|
||||
REST_GPR(2, r1)
|
||||
REST_GPR(13, r1)
|
||||
REST_NVGPRS(r1)
|
||||
ld r4,_CCR(r1)
|
||||
mtcr r4
|
||||
|
||||
addi r1,r1,SWITCH_FRAME_SIZE
|
||||
ld r0,16(r1)
|
||||
mtlr r0
|
||||
blr
|
@ -1,5 +1,7 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
|
||||
#include <linux/regset.h>
|
||||
|
||||
/*
|
||||
* Set of msr bits that gdb can change on behalf of a process.
|
||||
*/
|
||||
@ -55,6 +57,10 @@ enum powerpc_regset {
|
||||
REGSET_TAR, /* TAR register */
|
||||
REGSET_EBB, /* EBB registers */
|
||||
REGSET_PMR, /* Performance Monitor Registers */
|
||||
REGSET_DEXCR, /* DEXCR registers */
|
||||
#ifdef CONFIG_CHECKPOINT_RESTORE
|
||||
REGSET_HASHKEYR, /* HASHKEYR register */
|
||||
#endif
|
||||
#endif
|
||||
#ifdef CONFIG_PPC_MEM_KEYS
|
||||
REGSET_PKEY, /* AMR register */
|
||||
|
@ -454,7 +454,65 @@ static int pmu_set(struct task_struct *target, const struct user_regset *regset,
|
||||
5 * sizeof(unsigned long));
|
||||
return ret;
|
||||
}
|
||||
#endif
|
||||
|
||||
static int dexcr_active(struct task_struct *target, const struct user_regset *regset)
|
||||
{
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31))
|
||||
return -ENODEV;
|
||||
|
||||
return regset->n;
|
||||
}
|
||||
|
||||
static int dexcr_get(struct task_struct *target, const struct user_regset *regset,
|
||||
struct membuf to)
|
||||
{
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31))
|
||||
return -ENODEV;
|
||||
|
||||
/*
|
||||
* The DEXCR is currently static across all CPUs, so we don't
|
||||
* store the target's value anywhere, but the static value
|
||||
* will also be correct.
|
||||
*/
|
||||
membuf_store(&to, (u64)lower_32_bits(DEXCR_INIT));
|
||||
|
||||
/*
|
||||
* Technically the HDEXCR is per-cpu, but a hypervisor can't reasonably
|
||||
* change it between CPUs of the same guest.
|
||||
*/
|
||||
return membuf_store(&to, (u64)lower_32_bits(mfspr(SPRN_HDEXCR_RO)));
|
||||
}
|
||||
|
||||
#ifdef CONFIG_CHECKPOINT_RESTORE
|
||||
static int hashkeyr_active(struct task_struct *target, const struct user_regset *regset)
|
||||
{
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31))
|
||||
return -ENODEV;
|
||||
|
||||
return regset->n;
|
||||
}
|
||||
|
||||
static int hashkeyr_get(struct task_struct *target, const struct user_regset *regset,
|
||||
struct membuf to)
|
||||
{
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31))
|
||||
return -ENODEV;
|
||||
|
||||
return membuf_store(&to, target->thread.hashkeyr);
|
||||
}
|
||||
|
||||
static int hashkeyr_set(struct task_struct *target, const struct user_regset *regset,
|
||||
unsigned int pos, unsigned int count, const void *kbuf,
|
||||
const void __user *ubuf)
|
||||
{
|
||||
if (!cpu_has_feature(CPU_FTR_ARCH_31))
|
||||
return -ENODEV;
|
||||
|
||||
return user_regset_copyin(&pos, &count, &kbuf, &ubuf, &target->thread.hashkeyr,
|
||||
0, sizeof(unsigned long));
|
||||
}
|
||||
#endif /* CONFIG_CHECKPOINT_RESTORE */
|
||||
#endif /* CONFIG_PPC_BOOK3S_64 */
|
||||
|
||||
#ifdef CONFIG_PPC_MEM_KEYS
|
||||
static int pkey_active(struct task_struct *target, const struct user_regset *regset)
|
||||
@ -615,6 +673,18 @@ static const struct user_regset native_regsets[] = {
|
||||
.size = sizeof(u64), .align = sizeof(u64),
|
||||
.active = pmu_active, .regset_get = pmu_get, .set = pmu_set
|
||||
},
|
||||
[REGSET_DEXCR] = {
|
||||
.core_note_type = NT_PPC_DEXCR, .n = ELF_NDEXCR,
|
||||
.size = sizeof(u64), .align = sizeof(u64),
|
||||
.active = dexcr_active, .regset_get = dexcr_get
|
||||
},
|
||||
#ifdef CONFIG_CHECKPOINT_RESTORE
|
||||
[REGSET_HASHKEYR] = {
|
||||
.core_note_type = NT_PPC_HASHKEYR, .n = ELF_NHASHKEYR,
|
||||
.size = sizeof(u64), .align = sizeof(u64),
|
||||
.active = hashkeyr_active, .regset_get = hashkeyr_get, .set = hashkeyr_set
|
||||
},
|
||||
#endif
|
||||
#endif
|
||||
#ifdef CONFIG_PPC_MEM_KEYS
|
||||
[REGSET_PKEY] = {
|
||||
|
@ -264,7 +264,8 @@ static void prepare_save_user_regs(int ctx_has_vsx_region)
|
||||
#endif
|
||||
}
|
||||
|
||||
static int __unsafe_save_user_regs(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
static __always_inline int
|
||||
__unsafe_save_user_regs(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
struct mcontext __user *tm_frame, int ctx_has_vsx_region)
|
||||
{
|
||||
unsigned long msr = regs->msr;
|
||||
@ -364,7 +365,8 @@ static void prepare_save_tm_user_regs(void)
|
||||
current->thread.ckvrsave = mfspr(SPRN_VRSAVE);
|
||||
}
|
||||
|
||||
static int save_tm_user_regs_unsafe(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
static __always_inline int
|
||||
save_tm_user_regs_unsafe(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
struct mcontext __user *tm_frame, unsigned long msr)
|
||||
{
|
||||
/* Save both sets of general registers */
|
||||
@ -444,7 +446,8 @@ failed:
|
||||
#else
|
||||
static void prepare_save_tm_user_regs(void) { }
|
||||
|
||||
static int save_tm_user_regs_unsafe(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
static __always_inline int
|
||||
save_tm_user_regs_unsafe(struct pt_regs *regs, struct mcontext __user *frame,
|
||||
struct mcontext __user *tm_frame, unsigned long msr)
|
||||
{
|
||||
return 0;
|
||||
|
@ -291,7 +291,7 @@ void smp_muxed_ipi_set_message(int cpu, int msg)
|
||||
* Order previous accesses before accesses in the IPI handler.
|
||||
*/
|
||||
smp_mb();
|
||||
message[msg] = 1;
|
||||
WRITE_ONCE(message[msg], 1);
|
||||
}
|
||||
|
||||
void smp_muxed_ipi_message_pass(int cpu, int msg)
|
||||
@ -350,7 +350,7 @@ irqreturn_t smp_ipi_demux_relaxed(void)
|
||||
if (all & IPI_MESSAGE(PPC_MSG_NMI_IPI))
|
||||
nmi_ipi_action(0, NULL);
|
||||
#endif
|
||||
} while (info->messages);
|
||||
} while (READ_ONCE(info->messages));
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
@ -1,56 +1,21 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* PowerPC version
|
||||
* Copyright (C) 1995-1996 Gary Thomas (gdt@linuxppc.org)
|
||||
* Rewritten by Cort Dougan (cort@cs.nmt.edu) for PReP
|
||||
* Copyright (C) 1996 Cort Dougan <cort@cs.nmt.edu>
|
||||
* Adapted for Power Macintosh by Paul Mackerras.
|
||||
* Low-level exception handlers and MMU support
|
||||
* rewritten by Paul Mackerras.
|
||||
* Copyright (C) 1996 Paul Mackerras.
|
||||
* MPC8xx modifications Copyright (C) 1997 Dan Malek (dmalek@jlc.net).
|
||||
*
|
||||
* This file contains the system call entry code, context switch
|
||||
* code, and exception/interrupt return code for PowerPC.
|
||||
*/
|
||||
|
||||
#include <linux/objtool.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/err.h>
|
||||
#include <asm/cache.h>
|
||||
#include <asm/unistd.h>
|
||||
#include <asm/processor.h>
|
||||
#include <asm/page.h>
|
||||
#include <asm/mmu.h>
|
||||
#include <asm/thread_info.h>
|
||||
#include <asm/code-patching-asm.h>
|
||||
#include <asm/ppc_asm.h>
|
||||
#include <asm/asm-offsets.h>
|
||||
#include <asm/cputable.h>
|
||||
#include <asm/firmware.h>
|
||||
#include <asm/bug.h>
|
||||
#include <asm/ptrace.h>
|
||||
#include <asm/irqflags.h>
|
||||
#include <asm/hw_irq.h>
|
||||
#include <asm/context_tracking.h>
|
||||
#include <asm/ppc-opcode.h>
|
||||
#include <asm/barrier.h>
|
||||
#include <asm/export.h>
|
||||
#include <asm/asm-compat.h>
|
||||
#ifdef CONFIG_PPC_BOOK3S
|
||||
#include <asm/exception-64s.h>
|
||||
#else
|
||||
#include <asm/exception-64e.h>
|
||||
#endif
|
||||
#include <asm/feature-fixups.h>
|
||||
#include <asm/code-patching-asm.h>
|
||||
#include <asm/mmu.h>
|
||||
#include <asm/ppc_asm.h>
|
||||
#include <asm/kup.h>
|
||||
#include <asm/thread_info.h>
|
||||
|
||||
/*
|
||||
* System calls.
|
||||
*/
|
||||
.section ".text"
|
||||
.section ".text","ax",@progbits
|
||||
|
||||
#ifdef CONFIG_PPC_BOOK3S_64
|
||||
/*
|
||||
* Cancel all explict user streams as they will have no use after context
|
||||
* switch and will stop the HW from creating streams itself
|
||||
*/
|
||||
#define STOP_STREAMS \
|
||||
DCBT_BOOK3S_STOP_ALL_STREAM_IDS(r6)
|
||||
|
||||
#define FLUSH_COUNT_CACHE \
|
||||
1: nop; \
|
||||
@ -105,87 +70,13 @@ flush_branch_caches:
|
||||
.endr
|
||||
|
||||
blr
|
||||
#else
|
||||
#define FLUSH_COUNT_CACHE
|
||||
#endif /* CONFIG_PPC_BOOK3S_64 */
|
||||
|
||||
/*
|
||||
* This routine switches between two different tasks. The process
|
||||
* state of one is saved on its kernel stack. Then the state
|
||||
* of the other is restored from its kernel stack. The memory
|
||||
* management hardware is updated to the second process's state.
|
||||
* Finally, we can return to the second process, via interrupt_return.
|
||||
* On entry, r3 points to the THREAD for the current task, r4
|
||||
* points to the THREAD for the new task.
|
||||
*
|
||||
* Note: there are two ways to get to the "going out" portion
|
||||
* of this code; either by coming in via the entry (_switch)
|
||||
* or via "fork" which must set up an environment equivalent
|
||||
* to the "_switch" path. If you change this you'll have to change
|
||||
* the fork code also.
|
||||
*
|
||||
* The code which creates the new task context is in 'copy_thread'
|
||||
* in arch/powerpc/kernel/process.c
|
||||
*/
|
||||
.align 7
|
||||
_GLOBAL(_switch)
|
||||
mflr r0
|
||||
std r0,16(r1)
|
||||
stdu r1,-SWITCH_FRAME_SIZE(r1)
|
||||
/* r3-r13 are caller saved -- Cort */
|
||||
SAVE_NVGPRS(r1)
|
||||
std r0,_NIP(r1) /* Return to switch caller */
|
||||
mfcr r23
|
||||
std r23,_CCR(r1)
|
||||
std r1,KSP(r3) /* Set old stack pointer */
|
||||
|
||||
kuap_check_amr r9, r10
|
||||
|
||||
FLUSH_COUNT_CACHE /* Clobbers r9, ctr */
|
||||
|
||||
/*
|
||||
* On SMP kernels, care must be taken because a task may be
|
||||
* scheduled off CPUx and on to CPUy. Memory ordering must be
|
||||
* considered.
|
||||
*
|
||||
* Cacheable stores on CPUx will be visible when the task is
|
||||
* scheduled on CPUy by virtue of the core scheduler barriers
|
||||
* (see "Notes on Program-Order guarantees on SMP systems." in
|
||||
* kernel/sched/core.c).
|
||||
*
|
||||
* Uncacheable stores in the case of involuntary preemption must
|
||||
* be taken care of. The smp_mb__after_spinlock() in __schedule()
|
||||
* is implemented as hwsync on powerpc, which orders MMIO too. So
|
||||
* long as there is an hwsync in the context switch path, it will
|
||||
* be executed on the source CPU after the task has performed
|
||||
* all MMIO ops on that CPU, and on the destination CPU before the
|
||||
* task performs any MMIO ops there.
|
||||
*/
|
||||
|
||||
/*
|
||||
* The kernel context switch path must contain a spin_lock,
|
||||
* which contains larx/stcx, which will clear any reservation
|
||||
* of the task being switched.
|
||||
*/
|
||||
#ifdef CONFIG_PPC_BOOK3S
|
||||
/* Cancel all explict user streams as they will have no use after context
|
||||
* switch and will stop the HW from creating streams itself
|
||||
*/
|
||||
DCBT_BOOK3S_STOP_ALL_STREAM_IDS(r6)
|
||||
#endif
|
||||
|
||||
addi r6,r4,-THREAD /* Convert THREAD to 'current' */
|
||||
std r6,PACACURRENT(r13) /* Set new 'current' */
|
||||
#if defined(CONFIG_STACKPROTECTOR)
|
||||
ld r6, TASK_CANARY(r6)
|
||||
std r6, PACA_CANARY(r13)
|
||||
#endif
|
||||
|
||||
ld r8,KSP(r4) /* new stack pointer */
|
||||
#ifdef CONFIG_PPC_64S_HASH_MMU
|
||||
BEGIN_MMU_FTR_SECTION
|
||||
b 2f
|
||||
END_MMU_FTR_SECTION_IFSET(MMU_FTR_TYPE_RADIX)
|
||||
.balign 32
|
||||
/*
|
||||
* New stack pointer in r8, old stack pointer in r1, must not clobber r3
|
||||
*/
|
||||
pin_stack_slb:
|
||||
BEGIN_FTR_SECTION
|
||||
clrrdi r6,r8,28 /* get its ESID */
|
||||
clrrdi r9,r1,28 /* get current sp ESID */
|
||||
@ -233,15 +124,57 @@ BEGIN_FTR_SECTION
|
||||
END_FTR_SECTION_IFCLR(CPU_FTR_ARCH_207S)
|
||||
slbmte r7,r0
|
||||
isync
|
||||
2:
|
||||
2: blr
|
||||
.size pin_stack_slb,.-pin_stack_slb
|
||||
#endif /* CONFIG_PPC_64S_HASH_MMU */
|
||||
|
||||
clrrdi r7, r8, THREAD_SHIFT /* base of new stack */
|
||||
/* Note: this uses SWITCH_FRAME_SIZE rather than INT_FRAME_SIZE
|
||||
because we don't need to leave the 288-byte ABI gap at the
|
||||
top of the kernel stack. */
|
||||
addi r7,r7,THREAD_SIZE-SWITCH_FRAME_SIZE
|
||||
#else
|
||||
#define STOP_STREAMS
|
||||
#define FLUSH_COUNT_CACHE
|
||||
#endif /* CONFIG_PPC_BOOK3S_64 */
|
||||
|
||||
/*
|
||||
* do_switch_32/64 have the same calling convention as _switch, i.e., r3,r4
|
||||
* are prev and next thread_struct *, and returns prev task_struct * in r3.
|
||||
|
||||
* This switches the stack, current, and does other task switch housekeeping.
|
||||
*/
|
||||
.macro do_switch_32
|
||||
tophys(r0,r4)
|
||||
mtspr SPRN_SPRG_THREAD,r0 /* Update current THREAD phys addr */
|
||||
lwz r1,KSP(r4) /* Load new stack pointer */
|
||||
|
||||
/* save the old current 'last' for return value */
|
||||
mr r3,r2
|
||||
addi r2,r4,-THREAD /* Update current */
|
||||
.endm
|
||||
|
||||
.macro do_switch_64
|
||||
ld r8,KSP(r4) /* Load new stack pointer */
|
||||
|
||||
kuap_check_amr r9, r10
|
||||
|
||||
FLUSH_COUNT_CACHE /* Clobbers r9, ctr */
|
||||
|
||||
STOP_STREAMS /* Clobbers r6 */
|
||||
|
||||
addi r3,r3,-THREAD /* old thread -> task_struct for return value */
|
||||
addi r6,r4,-THREAD /* new thread -> task_struct */
|
||||
std r6,PACACURRENT(r13) /* Set new task_struct to 'current' */
|
||||
#if defined(CONFIG_STACKPROTECTOR)
|
||||
ld r6, TASK_CANARY(r6)
|
||||
std r6, PACA_CANARY(r13)
|
||||
#endif
|
||||
/* Set new PACAKSAVE */
|
||||
clrrdi r7,r8,THREAD_SHIFT /* base of new stack */
|
||||
addi r7,r7,THREAD_SIZE-SWITCH_FRAME_SIZE
|
||||
std r7,PACAKSAVE(r13)
|
||||
|
||||
#ifdef CONFIG_PPC_64S_HASH_MMU
|
||||
BEGIN_MMU_FTR_SECTION
|
||||
bl pin_stack_slb
|
||||
END_MMU_FTR_SECTION_IFCLR(MMU_FTR_TYPE_RADIX)
|
||||
#endif
|
||||
/*
|
||||
* PMU interrupts in radix may come in here. They will use r1, not
|
||||
* PACAKSAVE, so this stack switch will not cause a problem. They
|
||||
@ -251,81 +184,75 @@ END_FTR_SECTION_IFCLR(CPU_FTR_ARCH_207S)
|
||||
* active on the new CPU, will order those stores.
|
||||
*/
|
||||
mr r1,r8 /* start using new stack pointer */
|
||||
std r7,PACAKSAVE(r13)
|
||||
.endm
|
||||
|
||||
ld r6,_CCR(r1)
|
||||
mtcrf 0xFF,r6
|
||||
|
||||
/* r3-r13 are destroyed -- Cort */
|
||||
REST_NVGPRS(r1)
|
||||
|
||||
/* convert old thread to its task_struct for return value */
|
||||
addi r3,r3,-THREAD
|
||||
ld r7,_NIP(r1) /* Return to _switch caller in new task */
|
||||
mtlr r7
|
||||
addi r1,r1,SWITCH_FRAME_SIZE
|
||||
blr
|
||||
|
||||
_GLOBAL(enter_prom)
|
||||
mflr r0
|
||||
std r0,16(r1)
|
||||
stdu r1,-SWITCH_FRAME_SIZE(r1) /* Save SP and create stack space */
|
||||
|
||||
/* Because PROM is running in 32b mode, it clobbers the high order half
|
||||
* of all registers that it saves. We therefore save those registers
|
||||
* PROM might touch to the stack. (r0, r3-r13 are caller saved)
|
||||
/*
|
||||
* This routine switches between two different tasks. The process
|
||||
* state of one is saved on its kernel stack. Then the state
|
||||
* of the other is restored from its kernel stack. The memory
|
||||
* management hardware is updated to the second process's state.
|
||||
* Finally, we can return to the second process.
|
||||
* On entry, r3 points to the THREAD for the current task, r4
|
||||
* points to the THREAD for the new task.
|
||||
*
|
||||
* This routine is always called with interrupts disabled.
|
||||
*
|
||||
* Note: there are two ways to get to the "going out" portion
|
||||
* of this code; either by coming in via the entry (_switch)
|
||||
* or via "fork" which must set up an environment equivalent
|
||||
* to the "_switch" path. If you change this , you'll have to
|
||||
* change the fork code also.
|
||||
*
|
||||
* The code which creates the new task context is in 'copy_thread'
|
||||
* in arch/ppc/kernel/process.c
|
||||
*
|
||||
* Note: this uses SWITCH_FRAME_SIZE rather than USER_INT_FRAME_SIZE
|
||||
* because we don't need to leave the redzone ABI gap at the top of
|
||||
* the kernel stack.
|
||||
*/
|
||||
SAVE_GPR(2, r1)
|
||||
SAVE_GPR(13, r1)
|
||||
SAVE_NVGPRS(r1)
|
||||
mfcr r10
|
||||
mfmsr r11
|
||||
std r10,_CCR(r1)
|
||||
std r11,_MSR(r1)
|
||||
_GLOBAL(_switch)
|
||||
PPC_CREATE_STACK_FRAME(SWITCH_FRAME_SIZE)
|
||||
PPC_STL r1,KSP(r3) /* Set old stack pointer */
|
||||
SAVE_NVGPRS(r1) /* volatiles are caller-saved -- Cort */
|
||||
PPC_STL r0,_NIP(r1) /* Return to switch caller */
|
||||
mfcr r0
|
||||
stw r0,_CCR(r1)
|
||||
|
||||
/* Put PROM address in SRR0 */
|
||||
mtsrr0 r4
|
||||
|
||||
/* Setup our trampoline return addr in LR */
|
||||
bcl 20,31,$+4
|
||||
0: mflr r4
|
||||
addi r4,r4,(1f - 0b)
|
||||
mtlr r4
|
||||
|
||||
/* Prepare a 32-bit mode big endian MSR
|
||||
/*
|
||||
* On SMP kernels, care must be taken because a task may be
|
||||
* scheduled off CPUx and on to CPUy. Memory ordering must be
|
||||
* considered.
|
||||
*
|
||||
* Cacheable stores on CPUx will be visible when the task is
|
||||
* scheduled on CPUy by virtue of the core scheduler barriers
|
||||
* (see "Notes on Program-Order guarantees on SMP systems." in
|
||||
* kernel/sched/core.c).
|
||||
*
|
||||
* Uncacheable stores in the case of involuntary preemption must
|
||||
* be taken care of. The smp_mb__after_spinlock() in __schedule()
|
||||
* is implemented as hwsync on powerpc, which orders MMIO too. So
|
||||
* long as there is an hwsync in the context switch path, it will
|
||||
* be executed on the source CPU after the task has performed
|
||||
* all MMIO ops on that CPU, and on the destination CPU before the
|
||||
* task performs any MMIO ops there.
|
||||
*/
|
||||
#ifdef CONFIG_PPC_BOOK3E_64
|
||||
rlwinm r11,r11,0,1,31
|
||||
mtsrr1 r11
|
||||
rfi
|
||||
#else /* CONFIG_PPC_BOOK3E_64 */
|
||||
LOAD_REG_IMMEDIATE(r12, MSR_SF | MSR_LE)
|
||||
andc r11,r11,r12
|
||||
mtsrr1 r11
|
||||
RFI_TO_KERNEL
|
||||
#endif /* CONFIG_PPC_BOOK3E_64 */
|
||||
|
||||
1: /* Return from OF */
|
||||
FIXUP_ENDIAN
|
||||
|
||||
/* Just make sure that r1 top 32 bits didn't get
|
||||
* corrupt by OF
|
||||
/*
|
||||
* The kernel context switch path must contain a spin_lock,
|
||||
* which contains larx/stcx, which will clear any reservation
|
||||
* of the task being switched.
|
||||
*/
|
||||
rldicl r1,r1,0,32
|
||||
|
||||
/* Restore the MSR (back to 64 bits) */
|
||||
ld r0,_MSR(r1)
|
||||
MTMSRD(r0)
|
||||
isync
|
||||
#ifdef CONFIG_PPC32
|
||||
do_switch_32
|
||||
#else
|
||||
do_switch_64
|
||||
#endif
|
||||
|
||||
/* Restore other registers */
|
||||
REST_GPR(2, r1)
|
||||
REST_GPR(13, r1)
|
||||
REST_NVGPRS(r1)
|
||||
ld r4,_CCR(r1)
|
||||
mtcr r4
|
||||
|
||||
addi r1,r1,SWITCH_FRAME_SIZE
|
||||
ld r0,16(r1)
|
||||
lwz r0,_CCR(r1)
|
||||
mtcrf 0xFF,r0
|
||||
REST_NVGPRS(r1) /* volatiles are destroyed -- Cort */
|
||||
PPC_LL r0,_NIP(r1) /* Return to _switch caller in new task */
|
||||
mtlr r0
|
||||
addi r1,r1,SWITCH_FRAME_SIZE
|
||||
blr
|
@ -1516,6 +1516,22 @@ static void do_program_check(struct pt_regs *regs)
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (cpu_has_feature(CPU_FTR_DEXCR_NPHIE) && user_mode(regs)) {
|
||||
ppc_inst_t insn;
|
||||
|
||||
if (get_user_instr(insn, (void __user *)regs->nip)) {
|
||||
_exception(SIGSEGV, regs, SEGV_MAPERR, regs->nip);
|
||||
return;
|
||||
}
|
||||
|
||||
if (ppc_inst_primary_opcode(insn) == 31 &&
|
||||
get_xop(ppc_inst_val(insn)) == OP_31_XOP_HASHCHK) {
|
||||
_exception(SIGILL, regs, ILL_ILLOPN, regs->nip);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
_exception(SIGTRAP, regs, TRAP_BRKPT, regs->nip);
|
||||
return;
|
||||
}
|
||||
|
@ -56,6 +56,8 @@ KCSAN_SANITIZE := n
|
||||
ccflags-y := -fno-common -fno-builtin
|
||||
ldflags-y := -Wl,--hash-style=both -nostdlib -shared -z noexecstack
|
||||
ldflags-$(CONFIG_LD_IS_LLD) += $(call cc-option,--ld-path=$(LD),-fuse-ld=lld)
|
||||
ldflags-$(CONFIG_LD_ORPHAN_WARN) += -Wl,--orphan-handling=$(CONFIG_LD_ORPHAN_WARN_LEVEL)
|
||||
|
||||
# Filter flags that clang will warn are unused for linking
|
||||
ldflags-y += $(filter-out $(CC_AUTO_VAR_INIT_ZERO_ENABLER) $(CC_FLAGS_FTRACE) -Wa$(comma)%, $(KBUILD_CFLAGS))
|
||||
|
||||
|
@ -83,9 +83,11 @@ SECTIONS
|
||||
|
||||
/DISCARD/ : {
|
||||
*(.note.GNU-stack)
|
||||
*(*.EMB.apuinfo)
|
||||
*(.branch_lt)
|
||||
*(.data .data.* .gnu.linkonce.d.* .sdata*)
|
||||
*(.bss .sbss .dynbss .dynsbss)
|
||||
*(.got1)
|
||||
*(.got1 .glink .iplt .rela*)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -32,7 +32,7 @@ SECTIONS
|
||||
. = ALIGN(16);
|
||||
.text : {
|
||||
*(.text .stub .text.* .gnu.linkonce.t.* __ftr_alt_*)
|
||||
*(.sfpr .glink)
|
||||
*(.sfpr)
|
||||
} :text
|
||||
PROVIDE(__etext = .);
|
||||
PROVIDE(_etext = .);
|
||||
@ -81,10 +81,12 @@ SECTIONS
|
||||
|
||||
/DISCARD/ : {
|
||||
*(.note.GNU-stack)
|
||||
*(*.EMB.apuinfo)
|
||||
*(.branch_lt)
|
||||
*(.data .data.* .gnu.linkonce.d.* .sdata*)
|
||||
*(.bss .sbss .dynbss .dynsbss)
|
||||
*(.opd)
|
||||
*(.glink .iplt .plt .rela*)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -406,7 +406,7 @@ static long kvmppc_read_one_intr(bool *again)
|
||||
return 1;
|
||||
|
||||
/* see if a host IPI is pending */
|
||||
host_ipi = local_paca->kvm_hstate.host_ipi;
|
||||
host_ipi = READ_ONCE(local_paca->kvm_hstate.host_ipi);
|
||||
if (host_ipi)
|
||||
return 1;
|
||||
|
||||
@ -466,7 +466,7 @@ static long kvmppc_read_one_intr(bool *again)
|
||||
* meantime. If it's clear, we bounce the interrupt to the
|
||||
* guest
|
||||
*/
|
||||
host_ipi = local_paca->kvm_hstate.host_ipi;
|
||||
host_ipi = READ_ONCE(local_paca->kvm_hstate.host_ipi);
|
||||
if (unlikely(host_ipi != 0)) {
|
||||
/* We raced with the host,
|
||||
* we need to resend that IPI, bummer
|
||||
|
@ -44,7 +44,7 @@ obj-$(CONFIG_FUNCTION_ERROR_INJECTION) += error-inject.o
|
||||
# 64-bit linker creates .sfpr on demand for final link (vmlinux),
|
||||
# so it is only needed for modules, and only for older linkers which
|
||||
# do not support --save-restore-funcs
|
||||
ifeq ($(call ld-ifversion, -lt, 22500, y),y)
|
||||
ifndef CONFIG_LD_IS_BFD
|
||||
extra-$(CONFIG_PPC64) += crtsavres.o
|
||||
endif
|
||||
|
||||
|
@ -161,6 +161,8 @@ static __always_inline u32 publish_tail_cpu(struct qspinlock *lock, u32 tail)
|
||||
{
|
||||
u32 prev, tmp;
|
||||
|
||||
kcsan_release();
|
||||
|
||||
asm volatile(
|
||||
"\t" PPC_RELEASE_BARRIER " \n"
|
||||
"1: lwarx %0,0,%2 # publish_tail_cpu \n"
|
||||
@ -435,7 +437,7 @@ yield_prev:
|
||||
|
||||
smp_rmb(); /* See __yield_to_locked_owner comment */
|
||||
|
||||
if (!node->locked) {
|
||||
if (!READ_ONCE(node->locked)) {
|
||||
yield_to_preempted(prev_cpu, yield_count);
|
||||
spin_begin();
|
||||
return preempted;
|
||||
@ -570,6 +572,11 @@ static __always_inline void queued_spin_lock_mcs_queue(struct qspinlock *lock, b
|
||||
|
||||
tail = encode_tail_cpu(node->cpu);
|
||||
|
||||
/*
|
||||
* Assign all attributes of a node before it can be published.
|
||||
* Issues an lwsync, serving as a release barrier, as well as a
|
||||
* compiler barrier.
|
||||
*/
|
||||
old = publish_tail_cpu(lock, tail);
|
||||
|
||||
/*
|
||||
@ -584,7 +591,7 @@ static __always_inline void queued_spin_lock_mcs_queue(struct qspinlock *lock, b
|
||||
|
||||
/* Wait for mcs node lock to be released */
|
||||
spin_begin();
|
||||
while (!node->locked) {
|
||||
while (!READ_ONCE(node->locked)) {
|
||||
spec_barrier();
|
||||
|
||||
if (yield_to_prev(lock, node, old, paravirt))
|
||||
|
@ -199,12 +199,12 @@ _GLOBAL(add_hash_page)
|
||||
lis r6, (mmu_hash_lock - PAGE_OFFSET)@ha
|
||||
addi r6, r6, (mmu_hash_lock - PAGE_OFFSET)@l
|
||||
10: lwarx r0,0,r6 /* take the mmu_hash_lock */
|
||||
cmpi 0,r0,0
|
||||
cmpwi 0,r0,0
|
||||
bne- 11f
|
||||
stwcx. r8,0,r6
|
||||
beq+ 12f
|
||||
11: lwz r0,0(r6)
|
||||
cmpi 0,r0,0
|
||||
cmpwi 0,r0,0
|
||||
beq 10b
|
||||
b 11b
|
||||
12: isync
|
||||
@ -512,12 +512,12 @@ _GLOBAL(flush_hash_pages)
|
||||
lwz r8, TASK_CPU(r8)
|
||||
oris r8,r8,9
|
||||
10: lwarx r0,0,r9
|
||||
cmpi 0,r0,0
|
||||
cmpwi 0,r0,0
|
||||
bne- 11f
|
||||
stwcx. r8,0,r9
|
||||
beq+ 12f
|
||||
11: lwz r0,0(r9)
|
||||
cmpi 0,r0,0
|
||||
cmpwi 0,r0,0
|
||||
beq 10b
|
||||
b 11b
|
||||
12: isync
|
||||
|
@ -745,9 +745,9 @@ static void free_pud_table(pud_t *pud_start, p4d_t *p4d)
|
||||
}
|
||||
|
||||
static void remove_pte_table(pte_t *pte_start, unsigned long addr,
|
||||
unsigned long end)
|
||||
unsigned long end, bool direct)
|
||||
{
|
||||
unsigned long next;
|
||||
unsigned long next, pages = 0;
|
||||
pte_t *pte;
|
||||
|
||||
pte = pte_start + pte_index(addr);
|
||||
@ -769,13 +769,16 @@ static void remove_pte_table(pte_t *pte_start, unsigned long addr,
|
||||
}
|
||||
|
||||
pte_clear(&init_mm, addr, pte);
|
||||
pages++;
|
||||
}
|
||||
if (direct)
|
||||
update_page_count(mmu_virtual_psize, -pages);
|
||||
}
|
||||
|
||||
static void __meminit remove_pmd_table(pmd_t *pmd_start, unsigned long addr,
|
||||
unsigned long end)
|
||||
unsigned long end, bool direct)
|
||||
{
|
||||
unsigned long next;
|
||||
unsigned long next, pages = 0;
|
||||
pte_t *pte_base;
|
||||
pmd_t *pmd;
|
||||
|
||||
@ -793,19 +796,22 @@ static void __meminit remove_pmd_table(pmd_t *pmd_start, unsigned long addr,
|
||||
continue;
|
||||
}
|
||||
pte_clear(&init_mm, addr, (pte_t *)pmd);
|
||||
pages++;
|
||||
continue;
|
||||
}
|
||||
|
||||
pte_base = (pte_t *)pmd_page_vaddr(*pmd);
|
||||
remove_pte_table(pte_base, addr, next);
|
||||
remove_pte_table(pte_base, addr, next, direct);
|
||||
free_pte_table(pte_base, pmd);
|
||||
}
|
||||
if (direct)
|
||||
update_page_count(MMU_PAGE_2M, -pages);
|
||||
}
|
||||
|
||||
static void __meminit remove_pud_table(pud_t *pud_start, unsigned long addr,
|
||||
unsigned long end)
|
||||
unsigned long end, bool direct)
|
||||
{
|
||||
unsigned long next;
|
||||
unsigned long next, pages = 0;
|
||||
pmd_t *pmd_base;
|
||||
pud_t *pud;
|
||||
|
||||
@ -823,16 +829,20 @@ static void __meminit remove_pud_table(pud_t *pud_start, unsigned long addr,
|
||||
continue;
|
||||
}
|
||||
pte_clear(&init_mm, addr, (pte_t *)pud);
|
||||
pages++;
|
||||
continue;
|
||||
}
|
||||
|
||||
pmd_base = pud_pgtable(*pud);
|
||||
remove_pmd_table(pmd_base, addr, next);
|
||||
remove_pmd_table(pmd_base, addr, next, direct);
|
||||
free_pmd_table(pmd_base, pud);
|
||||
}
|
||||
if (direct)
|
||||
update_page_count(MMU_PAGE_1G, -pages);
|
||||
}
|
||||
|
||||
static void __meminit remove_pagetable(unsigned long start, unsigned long end)
|
||||
static void __meminit remove_pagetable(unsigned long start, unsigned long end,
|
||||
bool direct)
|
||||
{
|
||||
unsigned long addr, next;
|
||||
pud_t *pud_base;
|
||||
@ -861,7 +871,7 @@ static void __meminit remove_pagetable(unsigned long start, unsigned long end)
|
||||
}
|
||||
|
||||
pud_base = p4d_pgtable(*p4d);
|
||||
remove_pud_table(pud_base, addr, next);
|
||||
remove_pud_table(pud_base, addr, next, direct);
|
||||
free_pud_table(pud_base, p4d);
|
||||
}
|
||||
|
||||
@ -884,7 +894,7 @@ int __meminit radix__create_section_mapping(unsigned long start,
|
||||
|
||||
int __meminit radix__remove_section_mapping(unsigned long start, unsigned long end)
|
||||
{
|
||||
remove_pagetable(start, end);
|
||||
remove_pagetable(start, end, true);
|
||||
return 0;
|
||||
}
|
||||
#endif /* CONFIG_MEMORY_HOTPLUG */
|
||||
@ -902,7 +912,6 @@ int __meminit radix__vmemmap_create_mapping(unsigned long start,
|
||||
unsigned long phys)
|
||||
{
|
||||
/* Create a PTE encoding */
|
||||
unsigned long flags = _PAGE_PRESENT | _PAGE_ACCESSED | _PAGE_KERNEL_RW;
|
||||
int nid = early_pfn_to_nid(phys >> PAGE_SHIFT);
|
||||
int ret;
|
||||
|
||||
@ -911,7 +920,7 @@ int __meminit radix__vmemmap_create_mapping(unsigned long start,
|
||||
return -1;
|
||||
}
|
||||
|
||||
ret = __map_kernel_page_nid(start, phys, __pgprot(flags), page_size, nid);
|
||||
ret = __map_kernel_page_nid(start, phys, PAGE_KERNEL, page_size, nid);
|
||||
BUG_ON(ret);
|
||||
|
||||
return 0;
|
||||
@ -920,7 +929,7 @@ int __meminit radix__vmemmap_create_mapping(unsigned long start,
|
||||
#ifdef CONFIG_MEMORY_HOTPLUG
|
||||
void __meminit radix__vmemmap_remove_mapping(unsigned long start, unsigned long page_size)
|
||||
{
|
||||
remove_pagetable(start, start + page_size);
|
||||
remove_pagetable(start, start + page_size, false);
|
||||
}
|
||||
#endif
|
||||
#endif
|
||||
@ -952,7 +961,7 @@ unsigned long radix__pmd_hugepage_update(struct mm_struct *mm, unsigned long add
|
||||
assert_spin_locked(pmd_lockptr(mm, pmdp));
|
||||
#endif
|
||||
|
||||
old = radix__pte_update(mm, addr, (pte_t *)pmdp, clr, set, 1);
|
||||
old = radix__pte_update(mm, addr, pmdp_ptep(pmdp), clr, set, 1);
|
||||
trace_hugepage_update(addr, old, clr, set);
|
||||
|
||||
return old;
|
||||
|
@ -189,7 +189,7 @@ static bool altmap_cross_boundary(struct vmem_altmap *altmap, unsigned long star
|
||||
unsigned long nr_pfn = page_size / sizeof(struct page);
|
||||
unsigned long start_pfn = page_to_pfn((struct page *)start);
|
||||
|
||||
if ((start_pfn + nr_pfn) > altmap->end_pfn)
|
||||
if ((start_pfn + nr_pfn - 1) > altmap->end_pfn)
|
||||
return true;
|
||||
|
||||
if (start_pfn < altmap->base_pfn)
|
||||
|
@ -103,7 +103,7 @@ static struct i2c_driver avr_driver = {
|
||||
.driver = {
|
||||
.name = "akebono-avr",
|
||||
},
|
||||
.probe_new = avr_probe,
|
||||
.probe = avr_probe,
|
||||
.id_table = avr_id,
|
||||
};
|
||||
|
||||
|
@ -373,50 +373,32 @@ static int get_cs_ranges(struct device *dev)
|
||||
{
|
||||
int ret = -ENODEV;
|
||||
struct device_node *lb_node;
|
||||
const u32 *addr_cells_p;
|
||||
const u32 *size_cells_p;
|
||||
int proplen;
|
||||
size_t i;
|
||||
size_t i = 0;
|
||||
struct of_range_parser parser;
|
||||
struct of_range range;
|
||||
|
||||
lb_node = of_find_compatible_node(NULL, NULL, "fsl,mpc5121-localbus");
|
||||
if (!lb_node)
|
||||
return ret;
|
||||
|
||||
/*
|
||||
* The node defined as compatible with 'fsl,mpc5121-localbus'
|
||||
* should have two address cells and one size cell.
|
||||
* Every item of its ranges property should consist of:
|
||||
* - the first address cell which is the chipselect number;
|
||||
* - the second address cell which is the offset in the chipselect,
|
||||
* must be zero.
|
||||
* - CPU address of the beginning of an access window;
|
||||
* - the only size cell which is the size of an access window.
|
||||
*/
|
||||
addr_cells_p = of_get_property(lb_node, "#address-cells", NULL);
|
||||
size_cells_p = of_get_property(lb_node, "#size-cells", NULL);
|
||||
if (addr_cells_p == NULL || *addr_cells_p != 2 ||
|
||||
size_cells_p == NULL || *size_cells_p != 1) {
|
||||
goto end;
|
||||
}
|
||||
of_range_parser_init(&parser, lb_node);
|
||||
lpbfifo.cs_n = of_range_count(&parser);
|
||||
|
||||
proplen = of_property_count_u32_elems(lb_node, "ranges");
|
||||
if (proplen <= 0 || proplen % 4 != 0)
|
||||
goto end;
|
||||
|
||||
lpbfifo.cs_n = proplen / 4;
|
||||
lpbfifo.cs_ranges = devm_kcalloc(dev, lpbfifo.cs_n,
|
||||
sizeof(struct cs_range), GFP_KERNEL);
|
||||
if (!lpbfifo.cs_ranges)
|
||||
goto end;
|
||||
|
||||
if (of_property_read_u32_array(lb_node, "ranges",
|
||||
(u32 *)lpbfifo.cs_ranges, proplen) != 0) {
|
||||
for_each_of_range(&parser, &range) {
|
||||
u32 base = lower_32_bits(range.bus_addr);
|
||||
if (base)
|
||||
goto end;
|
||||
}
|
||||
|
||||
for (i = 0; i < lpbfifo.cs_n; i++) {
|
||||
if (lpbfifo.cs_ranges[i].base != 0)
|
||||
goto end;
|
||||
lpbfifo.cs_ranges[i].csnum = upper_32_bits(range.bus_addr);
|
||||
lpbfifo.cs_ranges[i].base = base;
|
||||
lpbfifo.cs_ranges[i].addr = range.cpu_addr;
|
||||
lpbfifo.cs_ranges[i].size = range.size;
|
||||
i++;
|
||||
}
|
||||
|
||||
ret = 0;
|
||||
|
@ -54,8 +54,3 @@ config PPC_MPC5200_BUGFIX
|
||||
for MPC5200B based boards.
|
||||
|
||||
It is safe to say 'Y' here
|
||||
|
||||
config PPC_MPC5200_LPBFIFO
|
||||
tristate "MPC5200 LocalPlus bus FIFO driver"
|
||||
depends on PPC_MPC52xx && PPC_BESTCOMM
|
||||
select PPC_BESTCOMM_GEN_BD
|
||||
|
@ -14,5 +14,3 @@ obj-$(CONFIG_PM) += mpc52xx_sleep.o mpc52xx_pm.o
|
||||
ifdef CONFIG_PPC_LITE5200
|
||||
obj-$(CONFIG_PM) += lite5200_sleep.o lite5200_pm.o
|
||||
endif
|
||||
|
||||
obj-$(CONFIG_PPC_MPC5200_LPBFIFO) += mpc52xx_lpbfifo.o
|
||||
|
@ -47,7 +47,7 @@ static int lite5200_pm_begin(suspend_state_t state)
|
||||
static int lite5200_pm_prepare(void)
|
||||
{
|
||||
struct device_node *np;
|
||||
const struct of_device_id immr_ids[] = {
|
||||
static const struct of_device_id immr_ids[] = {
|
||||
{ .compatible = "fsl,mpc5200-immr", },
|
||||
{ .compatible = "fsl,mpc5200b-immr", },
|
||||
{ .type = "soc", .compatible = "mpc5200", }, /* lite5200 */
|
||||
|
@ -1,594 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-only
|
||||
/*
|
||||
* LocalPlus Bus FIFO driver for the Freescale MPC52xx.
|
||||
*
|
||||
* Copyright (C) 2009 Secret Lab Technologies Ltd.
|
||||
*
|
||||
* Todo:
|
||||
* - Add support for multiple requests to be queued.
|
||||
*/
|
||||
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/spinlock.h>
|
||||
#include <linux/module.h>
|
||||
#include <asm/io.h>
|
||||
#include <asm/mpc52xx.h>
|
||||
#include <asm/time.h>
|
||||
|
||||
#include <linux/fsl/bestcomm/bestcomm.h>
|
||||
#include <linux/fsl/bestcomm/bestcomm_priv.h>
|
||||
#include <linux/fsl/bestcomm/gen_bd.h>
|
||||
|
||||
MODULE_AUTHOR("Grant Likely <grant.likely@secretlab.ca>");
|
||||
MODULE_DESCRIPTION("MPC5200 LocalPlus FIFO device driver");
|
||||
MODULE_LICENSE("GPL");
|
||||
|
||||
#define LPBFIFO_REG_PACKET_SIZE (0x00)
|
||||
#define LPBFIFO_REG_START_ADDRESS (0x04)
|
||||
#define LPBFIFO_REG_CONTROL (0x08)
|
||||
#define LPBFIFO_REG_ENABLE (0x0C)
|
||||
#define LPBFIFO_REG_BYTES_DONE_STATUS (0x14)
|
||||
#define LPBFIFO_REG_FIFO_DATA (0x40)
|
||||
#define LPBFIFO_REG_FIFO_STATUS (0x44)
|
||||
#define LPBFIFO_REG_FIFO_CONTROL (0x48)
|
||||
#define LPBFIFO_REG_FIFO_ALARM (0x4C)
|
||||
|
||||
struct mpc52xx_lpbfifo {
|
||||
struct device *dev;
|
||||
phys_addr_t regs_phys;
|
||||
void __iomem *regs;
|
||||
int irq;
|
||||
spinlock_t lock;
|
||||
|
||||
struct bcom_task *bcom_tx_task;
|
||||
struct bcom_task *bcom_rx_task;
|
||||
struct bcom_task *bcom_cur_task;
|
||||
|
||||
/* Current state data */
|
||||
struct mpc52xx_lpbfifo_request *req;
|
||||
int dma_irqs_enabled;
|
||||
};
|
||||
|
||||
/* The MPC5200 has only one fifo, so only need one instance structure */
|
||||
static struct mpc52xx_lpbfifo lpbfifo;
|
||||
|
||||
/**
|
||||
* mpc52xx_lpbfifo_kick - Trigger the next block of data to be transferred
|
||||
*
|
||||
* @req: Pointer to request structure
|
||||
*/
|
||||
static void mpc52xx_lpbfifo_kick(struct mpc52xx_lpbfifo_request *req)
|
||||
{
|
||||
size_t transfer_size = req->size - req->pos;
|
||||
struct bcom_bd *bd;
|
||||
void __iomem *reg;
|
||||
u32 *data;
|
||||
int i;
|
||||
int bit_fields;
|
||||
int dma = !(req->flags & MPC52XX_LPBFIFO_FLAG_NO_DMA);
|
||||
int write = req->flags & MPC52XX_LPBFIFO_FLAG_WRITE;
|
||||
int poll_dma = req->flags & MPC52XX_LPBFIFO_FLAG_POLL_DMA;
|
||||
|
||||
/* Set and clear the reset bits; is good practice in User Manual */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x01010000);
|
||||
|
||||
/* set master enable bit */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x00000001);
|
||||
if (!dma) {
|
||||
/* While the FIFO can be setup for transfer sizes as large as
|
||||
* 16M-1, the FIFO itself is only 512 bytes deep and it does
|
||||
* not generate interrupts for FIFO full events (only transfer
|
||||
* complete will raise an IRQ). Therefore when not using
|
||||
* Bestcomm to drive the FIFO it needs to either be polled, or
|
||||
* transfers need to constrained to the size of the fifo.
|
||||
*
|
||||
* This driver restricts the size of the transfer
|
||||
*/
|
||||
if (transfer_size > 512)
|
||||
transfer_size = 512;
|
||||
|
||||
/* Load the FIFO with data */
|
||||
if (write) {
|
||||
reg = lpbfifo.regs + LPBFIFO_REG_FIFO_DATA;
|
||||
data = req->data + req->pos;
|
||||
for (i = 0; i < transfer_size; i += 4)
|
||||
out_be32(reg, *data++);
|
||||
}
|
||||
|
||||
/* Unmask both error and completion irqs */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x00000301);
|
||||
} else {
|
||||
/* Choose the correct direction
|
||||
*
|
||||
* Configure the watermarks so DMA will always complete correctly.
|
||||
* It may be worth experimenting with the ALARM value to see if
|
||||
* there is a performance impact. However, if it is wrong there
|
||||
* is a risk of DMA not transferring the last chunk of data
|
||||
*/
|
||||
if (write) {
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_FIFO_ALARM, 0x1e4);
|
||||
out_8(lpbfifo.regs + LPBFIFO_REG_FIFO_CONTROL, 7);
|
||||
lpbfifo.bcom_cur_task = lpbfifo.bcom_tx_task;
|
||||
} else {
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_FIFO_ALARM, 0x1ff);
|
||||
out_8(lpbfifo.regs + LPBFIFO_REG_FIFO_CONTROL, 0);
|
||||
lpbfifo.bcom_cur_task = lpbfifo.bcom_rx_task;
|
||||
|
||||
if (poll_dma) {
|
||||
if (lpbfifo.dma_irqs_enabled) {
|
||||
disable_irq(bcom_get_task_irq(lpbfifo.bcom_rx_task));
|
||||
lpbfifo.dma_irqs_enabled = 0;
|
||||
}
|
||||
} else {
|
||||
if (!lpbfifo.dma_irqs_enabled) {
|
||||
enable_irq(bcom_get_task_irq(lpbfifo.bcom_rx_task));
|
||||
lpbfifo.dma_irqs_enabled = 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bd = bcom_prepare_next_buffer(lpbfifo.bcom_cur_task);
|
||||
bd->status = transfer_size;
|
||||
if (!write) {
|
||||
/*
|
||||
* In the DMA read case, the DMA doesn't complete,
|
||||
* possibly due to incorrect watermarks in the ALARM
|
||||
* and CONTROL regs. For now instead of trying to
|
||||
* determine the right watermarks that will make this
|
||||
* work, just increase the number of bytes the FIFO is
|
||||
* expecting.
|
||||
*
|
||||
* When submitting another operation, the FIFO will get
|
||||
* reset, so the condition of the FIFO waiting for a
|
||||
* non-existent 4 bytes will get cleared.
|
||||
*/
|
||||
transfer_size += 4; /* BLECH! */
|
||||
}
|
||||
bd->data[0] = req->data_phys + req->pos;
|
||||
bcom_submit_next_buffer(lpbfifo.bcom_cur_task, NULL);
|
||||
|
||||
/* error irq & master enabled bit */
|
||||
bit_fields = 0x00000201;
|
||||
|
||||
/* Unmask irqs */
|
||||
if (write && (!poll_dma))
|
||||
bit_fields |= 0x00000100; /* completion irq too */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, bit_fields);
|
||||
}
|
||||
|
||||
/* Set transfer size, width, chip select and READ mode */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_START_ADDRESS,
|
||||
req->offset + req->pos);
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_PACKET_SIZE, transfer_size);
|
||||
|
||||
bit_fields = req->cs << 24 | 0x000008;
|
||||
if (!write)
|
||||
bit_fields |= 0x010000; /* read mode */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_CONTROL, bit_fields);
|
||||
|
||||
/* Kick it off */
|
||||
if (!lpbfifo.req->defer_xfer_start)
|
||||
out_8(lpbfifo.regs + LPBFIFO_REG_PACKET_SIZE, 0x01);
|
||||
if (dma)
|
||||
bcom_enable(lpbfifo.bcom_cur_task);
|
||||
}
|
||||
|
||||
/**
|
||||
* mpc52xx_lpbfifo_irq - IRQ handler for LPB FIFO
|
||||
* @irq: IRQ number to be handled
|
||||
* @dev_id: device ID cookie
|
||||
*
|
||||
* On transmit, the dma completion irq triggers before the fifo completion
|
||||
* triggers. Handle the dma completion here instead of the LPB FIFO Bestcomm
|
||||
* task completion irq because everything is not really done until the LPB FIFO
|
||||
* completion irq triggers.
|
||||
*
|
||||
* In other words:
|
||||
* For DMA, on receive, the "Fat Lady" is the bestcom completion irq. on
|
||||
* transmit, the fifo completion irq is the "Fat Lady". The opera (or in this
|
||||
* case the DMA/FIFO operation) is not finished until the "Fat Lady" sings.
|
||||
*
|
||||
* Reasons for entering this routine:
|
||||
* 1) PIO mode rx and tx completion irq
|
||||
* 2) DMA interrupt mode tx completion irq
|
||||
* 3) DMA polled mode tx
|
||||
*
|
||||
* Exit conditions:
|
||||
* 1) Transfer aborted
|
||||
* 2) FIFO complete without DMA; more data to do
|
||||
* 3) FIFO complete without DMA; all data transferred
|
||||
* 4) FIFO complete using DMA
|
||||
*
|
||||
* Condition 1 can occur regardless of whether or not DMA is used.
|
||||
* It requires executing the callback to report the error and exiting
|
||||
* immediately.
|
||||
*
|
||||
* Condition 2 requires programming the FIFO with the next block of data
|
||||
*
|
||||
* Condition 3 requires executing the callback to report completion
|
||||
*
|
||||
* Condition 4 means the same as 3, except that we also retrieve the bcom
|
||||
* buffer so DMA doesn't get clogged up.
|
||||
*
|
||||
* To make things trickier, the spinlock must be dropped before
|
||||
* executing the callback, otherwise we could end up with a deadlock
|
||||
* or nested spinlock condition. The out path is non-trivial, so
|
||||
* extra fiddling is done to make sure all paths lead to the same
|
||||
* outbound code.
|
||||
*
|
||||
* Return: irqreturn code (%IRQ_HANDLED)
|
||||
*/
|
||||
static irqreturn_t mpc52xx_lpbfifo_irq(int irq, void *dev_id)
|
||||
{
|
||||
struct mpc52xx_lpbfifo_request *req;
|
||||
u32 status = in_8(lpbfifo.regs + LPBFIFO_REG_BYTES_DONE_STATUS);
|
||||
void __iomem *reg;
|
||||
u32 *data;
|
||||
int count, i;
|
||||
int do_callback = 0;
|
||||
u32 ts;
|
||||
unsigned long flags;
|
||||
int dma, write, poll_dma;
|
||||
|
||||
spin_lock_irqsave(&lpbfifo.lock, flags);
|
||||
ts = mftb();
|
||||
|
||||
req = lpbfifo.req;
|
||||
if (!req) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
pr_err("bogus LPBFIFO IRQ\n");
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
dma = !(req->flags & MPC52XX_LPBFIFO_FLAG_NO_DMA);
|
||||
write = req->flags & MPC52XX_LPBFIFO_FLAG_WRITE;
|
||||
poll_dma = req->flags & MPC52XX_LPBFIFO_FLAG_POLL_DMA;
|
||||
|
||||
if (dma && !write) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
pr_err("bogus LPBFIFO IRQ (dma and not writing)\n");
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
if ((status & 0x01) == 0) {
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* check abort bit */
|
||||
if (status & 0x10) {
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x01010000);
|
||||
do_callback = 1;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* Read result from hardware */
|
||||
count = in_be32(lpbfifo.regs + LPBFIFO_REG_BYTES_DONE_STATUS);
|
||||
count &= 0x00ffffff;
|
||||
|
||||
if (!dma && !write) {
|
||||
/* copy the data out of the FIFO */
|
||||
reg = lpbfifo.regs + LPBFIFO_REG_FIFO_DATA;
|
||||
data = req->data + req->pos;
|
||||
for (i = 0; i < count; i += 4)
|
||||
*data++ = in_be32(reg);
|
||||
}
|
||||
|
||||
/* Update transfer position and count */
|
||||
req->pos += count;
|
||||
|
||||
/* Decide what to do next */
|
||||
if (req->size - req->pos)
|
||||
mpc52xx_lpbfifo_kick(req); /* more work to do */
|
||||
else
|
||||
do_callback = 1;
|
||||
|
||||
out:
|
||||
/* Clear the IRQ */
|
||||
out_8(lpbfifo.regs + LPBFIFO_REG_BYTES_DONE_STATUS, 0x01);
|
||||
|
||||
if (dma && (status & 0x11)) {
|
||||
/*
|
||||
* Count the DMA as complete only when the FIFO completion
|
||||
* status or abort bits are set.
|
||||
*
|
||||
* (status & 0x01) should always be the case except sometimes
|
||||
* when using polled DMA.
|
||||
*
|
||||
* (status & 0x10) {transfer aborted}: This case needs more
|
||||
* testing.
|
||||
*/
|
||||
bcom_retrieve_buffer(lpbfifo.bcom_cur_task, &status, NULL);
|
||||
}
|
||||
req->last_byte = ((u8 *)req->data)[req->size - 1];
|
||||
|
||||
/* When the do_callback flag is set; it means the transfer is finished
|
||||
* so set the FIFO as idle */
|
||||
if (do_callback)
|
||||
lpbfifo.req = NULL;
|
||||
|
||||
if (irq != 0) /* don't increment on polled case */
|
||||
req->irq_count++;
|
||||
|
||||
req->irq_ticks += mftb() - ts;
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
|
||||
/* Spinlock is released; it is now safe to call the callback */
|
||||
if (do_callback && req->callback)
|
||||
req->callback(req);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
/**
|
||||
* mpc52xx_lpbfifo_bcom_irq - IRQ handler for LPB FIFO Bestcomm task
|
||||
* @irq: IRQ number to be handled
|
||||
* @dev_id: device ID cookie
|
||||
*
|
||||
* Only used when receiving data.
|
||||
*
|
||||
* Return: irqreturn code (%IRQ_HANDLED)
|
||||
*/
|
||||
static irqreturn_t mpc52xx_lpbfifo_bcom_irq(int irq, void *dev_id)
|
||||
{
|
||||
struct mpc52xx_lpbfifo_request *req;
|
||||
unsigned long flags;
|
||||
u32 status;
|
||||
u32 ts;
|
||||
|
||||
spin_lock_irqsave(&lpbfifo.lock, flags);
|
||||
ts = mftb();
|
||||
|
||||
req = lpbfifo.req;
|
||||
if (!req || (req->flags & MPC52XX_LPBFIFO_FLAG_NO_DMA)) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
if (irq != 0) /* don't increment on polled case */
|
||||
req->irq_count++;
|
||||
|
||||
if (!bcom_buffer_done(lpbfifo.bcom_cur_task)) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
|
||||
req->buffer_not_done_cnt++;
|
||||
if ((req->buffer_not_done_cnt % 1000) == 0)
|
||||
pr_err("transfer stalled\n");
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
bcom_retrieve_buffer(lpbfifo.bcom_cur_task, &status, NULL);
|
||||
|
||||
req->last_byte = ((u8 *)req->data)[req->size - 1];
|
||||
|
||||
req->pos = status & 0x00ffffff;
|
||||
|
||||
/* Mark the FIFO as idle */
|
||||
lpbfifo.req = NULL;
|
||||
|
||||
/* Release the lock before calling out to the callback. */
|
||||
req->irq_ticks += mftb() - ts;
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
|
||||
if (req->callback)
|
||||
req->callback(req);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
/**
|
||||
* mpc52xx_lpbfifo_poll - Poll for DMA completion
|
||||
*/
|
||||
void mpc52xx_lpbfifo_poll(void)
|
||||
{
|
||||
struct mpc52xx_lpbfifo_request *req = lpbfifo.req;
|
||||
int dma = !(req->flags & MPC52XX_LPBFIFO_FLAG_NO_DMA);
|
||||
int write = req->flags & MPC52XX_LPBFIFO_FLAG_WRITE;
|
||||
|
||||
/*
|
||||
* For more information, see comments on the "Fat Lady"
|
||||
*/
|
||||
if (dma && write)
|
||||
mpc52xx_lpbfifo_irq(0, NULL);
|
||||
else
|
||||
mpc52xx_lpbfifo_bcom_irq(0, NULL);
|
||||
}
|
||||
EXPORT_SYMBOL(mpc52xx_lpbfifo_poll);
|
||||
|
||||
/**
|
||||
* mpc52xx_lpbfifo_submit - Submit an LPB FIFO transfer request.
|
||||
* @req: Pointer to request structure
|
||||
*
|
||||
* Return: %0 on success, -errno code on error
|
||||
*/
|
||||
int mpc52xx_lpbfifo_submit(struct mpc52xx_lpbfifo_request *req)
|
||||
{
|
||||
unsigned long flags;
|
||||
|
||||
if (!lpbfifo.regs)
|
||||
return -ENODEV;
|
||||
|
||||
spin_lock_irqsave(&lpbfifo.lock, flags);
|
||||
|
||||
/* If the req pointer is already set, then a transfer is in progress */
|
||||
if (lpbfifo.req) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
/* Setup the transfer */
|
||||
lpbfifo.req = req;
|
||||
req->irq_count = 0;
|
||||
req->irq_ticks = 0;
|
||||
req->buffer_not_done_cnt = 0;
|
||||
req->pos = 0;
|
||||
|
||||
mpc52xx_lpbfifo_kick(req);
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(mpc52xx_lpbfifo_submit);
|
||||
|
||||
int mpc52xx_lpbfifo_start_xfer(struct mpc52xx_lpbfifo_request *req)
|
||||
{
|
||||
unsigned long flags;
|
||||
|
||||
if (!lpbfifo.regs)
|
||||
return -ENODEV;
|
||||
|
||||
spin_lock_irqsave(&lpbfifo.lock, flags);
|
||||
|
||||
/*
|
||||
* If the req pointer is already set and a transfer was
|
||||
* started on submit, then this transfer is in progress
|
||||
*/
|
||||
if (lpbfifo.req && !lpbfifo.req->defer_xfer_start) {
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
/*
|
||||
* If the req was previously submitted but not
|
||||
* started, start it now
|
||||
*/
|
||||
if (lpbfifo.req && lpbfifo.req == req &&
|
||||
lpbfifo.req->defer_xfer_start) {
|
||||
out_8(lpbfifo.regs + LPBFIFO_REG_PACKET_SIZE, 0x01);
|
||||
}
|
||||
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(mpc52xx_lpbfifo_start_xfer);
|
||||
|
||||
void mpc52xx_lpbfifo_abort(struct mpc52xx_lpbfifo_request *req)
|
||||
{
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&lpbfifo.lock, flags);
|
||||
if (lpbfifo.req == req) {
|
||||
/* Put it into reset and clear the state */
|
||||
bcom_gen_bd_rx_reset(lpbfifo.bcom_rx_task);
|
||||
bcom_gen_bd_tx_reset(lpbfifo.bcom_tx_task);
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x01010000);
|
||||
lpbfifo.req = NULL;
|
||||
}
|
||||
spin_unlock_irqrestore(&lpbfifo.lock, flags);
|
||||
}
|
||||
EXPORT_SYMBOL(mpc52xx_lpbfifo_abort);
|
||||
|
||||
static int mpc52xx_lpbfifo_probe(struct platform_device *op)
|
||||
{
|
||||
struct resource res;
|
||||
int rc = -ENOMEM;
|
||||
|
||||
if (lpbfifo.dev != NULL)
|
||||
return -ENOSPC;
|
||||
|
||||
lpbfifo.irq = irq_of_parse_and_map(op->dev.of_node, 0);
|
||||
if (!lpbfifo.irq)
|
||||
return -ENODEV;
|
||||
|
||||
if (of_address_to_resource(op->dev.of_node, 0, &res))
|
||||
return -ENODEV;
|
||||
lpbfifo.regs_phys = res.start;
|
||||
lpbfifo.regs = of_iomap(op->dev.of_node, 0);
|
||||
if (!lpbfifo.regs)
|
||||
return -ENOMEM;
|
||||
|
||||
spin_lock_init(&lpbfifo.lock);
|
||||
|
||||
/* Put FIFO into reset */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x01010000);
|
||||
|
||||
/* Register the interrupt handler */
|
||||
rc = request_irq(lpbfifo.irq, mpc52xx_lpbfifo_irq, 0,
|
||||
"mpc52xx-lpbfifo", &lpbfifo);
|
||||
if (rc)
|
||||
goto err_irq;
|
||||
|
||||
/* Request the Bestcomm receive (fifo --> memory) task and IRQ */
|
||||
lpbfifo.bcom_rx_task =
|
||||
bcom_gen_bd_rx_init(2, res.start + LPBFIFO_REG_FIFO_DATA,
|
||||
BCOM_INITIATOR_SCLPC, BCOM_IPR_SCLPC,
|
||||
16*1024*1024);
|
||||
if (!lpbfifo.bcom_rx_task)
|
||||
goto err_bcom_rx;
|
||||
|
||||
rc = request_irq(bcom_get_task_irq(lpbfifo.bcom_rx_task),
|
||||
mpc52xx_lpbfifo_bcom_irq, 0,
|
||||
"mpc52xx-lpbfifo-rx", &lpbfifo);
|
||||
if (rc)
|
||||
goto err_bcom_rx_irq;
|
||||
|
||||
lpbfifo.dma_irqs_enabled = 1;
|
||||
|
||||
/* Request the Bestcomm transmit (memory --> fifo) task and IRQ */
|
||||
lpbfifo.bcom_tx_task =
|
||||
bcom_gen_bd_tx_init(2, res.start + LPBFIFO_REG_FIFO_DATA,
|
||||
BCOM_INITIATOR_SCLPC, BCOM_IPR_SCLPC);
|
||||
if (!lpbfifo.bcom_tx_task)
|
||||
goto err_bcom_tx;
|
||||
|
||||
lpbfifo.dev = &op->dev;
|
||||
return 0;
|
||||
|
||||
err_bcom_tx:
|
||||
free_irq(bcom_get_task_irq(lpbfifo.bcom_rx_task), &lpbfifo);
|
||||
err_bcom_rx_irq:
|
||||
bcom_gen_bd_rx_release(lpbfifo.bcom_rx_task);
|
||||
err_bcom_rx:
|
||||
free_irq(lpbfifo.irq, &lpbfifo);
|
||||
err_irq:
|
||||
iounmap(lpbfifo.regs);
|
||||
lpbfifo.regs = NULL;
|
||||
|
||||
dev_err(&op->dev, "mpc52xx_lpbfifo_probe() failed\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
|
||||
static int mpc52xx_lpbfifo_remove(struct platform_device *op)
|
||||
{
|
||||
if (lpbfifo.dev != &op->dev)
|
||||
return 0;
|
||||
|
||||
/* Put FIFO in reset */
|
||||
out_be32(lpbfifo.regs + LPBFIFO_REG_ENABLE, 0x01010000);
|
||||
|
||||
/* Release the bestcomm transmit task */
|
||||
free_irq(bcom_get_task_irq(lpbfifo.bcom_tx_task), &lpbfifo);
|
||||
bcom_gen_bd_tx_release(lpbfifo.bcom_tx_task);
|
||||
|
||||
/* Release the bestcomm receive task */
|
||||
free_irq(bcom_get_task_irq(lpbfifo.bcom_rx_task), &lpbfifo);
|
||||
bcom_gen_bd_rx_release(lpbfifo.bcom_rx_task);
|
||||
|
||||
free_irq(lpbfifo.irq, &lpbfifo);
|
||||
iounmap(lpbfifo.regs);
|
||||
lpbfifo.regs = NULL;
|
||||
lpbfifo.dev = NULL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct of_device_id mpc52xx_lpbfifo_match[] = {
|
||||
{ .compatible = "fsl,mpc5200-lpbfifo", },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, mpc52xx_lpbfifo_match);
|
||||
|
||||
static struct platform_driver mpc52xx_lpbfifo_driver = {
|
||||
.driver = {
|
||||
.name = "mpc52xx-lpbfifo",
|
||||
.of_match_table = mpc52xx_lpbfifo_match,
|
||||
},
|
||||
.probe = mpc52xx_lpbfifo_probe,
|
||||
.remove = mpc52xx_lpbfifo_remove,
|
||||
};
|
||||
module_platform_driver(mpc52xx_lpbfifo_driver);
|
@ -60,7 +60,7 @@ int mpc52xx_set_wakeup_gpio(u8 pin, u8 level)
|
||||
int mpc52xx_pm_prepare(void)
|
||||
{
|
||||
struct device_node *np;
|
||||
const struct of_device_id immr_ids[] = {
|
||||
static const struct of_device_id immr_ids[] = {
|
||||
{ .compatible = "fsl,mpc5200-immr", },
|
||||
{ .compatible = "fsl,mpc5200b-immr", },
|
||||
{ .type = "soc", .compatible = "mpc5200", }, /* lite5200 */
|
||||
|
@ -211,7 +211,7 @@ static struct i2c_driver mcu_driver = {
|
||||
.name = "mcu-mpc8349emitx",
|
||||
.of_match_table = mcu_of_match_table,
|
||||
},
|
||||
.probe_new = mcu_probe,
|
||||
.probe = mcu_probe,
|
||||
.remove = mcu_remove,
|
||||
.id_table = mcu_ids,
|
||||
};
|
||||
|
@ -12,9 +12,6 @@ obj-y += common.o
|
||||
obj-$(CONFIG_BSC9131_RDB) += bsc913x_rdb.o
|
||||
obj-$(CONFIG_BSC9132_QDS) += bsc913x_qds.o
|
||||
obj-$(CONFIG_C293_PCIE) += c293pcie.o
|
||||
obj-$(CONFIG_MPC8540_ADS) += mpc85xx_ads.o
|
||||
obj-$(CONFIG_MPC8560_ADS) += mpc85xx_ads.o
|
||||
obj-$(CONFIG_MPC85xx_CDS) += mpc85xx_cds.o
|
||||
obj-$(CONFIG_MPC8536_DS) += mpc8536_ds.o
|
||||
obj8259-$(CONFIG_PPC_I8259) += mpc85xx_8259.o
|
||||
obj-$(CONFIG_MPC85xx_DS) += mpc85xx_ds.o $(obj8259-y)
|
||||
|
@ -1,162 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC85xx setup and early boot code plus other random bits.
|
||||
*
|
||||
* Maintained by Kumar Gala (see MAINTAINERS for contact information)
|
||||
*
|
||||
* Copyright 2005 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
#include <linux/stddef.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/kdev_t.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <linux/of_platform.h>
|
||||
|
||||
#include <asm/time.h>
|
||||
#include <asm/machdep.h>
|
||||
#include <asm/pci-bridge.h>
|
||||
#include <asm/mpic.h>
|
||||
#include <mm/mmu_decl.h>
|
||||
#include <asm/udbg.h>
|
||||
|
||||
#include <sysdev/fsl_soc.h>
|
||||
#include <sysdev/fsl_pci.h>
|
||||
|
||||
#ifdef CONFIG_CPM2
|
||||
#include <asm/cpm2.h>
|
||||
#include <sysdev/cpm2_pic.h>
|
||||
#endif
|
||||
|
||||
#include "mpc85xx.h"
|
||||
|
||||
static void __init mpc85xx_ads_pic_init(void)
|
||||
{
|
||||
struct mpic *mpic = mpic_alloc(NULL, 0, MPIC_BIG_ENDIAN,
|
||||
0, 256, " OpenPIC ");
|
||||
BUG_ON(mpic == NULL);
|
||||
mpic_init(mpic);
|
||||
|
||||
mpc85xx_cpm2_pic_init();
|
||||
}
|
||||
|
||||
/*
|
||||
* Setup the architecture
|
||||
*/
|
||||
#ifdef CONFIG_CPM2
|
||||
struct cpm_pin {
|
||||
int port, pin, flags;
|
||||
};
|
||||
|
||||
static const struct cpm_pin mpc8560_ads_pins[] = {
|
||||
/* SCC1 */
|
||||
{3, 29, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{3, 30, CPM_PIN_OUTPUT | CPM_PIN_SECONDARY},
|
||||
{3, 31, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
|
||||
/* SCC2 */
|
||||
{2, 12, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{2, 13, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{3, 26, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{3, 27, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{3, 28, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
|
||||
/* FCC2 */
|
||||
{1, 18, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 19, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 20, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 21, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 22, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 23, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 24, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 25, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 26, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 27, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 28, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 29, CPM_PIN_OUTPUT | CPM_PIN_SECONDARY},
|
||||
{1, 30, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 31, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{2, 18, CPM_PIN_INPUT | CPM_PIN_PRIMARY}, /* CLK14 */
|
||||
{2, 19, CPM_PIN_INPUT | CPM_PIN_PRIMARY}, /* CLK13 */
|
||||
|
||||
/* FCC3 */
|
||||
{1, 4, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 5, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 6, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 8, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 9, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 10, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 11, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 12, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 13, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 14, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 15, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
{1, 16, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{1, 17, CPM_PIN_INPUT | CPM_PIN_PRIMARY},
|
||||
{2, 16, CPM_PIN_INPUT | CPM_PIN_PRIMARY}, /* CLK16 */
|
||||
{2, 17, CPM_PIN_INPUT | CPM_PIN_PRIMARY}, /* CLK15 */
|
||||
{2, 27, CPM_PIN_OUTPUT | CPM_PIN_PRIMARY},
|
||||
};
|
||||
|
||||
static void __init init_ioports(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mpc8560_ads_pins); i++) {
|
||||
const struct cpm_pin *pin = &mpc8560_ads_pins[i];
|
||||
cpm2_set_pin(pin->port, pin->pin, pin->flags);
|
||||
}
|
||||
|
||||
cpm2_clk_setup(CPM_CLK_SCC1, CPM_BRG1, CPM_CLK_RX);
|
||||
cpm2_clk_setup(CPM_CLK_SCC1, CPM_BRG1, CPM_CLK_TX);
|
||||
cpm2_clk_setup(CPM_CLK_SCC2, CPM_BRG2, CPM_CLK_RX);
|
||||
cpm2_clk_setup(CPM_CLK_SCC2, CPM_BRG2, CPM_CLK_TX);
|
||||
cpm2_clk_setup(CPM_CLK_FCC2, CPM_CLK13, CPM_CLK_RX);
|
||||
cpm2_clk_setup(CPM_CLK_FCC2, CPM_CLK14, CPM_CLK_TX);
|
||||
cpm2_clk_setup(CPM_CLK_FCC3, CPM_CLK15, CPM_CLK_RX);
|
||||
cpm2_clk_setup(CPM_CLK_FCC3, CPM_CLK16, CPM_CLK_TX);
|
||||
}
|
||||
#endif
|
||||
|
||||
static void __init mpc85xx_ads_setup_arch(void)
|
||||
{
|
||||
if (ppc_md.progress)
|
||||
ppc_md.progress("mpc85xx_ads_setup_arch()", 0);
|
||||
|
||||
#ifdef CONFIG_CPM2
|
||||
cpm2_reset();
|
||||
init_ioports();
|
||||
#endif
|
||||
|
||||
fsl_pci_assign_primary();
|
||||
}
|
||||
|
||||
static void mpc85xx_ads_show_cpuinfo(struct seq_file *m)
|
||||
{
|
||||
uint pvid, svid, phid1;
|
||||
|
||||
pvid = mfspr(SPRN_PVR);
|
||||
svid = mfspr(SPRN_SVR);
|
||||
|
||||
seq_printf(m, "Vendor\t\t: Freescale Semiconductor\n");
|
||||
seq_printf(m, "PVR\t\t: 0x%x\n", pvid);
|
||||
seq_printf(m, "SVR\t\t: 0x%x\n", svid);
|
||||
|
||||
/* Display cpu Pll setting */
|
||||
phid1 = mfspr(SPRN_HID1);
|
||||
seq_printf(m, "PLL setting\t: 0x%x\n", ((phid1 >> 24) & 0x3f));
|
||||
}
|
||||
|
||||
machine_arch_initcall(mpc85xx_ads, mpc85xx_common_publish_devices);
|
||||
|
||||
define_machine(mpc85xx_ads) {
|
||||
.name = "MPC85xx ADS",
|
||||
.compatible = "MPC85xxADS",
|
||||
.setup_arch = mpc85xx_ads_setup_arch,
|
||||
.init_IRQ = mpc85xx_ads_pic_init,
|
||||
.show_cpuinfo = mpc85xx_ads_show_cpuinfo,
|
||||
.get_irq = mpic_get_irq,
|
||||
.progress = udbg_progress,
|
||||
};
|
@ -1,387 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* MPC85xx setup and early boot code plus other random bits.
|
||||
*
|
||||
* Maintained by Kumar Gala (see MAINTAINERS for contact information)
|
||||
*
|
||||
* Copyright 2005, 2011-2012 Freescale Semiconductor Inc.
|
||||
*/
|
||||
|
||||
#include <linux/stddef.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/reboot.h>
|
||||
#include <linux/pci.h>
|
||||
#include <linux/kdev_t.h>
|
||||
#include <linux/major.h>
|
||||
#include <linux/console.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <linux/initrd.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/fsl_devices.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/pgtable.h>
|
||||
|
||||
#include <asm/page.h>
|
||||
#include <linux/atomic.h>
|
||||
#include <asm/time.h>
|
||||
#include <asm/io.h>
|
||||
#include <asm/machdep.h>
|
||||
#include <asm/ipic.h>
|
||||
#include <asm/pci-bridge.h>
|
||||
#include <asm/irq.h>
|
||||
#include <mm/mmu_decl.h>
|
||||
#include <asm/udbg.h>
|
||||
#include <asm/mpic.h>
|
||||
#include <asm/i8259.h>
|
||||
|
||||
#include <sysdev/fsl_soc.h>
|
||||
#include <sysdev/fsl_pci.h>
|
||||
|
||||
#include "mpc85xx.h"
|
||||
|
||||
/*
|
||||
* The CDS board contains an FPGA/CPLD called "Cadmus", which collects
|
||||
* various logic and performs system control functions.
|
||||
* Here is the FPGA/CPLD register map.
|
||||
*/
|
||||
struct cadmus_reg {
|
||||
u8 cm_ver; /* Board version */
|
||||
u8 cm_csr; /* General control/status */
|
||||
u8 cm_rst; /* Reset control */
|
||||
u8 cm_hsclk; /* High speed clock */
|
||||
u8 cm_hsxclk; /* High speed clock extended */
|
||||
u8 cm_led; /* LED data */
|
||||
u8 cm_pci; /* PCI control/status */
|
||||
u8 cm_dma; /* DMA control */
|
||||
u8 res[248]; /* Total 256 bytes */
|
||||
};
|
||||
|
||||
static struct cadmus_reg *cadmus;
|
||||
|
||||
#ifdef CONFIG_PCI
|
||||
|
||||
#define ARCADIA_HOST_BRIDGE_IDSEL 17
|
||||
#define ARCADIA_2ND_BRIDGE_IDSEL 3
|
||||
|
||||
static int mpc85xx_exclude_device(struct pci_controller *hose,
|
||||
u_char bus, u_char devfn)
|
||||
{
|
||||
/* We explicitly do not go past the Tundra 320 Bridge */
|
||||
if ((bus == 1) && (PCI_SLOT(devfn) == ARCADIA_2ND_BRIDGE_IDSEL))
|
||||
return PCIBIOS_DEVICE_NOT_FOUND;
|
||||
if ((bus == 0) && (PCI_SLOT(devfn) == ARCADIA_2ND_BRIDGE_IDSEL))
|
||||
return PCIBIOS_DEVICE_NOT_FOUND;
|
||||
else
|
||||
return PCIBIOS_SUCCESSFUL;
|
||||
}
|
||||
|
||||
static int mpc85xx_cds_restart(struct notifier_block *this,
|
||||
unsigned long mode, void *cmd)
|
||||
{
|
||||
struct pci_dev *dev;
|
||||
u_char tmp;
|
||||
|
||||
if ((dev = pci_get_device(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686,
|
||||
NULL))) {
|
||||
|
||||
/* Use the VIA Super Southbridge to force a PCI reset */
|
||||
pci_read_config_byte(dev, 0x47, &tmp);
|
||||
pci_write_config_byte(dev, 0x47, tmp | 1);
|
||||
|
||||
/* Flush the outbound PCI write queues */
|
||||
pci_read_config_byte(dev, 0x47, &tmp);
|
||||
|
||||
/*
|
||||
* At this point, the hardware reset should have triggered.
|
||||
* However, if it doesn't work for some mysterious reason,
|
||||
* just fall through to the default reset below.
|
||||
*/
|
||||
|
||||
pci_dev_put(dev);
|
||||
}
|
||||
|
||||
/*
|
||||
* If we can't find the VIA chip (maybe the P2P bridge is
|
||||
* disabled) or the VIA chip reset didn't work, just return
|
||||
* and let default reset sequence happen.
|
||||
*/
|
||||
return NOTIFY_DONE;
|
||||
}
|
||||
|
||||
static int mpc85xx_cds_restart_register(void)
|
||||
{
|
||||
static struct notifier_block restart_handler;
|
||||
|
||||
restart_handler.notifier_call = mpc85xx_cds_restart;
|
||||
restart_handler.priority = 192;
|
||||
|
||||
return register_restart_handler(&restart_handler);
|
||||
}
|
||||
machine_arch_initcall(mpc85xx_cds, mpc85xx_cds_restart_register);
|
||||
|
||||
|
||||
static void __init mpc85xx_cds_pci_irq_fixup(struct pci_dev *dev)
|
||||
{
|
||||
u_char c;
|
||||
if (dev->vendor == PCI_VENDOR_ID_VIA) {
|
||||
switch (dev->device) {
|
||||
case PCI_DEVICE_ID_VIA_82C586_1:
|
||||
/*
|
||||
* U-Boot does not set the enable bits
|
||||
* for the IDE device. Force them on here.
|
||||
*/
|
||||
pci_read_config_byte(dev, 0x40, &c);
|
||||
c |= 0x03; /* IDE: Chip Enable Bits */
|
||||
pci_write_config_byte(dev, 0x40, c);
|
||||
|
||||
/*
|
||||
* Since only primary interface works, force the
|
||||
* IDE function to standard primary IDE interrupt
|
||||
* w/ 8259 offset
|
||||
*/
|
||||
dev->irq = 14;
|
||||
pci_write_config_byte(dev, PCI_INTERRUPT_LINE, dev->irq);
|
||||
break;
|
||||
/*
|
||||
* Force legacy USB interrupt routing
|
||||
*/
|
||||
case PCI_DEVICE_ID_VIA_82C586_2:
|
||||
/* There are two USB controllers.
|
||||
* Identify them by function number
|
||||
*/
|
||||
if (PCI_FUNC(dev->devfn) == 3)
|
||||
dev->irq = 11;
|
||||
else
|
||||
dev->irq = 10;
|
||||
pci_write_config_byte(dev, PCI_INTERRUPT_LINE, dev->irq);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void skip_fake_bridge(struct pci_dev *dev)
|
||||
{
|
||||
/* Make it an error to skip the fake bridge
|
||||
* in pci_setup_device() in probe.c */
|
||||
dev->hdr_type = 0x7f;
|
||||
}
|
||||
DECLARE_PCI_FIXUP_EARLY(0x1957, 0x3fff, skip_fake_bridge);
|
||||
DECLARE_PCI_FIXUP_EARLY(0x3fff, 0x1957, skip_fake_bridge);
|
||||
DECLARE_PCI_FIXUP_EARLY(0xff3f, 0x5719, skip_fake_bridge);
|
||||
|
||||
#define PCI_DEVICE_ID_IDT_TSI310 0x01a7
|
||||
|
||||
/*
|
||||
* Fix Tsi310 PCI-X bridge resource.
|
||||
* Force the bridge to open a window from 0x0000-0x1fff in PCI I/O space.
|
||||
* This allows legacy I/O(i8259, etc) on the VIA southbridge to be accessed.
|
||||
*/
|
||||
void mpc85xx_cds_fixup_bus(struct pci_bus *bus)
|
||||
{
|
||||
struct pci_dev *dev = bus->self;
|
||||
struct resource *res = bus->resource[0];
|
||||
|
||||
if (dev != NULL &&
|
||||
dev->vendor == PCI_VENDOR_ID_IBM &&
|
||||
dev->device == PCI_DEVICE_ID_IDT_TSI310) {
|
||||
if (res) {
|
||||
res->start = 0;
|
||||
res->end = 0x1fff;
|
||||
res->flags = IORESOURCE_IO;
|
||||
pr_info("mpc85xx_cds: PCI bridge resource fixup applied\n");
|
||||
pr_info("mpc85xx_cds: %pR\n", res);
|
||||
}
|
||||
}
|
||||
|
||||
fsl_pcibios_fixup_bus(bus);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PPC_I8259
|
||||
static void mpc85xx_8259_cascade_handler(struct irq_desc *desc)
|
||||
{
|
||||
unsigned int cascade_irq = i8259_irq();
|
||||
|
||||
if (cascade_irq)
|
||||
/* handle an interrupt from the 8259 */
|
||||
generic_handle_irq(cascade_irq);
|
||||
|
||||
/* check for any interrupts from the shared IRQ line */
|
||||
handle_fasteoi_irq(desc);
|
||||
}
|
||||
|
||||
static irqreturn_t mpc85xx_8259_cascade_action(int irq, void *dev_id)
|
||||
{
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
#endif /* PPC_I8259 */
|
||||
#endif /* CONFIG_PCI */
|
||||
|
||||
static void __init mpc85xx_cds_pic_init(void)
|
||||
{
|
||||
struct mpic *mpic;
|
||||
mpic = mpic_alloc(NULL, 0, MPIC_BIG_ENDIAN,
|
||||
0, 256, " OpenPIC ");
|
||||
BUG_ON(mpic == NULL);
|
||||
mpic_init(mpic);
|
||||
}
|
||||
|
||||
#if defined(CONFIG_PPC_I8259) && defined(CONFIG_PCI)
|
||||
static int mpc85xx_cds_8259_attach(void)
|
||||
{
|
||||
int ret;
|
||||
struct device_node *np = NULL;
|
||||
struct device_node *cascade_node = NULL;
|
||||
int cascade_irq;
|
||||
|
||||
/* Initialize the i8259 controller */
|
||||
for_each_node_by_type(np, "interrupt-controller")
|
||||
if (of_device_is_compatible(np, "chrp,iic")) {
|
||||
cascade_node = np;
|
||||
break;
|
||||
}
|
||||
|
||||
if (cascade_node == NULL) {
|
||||
printk(KERN_DEBUG "Could not find i8259 PIC\n");
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
cascade_irq = irq_of_parse_and_map(cascade_node, 0);
|
||||
if (!cascade_irq) {
|
||||
printk(KERN_ERR "Failed to map cascade interrupt\n");
|
||||
return -ENXIO;
|
||||
}
|
||||
|
||||
i8259_init(cascade_node, 0);
|
||||
of_node_put(cascade_node);
|
||||
|
||||
/*
|
||||
* Hook the interrupt to make sure desc->action is never NULL.
|
||||
* This is required to ensure that the interrupt does not get
|
||||
* disabled when the last user of the shared IRQ line frees their
|
||||
* interrupt.
|
||||
*/
|
||||
ret = request_irq(cascade_irq, mpc85xx_8259_cascade_action,
|
||||
IRQF_SHARED | IRQF_NO_THREAD, "8259 cascade",
|
||||
cascade_node);
|
||||
if (ret) {
|
||||
printk(KERN_ERR "Failed to setup cascade interrupt\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Success. Connect our low-level cascade handler. */
|
||||
irq_set_handler(cascade_irq, mpc85xx_8259_cascade_handler);
|
||||
|
||||
return 0;
|
||||
}
|
||||
machine_device_initcall(mpc85xx_cds, mpc85xx_cds_8259_attach);
|
||||
|
||||
#endif /* CONFIG_PPC_I8259 */
|
||||
|
||||
static void __init mpc85xx_cds_pci_assign_primary(void)
|
||||
{
|
||||
#ifdef CONFIG_PCI
|
||||
struct device_node *np;
|
||||
|
||||
if (fsl_pci_primary)
|
||||
return;
|
||||
|
||||
/*
|
||||
* MPC85xx_CDS has ISA bridge but unfortunately there is no
|
||||
* isa node in device tree. We now looking for i8259 node as
|
||||
* a workaround for such a broken device tree. This routine
|
||||
* is for complying to all device trees.
|
||||
*/
|
||||
np = of_find_node_by_name(NULL, "i8259");
|
||||
while ((fsl_pci_primary = of_get_parent(np))) {
|
||||
of_node_put(np);
|
||||
np = fsl_pci_primary;
|
||||
|
||||
if ((of_device_is_compatible(np, "fsl,mpc8540-pci") ||
|
||||
of_device_is_compatible(np, "fsl,mpc8548-pcie")) &&
|
||||
of_device_is_available(np))
|
||||
return;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
/*
|
||||
* Setup the architecture
|
||||
*/
|
||||
static void __init mpc85xx_cds_setup_arch(void)
|
||||
{
|
||||
struct device_node *np;
|
||||
int cds_pci_slot;
|
||||
|
||||
if (ppc_md.progress)
|
||||
ppc_md.progress("mpc85xx_cds_setup_arch()", 0);
|
||||
|
||||
np = of_find_compatible_node(NULL, NULL, "fsl,mpc8548cds-fpga");
|
||||
if (!np) {
|
||||
pr_err("Could not find FPGA node.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
cadmus = of_iomap(np, 0);
|
||||
of_node_put(np);
|
||||
if (!cadmus) {
|
||||
pr_err("Fail to map FPGA area.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
if (ppc_md.progress) {
|
||||
char buf[40];
|
||||
cds_pci_slot = ((in_8(&cadmus->cm_csr) >> 6) & 0x3) + 1;
|
||||
snprintf(buf, 40, "CDS Version = 0x%x in slot %d\n",
|
||||
in_8(&cadmus->cm_ver), cds_pci_slot);
|
||||
ppc_md.progress(buf, 0);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PCI
|
||||
ppc_md.pci_irq_fixup = mpc85xx_cds_pci_irq_fixup;
|
||||
ppc_md.pci_exclude_device = mpc85xx_exclude_device;
|
||||
#endif
|
||||
|
||||
mpc85xx_cds_pci_assign_primary();
|
||||
fsl_pci_assign_primary();
|
||||
}
|
||||
|
||||
static void mpc85xx_cds_show_cpuinfo(struct seq_file *m)
|
||||
{
|
||||
uint pvid, svid, phid1;
|
||||
|
||||
pvid = mfspr(SPRN_PVR);
|
||||
svid = mfspr(SPRN_SVR);
|
||||
|
||||
seq_printf(m, "Vendor\t\t: Freescale Semiconductor\n");
|
||||
seq_printf(m, "Machine\t\t: MPC85xx CDS (0x%x)\n",
|
||||
in_8(&cadmus->cm_ver));
|
||||
seq_printf(m, "PVR\t\t: 0x%x\n", pvid);
|
||||
seq_printf(m, "SVR\t\t: 0x%x\n", svid);
|
||||
|
||||
/* Display cpu Pll setting */
|
||||
phid1 = mfspr(SPRN_HID1);
|
||||
seq_printf(m, "PLL setting\t: 0x%x\n", ((phid1 >> 24) & 0x3f));
|
||||
}
|
||||
|
||||
machine_arch_initcall(mpc85xx_cds, mpc85xx_common_publish_devices);
|
||||
|
||||
define_machine(mpc85xx_cds) {
|
||||
.name = "MPC85xx CDS",
|
||||
.compatible = "MPC85xxCDS",
|
||||
.setup_arch = mpc85xx_cds_setup_arch,
|
||||
.init_IRQ = mpc85xx_cds_pic_init,
|
||||
.show_cpuinfo = mpc85xx_cds_show_cpuinfo,
|
||||
.get_irq = mpic_get_irq,
|
||||
#ifdef CONFIG_PCI
|
||||
.pcibios_fixup_bus = mpc85xx_cds_fixup_bus,
|
||||
.pcibios_fixup_phb = fsl_pcibios_fixup_phb,
|
||||
#endif
|
||||
.progress = udbg_progress,
|
||||
};
|
@ -1,5 +1,4 @@
|
||||
# SPDX-License-Identifier: GPL-2.0
|
||||
config PPC_86xx
|
||||
menuconfig PPC_86xx
|
||||
bool "86xx-based boards"
|
||||
depends on PPC_BOOK3S_32
|
||||
|
@ -326,12 +326,6 @@ spu_irq_class_1(int irq, void *data)
|
||||
if (stat & CLASS1_STORAGE_FAULT_INTR)
|
||||
__spu_trap_data_map(spu, dar, dsisr);
|
||||
|
||||
if (stat & CLASS1_LS_COMPARE_SUSPEND_ON_GET_INTR)
|
||||
;
|
||||
|
||||
if (stat & CLASS1_LS_COMPARE_SUSPEND_ON_PUT_INTR)
|
||||
;
|
||||
|
||||
spu->class_1_dsisr = 0;
|
||||
spu->class_1_dar = 0;
|
||||
|
||||
|
@ -10,7 +10,7 @@ config LINKSTATION
|
||||
select FSL_SOC
|
||||
select PPC_UDBG_16550 if SERIAL_8250
|
||||
select DEFAULT_UIMAGE
|
||||
select MPC10X_BRIDGE
|
||||
imply MPC10X_BRIDGE if PCI
|
||||
help
|
||||
Select LINKSTATION if configuring for one of PPC- (MPC8241)
|
||||
based NAS systems from Buffalo Technology. So far only
|
||||
@ -24,7 +24,7 @@ config STORCENTER
|
||||
select MPIC
|
||||
select FSL_SOC
|
||||
select PPC_UDBG_16550 if SERIAL_8250
|
||||
select MPC10X_BRIDGE
|
||||
imply MPC10X_BRIDGE if PCI
|
||||
help
|
||||
Select STORCENTER if configuring for the iomega StorCenter
|
||||
with an 8241 CPU in it.
|
||||
|
@ -1053,11 +1053,11 @@ core99_reset_cpu(struct device_node *node, long param, long value)
|
||||
return -ENODEV;
|
||||
|
||||
for_each_of_cpu_node(np) {
|
||||
const u32 *num = of_get_property(np, "reg", NULL);
|
||||
const u32 *rst = of_get_property(np, "soft-reset", NULL);
|
||||
if (num == NULL || rst == NULL)
|
||||
if (!rst)
|
||||
continue;
|
||||
if (param == *num) {
|
||||
if (param == of_get_cpu_hwid(np, 0)) {
|
||||
of_node_put(np);
|
||||
reset_io = *rst;
|
||||
break;
|
||||
}
|
||||
@ -1499,11 +1499,11 @@ static long g5_reset_cpu(struct device_node *node, long param, long value)
|
||||
return -ENODEV;
|
||||
|
||||
for_each_of_cpu_node(np) {
|
||||
const u32 *num = of_get_property(np, "reg", NULL);
|
||||
const u32 *rst = of_get_property(np, "soft-reset", NULL);
|
||||
if (num == NULL || rst == NULL)
|
||||
if (!rst)
|
||||
continue;
|
||||
if (param == *num) {
|
||||
if (param == of_get_cpu_hwid(np, 0)) {
|
||||
of_node_put(np);
|
||||
reset_io = *rst;
|
||||
break;
|
||||
}
|
||||
|
@ -246,9 +246,9 @@ static inline void atomic_lock_thread_idle(void)
|
||||
{
|
||||
int cpu = raw_smp_processor_id();
|
||||
int first = cpu_first_thread_sibling(cpu);
|
||||
unsigned long *state = &paca_ptrs[first]->idle_state;
|
||||
unsigned long *lock = &paca_ptrs[first]->idle_lock;
|
||||
|
||||
while (unlikely(test_and_set_bit_lock(NR_PNV_CORE_IDLE_LOCK_BIT, state)))
|
||||
while (unlikely(test_and_set_bit_lock(NR_PNV_CORE_IDLE_LOCK_BIT, lock)))
|
||||
barrier();
|
||||
}
|
||||
|
||||
@ -258,29 +258,31 @@ static inline void atomic_unlock_and_stop_thread_idle(void)
|
||||
int first = cpu_first_thread_sibling(cpu);
|
||||
unsigned long thread = 1UL << cpu_thread_in_core(cpu);
|
||||
unsigned long *state = &paca_ptrs[first]->idle_state;
|
||||
unsigned long *lock = &paca_ptrs[first]->idle_lock;
|
||||
u64 s = READ_ONCE(*state);
|
||||
u64 new, tmp;
|
||||
|
||||
BUG_ON(!(s & PNV_CORE_IDLE_LOCK_BIT));
|
||||
BUG_ON(!(READ_ONCE(*lock) & PNV_CORE_IDLE_LOCK_BIT));
|
||||
BUG_ON(s & thread);
|
||||
|
||||
again:
|
||||
new = (s | thread) & ~PNV_CORE_IDLE_LOCK_BIT;
|
||||
new = s | thread;
|
||||
tmp = cmpxchg(state, s, new);
|
||||
if (unlikely(tmp != s)) {
|
||||
s = tmp;
|
||||
goto again;
|
||||
}
|
||||
clear_bit_unlock(NR_PNV_CORE_IDLE_LOCK_BIT, lock);
|
||||
}
|
||||
|
||||
static inline void atomic_unlock_thread_idle(void)
|
||||
{
|
||||
int cpu = raw_smp_processor_id();
|
||||
int first = cpu_first_thread_sibling(cpu);
|
||||
unsigned long *state = &paca_ptrs[first]->idle_state;
|
||||
unsigned long *lock = &paca_ptrs[first]->idle_lock;
|
||||
|
||||
BUG_ON(!test_bit(NR_PNV_CORE_IDLE_LOCK_BIT, state));
|
||||
clear_bit_unlock(NR_PNV_CORE_IDLE_LOCK_BIT, state);
|
||||
BUG_ON(!test_bit(NR_PNV_CORE_IDLE_LOCK_BIT, lock));
|
||||
clear_bit_unlock(NR_PNV_CORE_IDLE_LOCK_BIT, lock);
|
||||
}
|
||||
|
||||
/* P7 and P8 */
|
||||
|
@ -167,8 +167,6 @@ OPAL_CALL(opal_pci_map_pe_mmio_window, OPAL_PCI_MAP_PE_MMIO_WINDOW);
|
||||
OPAL_CALL(opal_pci_set_phb_table_memory, OPAL_PCI_SET_PHB_TABLE_MEMORY);
|
||||
OPAL_CALL(opal_pci_set_pe, OPAL_PCI_SET_PE);
|
||||
OPAL_CALL(opal_pci_set_peltv, OPAL_PCI_SET_PELTV);
|
||||
OPAL_CALL(opal_pci_set_mve, OPAL_PCI_SET_MVE);
|
||||
OPAL_CALL(opal_pci_set_mve_enable, OPAL_PCI_SET_MVE_ENABLE);
|
||||
OPAL_CALL(opal_pci_get_xive_reissue, OPAL_PCI_GET_XIVE_REISSUE);
|
||||
OPAL_CALL(opal_pci_set_xive_reissue, OPAL_PCI_SET_XIVE_REISSUE);
|
||||
OPAL_CALL(opal_pci_set_xive_pe, OPAL_PCI_SET_XIVE_PE);
|
||||
|
@ -59,7 +59,7 @@ again:
|
||||
|
||||
cond_resched();
|
||||
}
|
||||
last_outstanding_events = 0;
|
||||
WRITE_ONCE(last_outstanding_events, 0);
|
||||
if (opal_poll_events(&events) != OPAL_SUCCESS)
|
||||
return;
|
||||
e = be64_to_cpu(events) & opal_event_irqchip.mask;
|
||||
@ -69,7 +69,7 @@ again:
|
||||
|
||||
bool opal_have_pending_events(void)
|
||||
{
|
||||
if (last_outstanding_events & opal_event_irqchip.mask)
|
||||
if (READ_ONCE(last_outstanding_events) & opal_event_irqchip.mask)
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
@ -124,7 +124,7 @@ static irqreturn_t opal_interrupt(int irq, void *data)
|
||||
__be64 events;
|
||||
|
||||
opal_handle_interrupt(virq_to_hw(irq), &events);
|
||||
last_outstanding_events = be64_to_cpu(events);
|
||||
WRITE_ONCE(last_outstanding_events, be64_to_cpu(events));
|
||||
if (opal_have_pending_events())
|
||||
opal_wake_poller();
|
||||
|
||||
|
@ -45,11 +45,8 @@
|
||||
#include "pci.h"
|
||||
#include "../../../../drivers/pci/pci.h"
|
||||
|
||||
#define PNV_IODA1_M64_NUM 16 /* Number of M64 BARs */
|
||||
#define PNV_IODA1_M64_SEGS 8 /* Segments per M64 BAR */
|
||||
#define PNV_IODA1_DMA32_SEGSIZE 0x10000000
|
||||
|
||||
static const char * const pnv_phb_names[] = { "IODA1", "IODA2", "NPU_OCAPI" };
|
||||
/* This array is indexed with enum pnv_phb_type */
|
||||
static const char * const pnv_phb_names[] = { "IODA2", "NPU_OCAPI" };
|
||||
|
||||
static void pnv_pci_ioda2_set_bypass(struct pnv_ioda_pe *pe, bool enable);
|
||||
static void pnv_pci_configure_bus(struct pci_bus *bus);
|
||||
@ -280,86 +277,6 @@ static void pnv_ioda_reserve_dev_m64_pe(struct pci_dev *pdev,
|
||||
}
|
||||
}
|
||||
|
||||
static int pnv_ioda1_init_m64(struct pnv_phb *phb)
|
||||
{
|
||||
struct resource *r;
|
||||
int index;
|
||||
|
||||
/*
|
||||
* There are 16 M64 BARs, each of which has 8 segments. So
|
||||
* there are as many M64 segments as the maximum number of
|
||||
* PEs, which is 128.
|
||||
*/
|
||||
for (index = 0; index < PNV_IODA1_M64_NUM; index++) {
|
||||
unsigned long base, segsz = phb->ioda.m64_segsize;
|
||||
int64_t rc;
|
||||
|
||||
base = phb->ioda.m64_base +
|
||||
index * PNV_IODA1_M64_SEGS * segsz;
|
||||
rc = opal_pci_set_phb_mem_window(phb->opal_id,
|
||||
OPAL_M64_WINDOW_TYPE, index, base, 0,
|
||||
PNV_IODA1_M64_SEGS * segsz);
|
||||
if (rc != OPAL_SUCCESS) {
|
||||
pr_warn(" Error %lld setting M64 PHB#%x-BAR#%d\n",
|
||||
rc, phb->hose->global_number, index);
|
||||
goto fail;
|
||||
}
|
||||
|
||||
rc = opal_pci_phb_mmio_enable(phb->opal_id,
|
||||
OPAL_M64_WINDOW_TYPE, index,
|
||||
OPAL_ENABLE_M64_SPLIT);
|
||||
if (rc != OPAL_SUCCESS) {
|
||||
pr_warn(" Error %lld enabling M64 PHB#%x-BAR#%d\n",
|
||||
rc, phb->hose->global_number, index);
|
||||
goto fail;
|
||||
}
|
||||
}
|
||||
|
||||
for (index = 0; index < phb->ioda.total_pe_num; index++) {
|
||||
int64_t rc;
|
||||
|
||||
/*
|
||||
* P7IOC supports M64DT, which helps mapping M64 segment
|
||||
* to one particular PE#. However, PHB3 has fixed mapping
|
||||
* between M64 segment and PE#. In order to have same logic
|
||||
* for P7IOC and PHB3, we enforce fixed mapping between M64
|
||||
* segment and PE# on P7IOC.
|
||||
*/
|
||||
rc = opal_pci_map_pe_mmio_window(phb->opal_id,
|
||||
index, OPAL_M64_WINDOW_TYPE,
|
||||
index / PNV_IODA1_M64_SEGS,
|
||||
index % PNV_IODA1_M64_SEGS);
|
||||
if (rc != OPAL_SUCCESS) {
|
||||
pr_warn("%s: Error %lld mapping M64 for PHB#%x-PE#%x\n",
|
||||
__func__, rc, phb->hose->global_number,
|
||||
index);
|
||||
goto fail;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Exclude the segments for reserved and root bus PE, which
|
||||
* are first or last two PEs.
|
||||
*/
|
||||
r = &phb->hose->mem_resources[1];
|
||||
if (phb->ioda.reserved_pe_idx == 0)
|
||||
r->start += (2 * phb->ioda.m64_segsize);
|
||||
else if (phb->ioda.reserved_pe_idx == (phb->ioda.total_pe_num - 1))
|
||||
r->end -= (2 * phb->ioda.m64_segsize);
|
||||
else
|
||||
WARN(1, "Wrong reserved PE#%x on PHB#%x\n",
|
||||
phb->ioda.reserved_pe_idx, phb->hose->global_number);
|
||||
|
||||
return 0;
|
||||
|
||||
fail:
|
||||
for ( ; index >= 0; index--)
|
||||
opal_pci_phb_mmio_enable(phb->opal_id,
|
||||
OPAL_M64_WINDOW_TYPE, index, OPAL_DISABLE_M64);
|
||||
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
static void pnv_ioda_reserve_m64_pe(struct pci_bus *bus,
|
||||
unsigned long *pe_bitmap,
|
||||
bool all)
|
||||
@ -443,7 +360,7 @@ static void __init pnv_ioda_parse_m64_window(struct pnv_phb *phb)
|
||||
const __be32 *r;
|
||||
u64 pci_addr;
|
||||
|
||||
if (phb->type != PNV_PHB_IODA1 && phb->type != PNV_PHB_IODA2) {
|
||||
if (phb->type != PNV_PHB_IODA2) {
|
||||
pr_info(" Not support M64 window\n");
|
||||
return;
|
||||
}
|
||||
@ -518,9 +435,6 @@ static void __init pnv_ioda_parse_m64_window(struct pnv_phb *phb)
|
||||
* Setup init functions for M64 based on IODA version, IODA3 uses
|
||||
* the IODA2 code.
|
||||
*/
|
||||
if (phb->type == PNV_PHB_IODA1)
|
||||
phb->init_m64 = pnv_ioda1_init_m64;
|
||||
else
|
||||
phb->init_m64 = pnv_ioda2_init_m64;
|
||||
}
|
||||
|
||||
@ -952,29 +866,8 @@ int pnv_ioda_configure_pe(struct pnv_phb *phb, struct pnv_ioda_pe *pe)
|
||||
for (rid = pe->rid; rid < rid_end; rid++)
|
||||
phb->ioda.pe_rmap[rid] = pe->pe_number;
|
||||
|
||||
/* Setup one MVTs on IODA1 */
|
||||
if (phb->type != PNV_PHB_IODA1) {
|
||||
pe->mve_number = 0;
|
||||
goto out;
|
||||
}
|
||||
|
||||
pe->mve_number = pe->pe_number;
|
||||
rc = opal_pci_set_mve(phb->opal_id, pe->mve_number, pe->pe_number);
|
||||
if (rc != OPAL_SUCCESS) {
|
||||
pe_err(pe, "OPAL error %ld setting up MVE %x\n",
|
||||
rc, pe->mve_number);
|
||||
pe->mve_number = -1;
|
||||
} else {
|
||||
rc = opal_pci_set_mve_enable(phb->opal_id,
|
||||
pe->mve_number, OPAL_ENABLE_MVE);
|
||||
if (rc) {
|
||||
pe_err(pe, "OPAL error %ld enabling MVE %x\n",
|
||||
rc, pe->mve_number);
|
||||
pe->mve_number = -1;
|
||||
}
|
||||
}
|
||||
|
||||
out:
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -1097,9 +990,6 @@ static struct pnv_ioda_pe *pnv_ioda_setup_bus_PE(struct pci_bus *bus, bool all)
|
||||
return pe;
|
||||
}
|
||||
|
||||
static void pnv_pci_ioda1_setup_dma_pe(struct pnv_phb *phb,
|
||||
struct pnv_ioda_pe *pe);
|
||||
|
||||
static void pnv_pci_ioda_dma_dev_setup(struct pci_dev *pdev)
|
||||
{
|
||||
struct pnv_phb *phb = pci_bus_to_pnvhb(pdev->bus);
|
||||
@ -1134,9 +1024,6 @@ static void pnv_pci_ioda_dma_dev_setup(struct pci_dev *pdev)
|
||||
*/
|
||||
if (!pe->dma_setup_done && !pci_is_bridge(pdev)) {
|
||||
switch (phb->type) {
|
||||
case PNV_PHB_IODA1:
|
||||
pnv_pci_ioda1_setup_dma_pe(phb, pe);
|
||||
break;
|
||||
case PNV_PHB_IODA2:
|
||||
pnv_pci_ioda2_setup_dma_pe(phb, pe);
|
||||
break;
|
||||
@ -1273,53 +1160,6 @@ static inline __be64 __iomem *pnv_ioda_get_inval_reg(struct pnv_phb *phb)
|
||||
return phb->regs + 0x210;
|
||||
}
|
||||
|
||||
static void pnv_pci_p7ioc_tce_invalidate(struct iommu_table *tbl,
|
||||
unsigned long index, unsigned long npages)
|
||||
{
|
||||
struct iommu_table_group_link *tgl = list_first_entry_or_null(
|
||||
&tbl->it_group_list, struct iommu_table_group_link,
|
||||
next);
|
||||
struct pnv_ioda_pe *pe = container_of(tgl->table_group,
|
||||
struct pnv_ioda_pe, table_group);
|
||||
__be64 __iomem *invalidate = pnv_ioda_get_inval_reg(pe->phb);
|
||||
unsigned long start, end, inc;
|
||||
|
||||
start = __pa(((__be64 *)tbl->it_base) + index - tbl->it_offset);
|
||||
end = __pa(((__be64 *)tbl->it_base) + index - tbl->it_offset +
|
||||
npages - 1);
|
||||
|
||||
/* p7ioc-style invalidation, 2 TCEs per write */
|
||||
start |= (1ull << 63);
|
||||
end |= (1ull << 63);
|
||||
inc = 16;
|
||||
end |= inc - 1; /* round up end to be different than start */
|
||||
|
||||
mb(); /* Ensure above stores are visible */
|
||||
while (start <= end) {
|
||||
__raw_writeq_be(start, invalidate);
|
||||
start += inc;
|
||||
}
|
||||
|
||||
/*
|
||||
* The iommu layer will do another mb() for us on build()
|
||||
* and we don't care on free()
|
||||
*/
|
||||
}
|
||||
|
||||
static int pnv_ioda1_tce_build(struct iommu_table *tbl, long index,
|
||||
long npages, unsigned long uaddr,
|
||||
enum dma_data_direction direction,
|
||||
unsigned long attrs)
|
||||
{
|
||||
int ret = pnv_tce_build(tbl, index, npages, uaddr, direction,
|
||||
attrs);
|
||||
|
||||
if (!ret)
|
||||
pnv_pci_p7ioc_tce_invalidate(tbl, index, npages);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_IOMMU_API
|
||||
/* Common for IODA1 and IODA2 */
|
||||
static int pnv_ioda_tce_xchg_no_kill(struct iommu_table *tbl, long index,
|
||||
@ -1329,25 +1169,6 @@ static int pnv_ioda_tce_xchg_no_kill(struct iommu_table *tbl, long index,
|
||||
}
|
||||
#endif
|
||||
|
||||
static void pnv_ioda1_tce_free(struct iommu_table *tbl, long index,
|
||||
long npages)
|
||||
{
|
||||
pnv_tce_free(tbl, index, npages);
|
||||
|
||||
pnv_pci_p7ioc_tce_invalidate(tbl, index, npages);
|
||||
}
|
||||
|
||||
static struct iommu_table_ops pnv_ioda1_iommu_ops = {
|
||||
.set = pnv_ioda1_tce_build,
|
||||
#ifdef CONFIG_IOMMU_API
|
||||
.xchg_no_kill = pnv_ioda_tce_xchg_no_kill,
|
||||
.tce_kill = pnv_pci_p7ioc_tce_invalidate,
|
||||
.useraddrptr = pnv_tce_useraddrptr,
|
||||
#endif
|
||||
.clear = pnv_ioda1_tce_free,
|
||||
.get = pnv_tce_get,
|
||||
};
|
||||
|
||||
#define PHB3_TCE_KILL_INVAL_ALL PPC_BIT(0)
|
||||
#define PHB3_TCE_KILL_INVAL_PE PPC_BIT(1)
|
||||
#define PHB3_TCE_KILL_INVAL_ONE PPC_BIT(2)
|
||||
@ -1453,182 +1274,6 @@ static struct iommu_table_ops pnv_ioda2_iommu_ops = {
|
||||
.free = pnv_pci_ioda2_table_free_pages,
|
||||
};
|
||||
|
||||
static int pnv_pci_ioda_dev_dma_weight(struct pci_dev *dev, void *data)
|
||||
{
|
||||
unsigned int *weight = (unsigned int *)data;
|
||||
|
||||
/* This is quite simplistic. The "base" weight of a device
|
||||
* is 10. 0 means no DMA is to be accounted for it.
|
||||
*/
|
||||
if (dev->hdr_type != PCI_HEADER_TYPE_NORMAL)
|
||||
return 0;
|
||||
|
||||
if (dev->class == PCI_CLASS_SERIAL_USB_UHCI ||
|
||||
dev->class == PCI_CLASS_SERIAL_USB_OHCI ||
|
||||
dev->class == PCI_CLASS_SERIAL_USB_EHCI)
|
||||
*weight += 3;
|
||||
else if ((dev->class >> 8) == PCI_CLASS_STORAGE_RAID)
|
||||
*weight += 15;
|
||||
else
|
||||
*weight += 10;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static unsigned int pnv_pci_ioda_pe_dma_weight(struct pnv_ioda_pe *pe)
|
||||
{
|
||||
unsigned int weight = 0;
|
||||
|
||||
/* SRIOV VF has same DMA32 weight as its PF */
|
||||
#ifdef CONFIG_PCI_IOV
|
||||
if ((pe->flags & PNV_IODA_PE_VF) && pe->parent_dev) {
|
||||
pnv_pci_ioda_dev_dma_weight(pe->parent_dev, &weight);
|
||||
return weight;
|
||||
}
|
||||
#endif
|
||||
|
||||
if ((pe->flags & PNV_IODA_PE_DEV) && pe->pdev) {
|
||||
pnv_pci_ioda_dev_dma_weight(pe->pdev, &weight);
|
||||
} else if ((pe->flags & PNV_IODA_PE_BUS) && pe->pbus) {
|
||||
struct pci_dev *pdev;
|
||||
|
||||
list_for_each_entry(pdev, &pe->pbus->devices, bus_list)
|
||||
pnv_pci_ioda_dev_dma_weight(pdev, &weight);
|
||||
} else if ((pe->flags & PNV_IODA_PE_BUS_ALL) && pe->pbus) {
|
||||
pci_walk_bus(pe->pbus, pnv_pci_ioda_dev_dma_weight, &weight);
|
||||
}
|
||||
|
||||
return weight;
|
||||
}
|
||||
|
||||
static void pnv_pci_ioda1_setup_dma_pe(struct pnv_phb *phb,
|
||||
struct pnv_ioda_pe *pe)
|
||||
{
|
||||
|
||||
struct page *tce_mem = NULL;
|
||||
struct iommu_table *tbl;
|
||||
unsigned int weight, total_weight = 0;
|
||||
unsigned int tce32_segsz, base, segs, avail, i;
|
||||
int64_t rc;
|
||||
void *addr;
|
||||
|
||||
/* XXX FIXME: Handle 64-bit only DMA devices */
|
||||
/* XXX FIXME: Provide 64-bit DMA facilities & non-4K TCE tables etc.. */
|
||||
/* XXX FIXME: Allocate multi-level tables on PHB3 */
|
||||
weight = pnv_pci_ioda_pe_dma_weight(pe);
|
||||
if (!weight)
|
||||
return;
|
||||
|
||||
pci_walk_bus(phb->hose->bus, pnv_pci_ioda_dev_dma_weight,
|
||||
&total_weight);
|
||||
segs = (weight * phb->ioda.dma32_count) / total_weight;
|
||||
if (!segs)
|
||||
segs = 1;
|
||||
|
||||
/*
|
||||
* Allocate contiguous DMA32 segments. We begin with the expected
|
||||
* number of segments. With one more attempt, the number of DMA32
|
||||
* segments to be allocated is decreased by one until one segment
|
||||
* is allocated successfully.
|
||||
*/
|
||||
do {
|
||||
for (base = 0; base <= phb->ioda.dma32_count - segs; base++) {
|
||||
for (avail = 0, i = base; i < base + segs; i++) {
|
||||
if (phb->ioda.dma32_segmap[i] ==
|
||||
IODA_INVALID_PE)
|
||||
avail++;
|
||||
}
|
||||
|
||||
if (avail == segs)
|
||||
goto found;
|
||||
}
|
||||
} while (--segs);
|
||||
|
||||
if (!segs) {
|
||||
pe_warn(pe, "No available DMA32 segments\n");
|
||||
return;
|
||||
}
|
||||
|
||||
found:
|
||||
tbl = pnv_pci_table_alloc(phb->hose->node);
|
||||
if (WARN_ON(!tbl))
|
||||
return;
|
||||
|
||||
#ifdef CONFIG_IOMMU_API
|
||||
pe->table_group.ops = &spapr_tce_table_group_ops;
|
||||
pe->table_group.pgsizes = SZ_4K;
|
||||
#endif
|
||||
iommu_register_group(&pe->table_group, phb->hose->global_number,
|
||||
pe->pe_number);
|
||||
pnv_pci_link_table_and_group(phb->hose->node, 0, tbl, &pe->table_group);
|
||||
|
||||
/* Grab a 32-bit TCE table */
|
||||
pe_info(pe, "DMA weight %d (%d), assigned (%d) %d DMA32 segments\n",
|
||||
weight, total_weight, base, segs);
|
||||
pe_info(pe, " Setting up 32-bit TCE table at %08x..%08x\n",
|
||||
base * PNV_IODA1_DMA32_SEGSIZE,
|
||||
(base + segs) * PNV_IODA1_DMA32_SEGSIZE - 1);
|
||||
|
||||
/* XXX Currently, we allocate one big contiguous table for the
|
||||
* TCEs. We only really need one chunk per 256M of TCE space
|
||||
* (ie per segment) but that's an optimization for later, it
|
||||
* requires some added smarts with our get/put_tce implementation
|
||||
*
|
||||
* Each TCE page is 4KB in size and each TCE entry occupies 8
|
||||
* bytes
|
||||
*/
|
||||
tce32_segsz = PNV_IODA1_DMA32_SEGSIZE >> (IOMMU_PAGE_SHIFT_4K - 3);
|
||||
tce_mem = alloc_pages_node(phb->hose->node, GFP_KERNEL,
|
||||
get_order(tce32_segsz * segs));
|
||||
if (!tce_mem) {
|
||||
pe_err(pe, " Failed to allocate a 32-bit TCE memory\n");
|
||||
goto fail;
|
||||
}
|
||||
addr = page_address(tce_mem);
|
||||
memset(addr, 0, tce32_segsz * segs);
|
||||
|
||||
/* Configure HW */
|
||||
for (i = 0; i < segs; i++) {
|
||||
rc = opal_pci_map_pe_dma_window(phb->opal_id,
|
||||
pe->pe_number,
|
||||
base + i, 1,
|
||||
__pa(addr) + tce32_segsz * i,
|
||||
tce32_segsz, IOMMU_PAGE_SIZE_4K);
|
||||
if (rc) {
|
||||
pe_err(pe, " Failed to configure 32-bit TCE table, err %lld\n",
|
||||
rc);
|
||||
goto fail;
|
||||
}
|
||||
}
|
||||
|
||||
/* Setup DMA32 segment mapping */
|
||||
for (i = base; i < base + segs; i++)
|
||||
phb->ioda.dma32_segmap[i] = pe->pe_number;
|
||||
|
||||
/* Setup linux iommu table */
|
||||
pnv_pci_setup_iommu_table(tbl, addr, tce32_segsz * segs,
|
||||
base * PNV_IODA1_DMA32_SEGSIZE,
|
||||
IOMMU_PAGE_SHIFT_4K);
|
||||
|
||||
tbl->it_ops = &pnv_ioda1_iommu_ops;
|
||||
pe->table_group.tce32_start = tbl->it_offset << tbl->it_page_shift;
|
||||
pe->table_group.tce32_size = tbl->it_size << tbl->it_page_shift;
|
||||
tbl->it_index = (phb->hose->global_number << 16) | pe->pe_number;
|
||||
if (!iommu_init_table(tbl, phb->hose->node, 0, 0))
|
||||
panic("Failed to initialize iommu table");
|
||||
|
||||
pe->dma_setup_done = true;
|
||||
return;
|
||||
fail:
|
||||
/* XXX Failure: Try to fallback to 64-bit only ? */
|
||||
if (tce_mem)
|
||||
__free_pages(tce_mem, get_order(tce32_segsz * segs));
|
||||
if (tbl) {
|
||||
pnv_pci_unlink_table_and_group(tbl, &pe->table_group);
|
||||
iommu_tce_table_put(tbl);
|
||||
}
|
||||
}
|
||||
|
||||
static long pnv_pci_ioda2_set_window(struct iommu_table_group *table_group,
|
||||
int num, struct iommu_table *tbl)
|
||||
{
|
||||
@ -2707,57 +2352,6 @@ static bool pnv_ocapi_enable_device_hook(struct pci_dev *dev)
|
||||
return true;
|
||||
}
|
||||
|
||||
static long pnv_pci_ioda1_unset_window(struct iommu_table_group *table_group,
|
||||
int num)
|
||||
{
|
||||
struct pnv_ioda_pe *pe = container_of(table_group,
|
||||
struct pnv_ioda_pe, table_group);
|
||||
struct pnv_phb *phb = pe->phb;
|
||||
unsigned int idx;
|
||||
long rc;
|
||||
|
||||
pe_info(pe, "Removing DMA window #%d\n", num);
|
||||
for (idx = 0; idx < phb->ioda.dma32_count; idx++) {
|
||||
if (phb->ioda.dma32_segmap[idx] != pe->pe_number)
|
||||
continue;
|
||||
|
||||
rc = opal_pci_map_pe_dma_window(phb->opal_id, pe->pe_number,
|
||||
idx, 0, 0ul, 0ul, 0ul);
|
||||
if (rc != OPAL_SUCCESS) {
|
||||
pe_warn(pe, "Failure %ld unmapping DMA32 segment#%d\n",
|
||||
rc, idx);
|
||||
return rc;
|
||||
}
|
||||
|
||||
phb->ioda.dma32_segmap[idx] = IODA_INVALID_PE;
|
||||
}
|
||||
|
||||
pnv_pci_unlink_table_and_group(table_group->tables[num], table_group);
|
||||
return OPAL_SUCCESS;
|
||||
}
|
||||
|
||||
static void pnv_pci_ioda1_release_pe_dma(struct pnv_ioda_pe *pe)
|
||||
{
|
||||
struct iommu_table *tbl = pe->table_group.tables[0];
|
||||
int64_t rc;
|
||||
|
||||
if (!pe->dma_setup_done)
|
||||
return;
|
||||
|
||||
rc = pnv_pci_ioda1_unset_window(&pe->table_group, 0);
|
||||
if (rc != OPAL_SUCCESS)
|
||||
return;
|
||||
|
||||
pnv_pci_p7ioc_tce_invalidate(tbl, tbl->it_offset, tbl->it_size);
|
||||
if (pe->table_group.group) {
|
||||
iommu_group_put(pe->table_group.group);
|
||||
WARN_ON(pe->table_group.group);
|
||||
}
|
||||
|
||||
free_pages(tbl->it_base, get_order(tbl->it_size << 3));
|
||||
iommu_tce_table_put(tbl);
|
||||
}
|
||||
|
||||
void pnv_pci_ioda2_release_pe_dma(struct pnv_ioda_pe *pe)
|
||||
{
|
||||
struct iommu_table *tbl = pe->table_group.tables[0];
|
||||
@ -2806,13 +2400,7 @@ static void pnv_ioda_release_pe_seg(struct pnv_ioda_pe *pe)
|
||||
{
|
||||
struct pnv_phb *phb = pe->phb;
|
||||
|
||||
if (phb->type == PNV_PHB_IODA1) {
|
||||
pnv_ioda_free_pe_seg(pe, OPAL_IO_WINDOW_TYPE,
|
||||
phb->ioda.io_segmap);
|
||||
pnv_ioda_free_pe_seg(pe, OPAL_M32_WINDOW_TYPE,
|
||||
phb->ioda.m32_segmap);
|
||||
/* M64 is pre-configured by pnv_ioda1_init_m64() */
|
||||
} else if (phb->type == PNV_PHB_IODA2) {
|
||||
if (phb->type == PNV_PHB_IODA2) {
|
||||
pnv_ioda_free_pe_seg(pe, OPAL_M32_WINDOW_TYPE,
|
||||
phb->ioda.m32_segmap);
|
||||
}
|
||||
@ -2830,9 +2418,6 @@ static void pnv_ioda_release_pe(struct pnv_ioda_pe *pe)
|
||||
mutex_unlock(&phb->ioda.pe_list_mutex);
|
||||
|
||||
switch (phb->type) {
|
||||
case PNV_PHB_IODA1:
|
||||
pnv_pci_ioda1_release_pe_dma(pe);
|
||||
break;
|
||||
case PNV_PHB_IODA2:
|
||||
pnv_pci_ioda2_release_pe_dma(pe);
|
||||
break;
|
||||
@ -2981,7 +2566,6 @@ static void __init pnv_pci_init_ioda_phb(struct device_node *np,
|
||||
struct pci_controller *hose;
|
||||
struct pnv_phb *phb;
|
||||
unsigned long size, m64map_off, m32map_off, pemap_off;
|
||||
unsigned long iomap_off = 0, dma32map_off = 0;
|
||||
struct pnv_ioda_pe *root_pe;
|
||||
struct resource r;
|
||||
const __be64 *prop64;
|
||||
@ -3092,10 +2676,6 @@ static void __init pnv_pci_init_ioda_phb(struct device_node *np,
|
||||
phb->ioda.io_segsize = phb->ioda.io_size / phb->ioda.total_pe_num;
|
||||
phb->ioda.io_pci_base = 0; /* XXX calculate this ? */
|
||||
|
||||
/* Calculate how many 32-bit TCE segments we have */
|
||||
phb->ioda.dma32_count = phb->ioda.m32_pci_base /
|
||||
PNV_IODA1_DMA32_SEGSIZE;
|
||||
|
||||
/* Allocate aux data & arrays. We don't have IO ports on PHB3 */
|
||||
size = ALIGN(max_t(unsigned, phb->ioda.total_pe_num, 8) / 8,
|
||||
sizeof(unsigned long));
|
||||
@ -3103,13 +2683,6 @@ static void __init pnv_pci_init_ioda_phb(struct device_node *np,
|
||||
size += phb->ioda.total_pe_num * sizeof(phb->ioda.m64_segmap[0]);
|
||||
m32map_off = size;
|
||||
size += phb->ioda.total_pe_num * sizeof(phb->ioda.m32_segmap[0]);
|
||||
if (phb->type == PNV_PHB_IODA1) {
|
||||
iomap_off = size;
|
||||
size += phb->ioda.total_pe_num * sizeof(phb->ioda.io_segmap[0]);
|
||||
dma32map_off = size;
|
||||
size += phb->ioda.dma32_count *
|
||||
sizeof(phb->ioda.dma32_segmap[0]);
|
||||
}
|
||||
pemap_off = size;
|
||||
size += phb->ioda.total_pe_num * sizeof(struct pnv_ioda_pe);
|
||||
aux = kzalloc(size, GFP_KERNEL);
|
||||
@ -3123,15 +2696,6 @@ static void __init pnv_pci_init_ioda_phb(struct device_node *np,
|
||||
phb->ioda.m64_segmap[segno] = IODA_INVALID_PE;
|
||||
phb->ioda.m32_segmap[segno] = IODA_INVALID_PE;
|
||||
}
|
||||
if (phb->type == PNV_PHB_IODA1) {
|
||||
phb->ioda.io_segmap = aux + iomap_off;
|
||||
for (segno = 0; segno < phb->ioda.total_pe_num; segno++)
|
||||
phb->ioda.io_segmap[segno] = IODA_INVALID_PE;
|
||||
|
||||
phb->ioda.dma32_segmap = aux + dma32map_off;
|
||||
for (segno = 0; segno < phb->ioda.dma32_count; segno++)
|
||||
phb->ioda.dma32_segmap[segno] = IODA_INVALID_PE;
|
||||
}
|
||||
phb->ioda.pe_array = aux + pemap_off;
|
||||
|
||||
/*
|
||||
@ -3155,10 +2719,6 @@ static void __init pnv_pci_init_ioda_phb(struct device_node *np,
|
||||
INIT_LIST_HEAD(&phb->ioda.pe_list);
|
||||
mutex_init(&phb->ioda.pe_list_mutex);
|
||||
|
||||
/* Calculate how many 32-bit TCE segments we have */
|
||||
phb->ioda.dma32_count = phb->ioda.m32_pci_base /
|
||||
PNV_IODA1_DMA32_SEGSIZE;
|
||||
|
||||
#if 0 /* We should really do that ... */
|
||||
rc = opal_pci_set_phb_mem_window(opal->phb_id,
|
||||
window_type,
|
||||
@ -3265,27 +2825,3 @@ static void pnv_npu2_opencapi_cfg_size_fixup(struct pci_dev *dev)
|
||||
dev->cfg_size = PCI_CFG_SPACE_EXP_SIZE;
|
||||
}
|
||||
DECLARE_PCI_FIXUP_EARLY(PCI_ANY_ID, PCI_ANY_ID, pnv_npu2_opencapi_cfg_size_fixup);
|
||||
|
||||
void __init pnv_pci_init_ioda_hub(struct device_node *np)
|
||||
{
|
||||
struct device_node *phbn;
|
||||
const __be64 *prop64;
|
||||
u64 hub_id;
|
||||
|
||||
pr_info("Probing IODA IO-Hub %pOF\n", np);
|
||||
|
||||
prop64 = of_get_property(np, "ibm,opal-hubid", NULL);
|
||||
if (!prop64) {
|
||||
pr_err(" Missing \"ibm,opal-hubid\" property !\n");
|
||||
return;
|
||||
}
|
||||
hub_id = be64_to_cpup(prop64);
|
||||
pr_devel(" HUB-ID : 0x%016llx\n", hub_id);
|
||||
|
||||
/* Count child PHBs */
|
||||
for_each_child_of_node(np, phbn) {
|
||||
/* Look for IODA1 PHBs */
|
||||
if (of_device_is_compatible(phbn, "ibm,ioda-phb"))
|
||||
pnv_pci_init_ioda_phb(phbn, hub_id, PNV_PHB_IODA1);
|
||||
}
|
||||
}
|
||||
|
@ -594,12 +594,12 @@ static void pnv_pci_sriov_disable(struct pci_dev *pdev)
|
||||
struct pnv_iov_data *iov;
|
||||
|
||||
iov = pnv_iov_get(pdev);
|
||||
num_vfs = iov->num_vfs;
|
||||
base_pe = iov->vf_pe_arr[0].pe_number;
|
||||
|
||||
if (WARN_ON(!iov))
|
||||
return;
|
||||
|
||||
num_vfs = iov->num_vfs;
|
||||
base_pe = iov->vf_pe_arr[0].pe_number;
|
||||
|
||||
/* Release VF PEs */
|
||||
pnv_ioda_release_vf_PE(pdev);
|
||||
|
||||
|
@ -845,11 +845,6 @@ void __init pnv_pci_init(void)
|
||||
pcie_ports_disabled = true;
|
||||
#endif
|
||||
|
||||
/* Look for IODA IO-Hubs. */
|
||||
for_each_compatible_node(np, NULL, "ibm,ioda-hub") {
|
||||
pnv_pci_init_ioda_hub(np);
|
||||
}
|
||||
|
||||
/* Look for ioda2 built-in PHB3's */
|
||||
for_each_compatible_node(np, NULL, "ibm,ioda2-phb")
|
||||
pnv_pci_init_ioda2_phb(np);
|
||||
|
@ -10,7 +10,6 @@
|
||||
struct pci_dn;
|
||||
|
||||
enum pnv_phb_type {
|
||||
PNV_PHB_IODA1,
|
||||
PNV_PHB_IODA2,
|
||||
PNV_PHB_NPU_OCAPI,
|
||||
};
|
||||
@ -163,10 +162,6 @@ struct pnv_phb {
|
||||
unsigned int *m32_segmap;
|
||||
unsigned int *io_segmap;
|
||||
|
||||
/* DMA32 segment maps - IODA1 only */
|
||||
unsigned int dma32_count;
|
||||
unsigned int *dma32_segmap;
|
||||
|
||||
/* IRQ chip */
|
||||
int irq_chip_init;
|
||||
struct irq_chip irq_chip;
|
||||
|
@ -1310,8 +1310,8 @@ int vas_win_close(struct vas_window *vwin)
|
||||
/* if send window, drop reference to matching receive window */
|
||||
if (window->tx_win) {
|
||||
if (window->user_win) {
|
||||
put_vas_user_win_ref(&vwin->task_ref);
|
||||
mm_context_remove_vas_window(vwin->task_ref.mm);
|
||||
put_vas_user_win_ref(&vwin->task_ref);
|
||||
}
|
||||
put_rx_win(window->rxwin);
|
||||
}
|
||||
|
@ -372,6 +372,7 @@ struct dynamic_dma_window_prop {
|
||||
struct dma_win {
|
||||
struct device_node *device;
|
||||
const struct dynamic_dma_window_prop *prop;
|
||||
bool direct;
|
||||
struct list_head list;
|
||||
};
|
||||
|
||||
@ -948,6 +949,7 @@ static struct dma_win *ddw_list_new_entry(struct device_node *pdn,
|
||||
|
||||
window->device = pdn;
|
||||
window->prop = dma64;
|
||||
window->direct = false;
|
||||
|
||||
return window;
|
||||
}
|
||||
@ -1418,6 +1420,8 @@ static bool enable_ddw(struct pci_dev *dev, struct device_node *pdn)
|
||||
goto out_del_prop;
|
||||
|
||||
if (direct_mapping) {
|
||||
window->direct = true;
|
||||
|
||||
/* DDW maps the whole partition, so enable direct DMA mapping */
|
||||
ret = walk_system_ram_range(0, memblock_end_of_DRAM() >> PAGE_SHIFT,
|
||||
win64->value, tce_setrange_multi_pSeriesLP_walk);
|
||||
@ -1434,6 +1438,8 @@ static bool enable_ddw(struct pci_dev *dev, struct device_node *pdn)
|
||||
int i;
|
||||
unsigned long start = 0, end = 0;
|
||||
|
||||
window->direct = false;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(pci->phb->mem_resources); i++) {
|
||||
const unsigned long mask = IORESOURCE_MEM_64 | IORESOURCE_MEM;
|
||||
|
||||
@ -1596,8 +1602,10 @@ static int iommu_mem_notifier(struct notifier_block *nb, unsigned long action,
|
||||
case MEM_GOING_ONLINE:
|
||||
spin_lock(&dma_win_list_lock);
|
||||
list_for_each_entry(window, &dma_win_list, list) {
|
||||
if (window->direct) {
|
||||
ret |= tce_setrange_multi_pSeriesLP(arg->start_pfn,
|
||||
arg->nr_pages, window->prop);
|
||||
}
|
||||
/* XXX log error */
|
||||
}
|
||||
spin_unlock(&dma_win_list_lock);
|
||||
@ -1606,8 +1614,10 @@ static int iommu_mem_notifier(struct notifier_block *nb, unsigned long action,
|
||||
case MEM_OFFLINE:
|
||||
spin_lock(&dma_win_list_lock);
|
||||
list_for_each_entry(window, &dma_win_list, list) {
|
||||
if (window->direct) {
|
||||
ret |= tce_clearrange_multi_pSeriesLP(arg->start_pfn,
|
||||
arg->nr_pages, window->prop);
|
||||
}
|
||||
/* XXX log error */
|
||||
}
|
||||
spin_unlock(&dma_win_list_lock);
|
||||
|
@ -507,8 +507,8 @@ static int vas_deallocate_window(struct vas_window *vwin)
|
||||
vascaps[win->win_type].nr_open_windows--;
|
||||
mutex_unlock(&vas_pseries_mutex);
|
||||
|
||||
put_vas_user_win_ref(&vwin->task_ref);
|
||||
mm_context_remove_vas_window(vwin->task_ref.mm);
|
||||
put_vas_user_win_ref(&vwin->task_ref);
|
||||
|
||||
kfree(win);
|
||||
return 0;
|
||||
|
@ -448,13 +448,11 @@ int fsl_rio_setup(struct platform_device *dev)
|
||||
struct rio_mport *port;
|
||||
struct rio_priv *priv;
|
||||
int rc = 0;
|
||||
const u32 *dt_range, *cell, *port_index;
|
||||
const u32 *port_index;
|
||||
u32 active_ports = 0;
|
||||
struct device_node *np, *rmu_node;
|
||||
int rlen;
|
||||
u32 ccsr;
|
||||
u64 range_start, range_size;
|
||||
int paw, aw, sw;
|
||||
u64 range_start;
|
||||
u32 i;
|
||||
static int tmp;
|
||||
struct device_node *rmu_np[MAX_MSG_UNIT_NUM] = {NULL};
|
||||
@ -528,15 +526,12 @@ int fsl_rio_setup(struct platform_device *dev)
|
||||
dbell->bellirq = irq_of_parse_and_map(np, 1);
|
||||
dev_info(&dev->dev, "bellirq: %d\n", dbell->bellirq);
|
||||
|
||||
aw = of_n_addr_cells(np);
|
||||
dt_range = of_get_property(np, "reg", &rlen);
|
||||
if (!dt_range) {
|
||||
if (of_property_read_reg(np, 0, &range_start, NULL)) {
|
||||
pr_err("%pOF: unable to find 'reg' property\n",
|
||||
np);
|
||||
rc = -ENOMEM;
|
||||
goto err_pw;
|
||||
}
|
||||
range_start = of_read_number(dt_range, aw);
|
||||
dbell->dbell_regs = (struct rio_dbell_regs *)(rmu_regs_win +
|
||||
(u32)range_start);
|
||||
|
||||
@ -556,19 +551,18 @@ int fsl_rio_setup(struct platform_device *dev)
|
||||
pw->dev = &dev->dev;
|
||||
pw->pwirq = irq_of_parse_and_map(np, 0);
|
||||
dev_info(&dev->dev, "pwirq: %d\n", pw->pwirq);
|
||||
aw = of_n_addr_cells(np);
|
||||
dt_range = of_get_property(np, "reg", &rlen);
|
||||
if (!dt_range) {
|
||||
if (of_property_read_reg(np, 0, &range_start, NULL)) {
|
||||
pr_err("%pOF: unable to find 'reg' property\n",
|
||||
np);
|
||||
rc = -ENOMEM;
|
||||
goto err;
|
||||
}
|
||||
range_start = of_read_number(dt_range, aw);
|
||||
pw->pw_regs = (struct rio_pw_regs *)(rmu_regs_win + (u32)range_start);
|
||||
|
||||
/*set up ports node*/
|
||||
for_each_child_of_node(dev->dev.of_node, np) {
|
||||
struct resource res;
|
||||
|
||||
port_index = of_get_property(np, "cell-index", NULL);
|
||||
if (!port_index) {
|
||||
dev_err(&dev->dev, "Can't get %pOF property 'cell-index'\n",
|
||||
@ -576,32 +570,14 @@ int fsl_rio_setup(struct platform_device *dev)
|
||||
continue;
|
||||
}
|
||||
|
||||
dt_range = of_get_property(np, "ranges", &rlen);
|
||||
if (!dt_range) {
|
||||
if (of_range_to_resource(np, 0, &res)) {
|
||||
dev_err(&dev->dev, "Can't get %pOF property 'ranges'\n",
|
||||
np);
|
||||
continue;
|
||||
}
|
||||
|
||||
/* Get node address wide */
|
||||
cell = of_get_property(np, "#address-cells", NULL);
|
||||
if (cell)
|
||||
aw = *cell;
|
||||
else
|
||||
aw = of_n_addr_cells(np);
|
||||
/* Get node size wide */
|
||||
cell = of_get_property(np, "#size-cells", NULL);
|
||||
if (cell)
|
||||
sw = *cell;
|
||||
else
|
||||
sw = of_n_size_cells(np);
|
||||
/* Get parent address wide wide */
|
||||
paw = of_n_addr_cells(np);
|
||||
range_start = of_read_number(dt_range + aw, paw);
|
||||
range_size = of_read_number(dt_range + aw + paw, sw);
|
||||
|
||||
dev_info(&dev->dev, "%pOF: LAW start 0x%016llx, size 0x%016llx.\n",
|
||||
np, range_start, range_size);
|
||||
dev_info(&dev->dev, "%pOF: LAW %pR\n",
|
||||
np, &res);
|
||||
|
||||
port = kzalloc(sizeof(struct rio_mport), GFP_KERNEL);
|
||||
if (!port)
|
||||
@ -624,9 +600,7 @@ int fsl_rio_setup(struct platform_device *dev)
|
||||
}
|
||||
|
||||
INIT_LIST_HEAD(&port->dbells);
|
||||
port->iores.start = range_start;
|
||||
port->iores.end = port->iores.start + range_size - 1;
|
||||
port->iores.flags = IORESOURCE_MEM;
|
||||
port->iores = res; /* struct copy */
|
||||
port->iores.name = "rio_io_win";
|
||||
|
||||
if (request_resource(&iomem_resource, &port->iores) < 0) {
|
||||
|
@ -23,6 +23,7 @@
|
||||
#include <linux/types.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/slab.h>
|
||||
@ -1067,9 +1068,6 @@ int fsl_rio_setup_rmu(struct rio_mport *mport, struct device_node *node)
|
||||
struct rio_priv *priv;
|
||||
struct fsl_rmu *rmu;
|
||||
u64 msg_start;
|
||||
const u32 *msg_addr;
|
||||
int mlen;
|
||||
int aw;
|
||||
|
||||
if (!mport || !mport->priv)
|
||||
return -EINVAL;
|
||||
@ -1086,16 +1084,12 @@ int fsl_rio_setup_rmu(struct rio_mport *mport, struct device_node *node)
|
||||
if (!rmu)
|
||||
return -ENOMEM;
|
||||
|
||||
aw = of_n_addr_cells(node);
|
||||
msg_addr = of_get_property(node, "reg", &mlen);
|
||||
if (!msg_addr) {
|
||||
if (of_property_read_reg(node, 0, &msg_start, NULL)) {
|
||||
pr_err("%pOF: unable to find 'reg' property of message-unit\n",
|
||||
node);
|
||||
kfree(rmu);
|
||||
return -ENOMEM;
|
||||
}
|
||||
msg_start = of_read_number(msg_addr, aw);
|
||||
|
||||
rmu->msg_regs = (struct rio_msg_regs *)
|
||||
(rmu_regs_win + (u32)msg_start);
|
||||
|
||||
|
@ -51,18 +51,10 @@ phys_addr_t get_immrbase(void)
|
||||
|
||||
soc = of_find_node_by_type(NULL, "soc");
|
||||
if (soc) {
|
||||
int size;
|
||||
u32 naddr;
|
||||
const __be32 *prop = of_get_property(soc, "#address-cells", &size);
|
||||
struct resource res;
|
||||
|
||||
if (prop && size == 4)
|
||||
naddr = be32_to_cpup(prop);
|
||||
else
|
||||
naddr = 2;
|
||||
|
||||
prop = of_get_property(soc, "ranges", &size);
|
||||
if (prop)
|
||||
immrbase = of_translate_address(soc, prop + naddr);
|
||||
if (!of_range_to_resource(soc, 0, &res))
|
||||
immrbase = res.start;
|
||||
|
||||
of_node_put(soc);
|
||||
}
|
||||
|
@ -7,20 +7,21 @@ set -o pipefail
|
||||
# To debug, uncomment the following line
|
||||
# set -x
|
||||
|
||||
# -mprofile-kernel is only supported on 64le, so this should not be invoked
|
||||
# for other targets. Therefore we can pass in -m64 and -mlittle-endian
|
||||
# explicitly, to take care of toolchains defaulting to other targets.
|
||||
# -mprofile-kernel is only supported on 64-bit, so this should not be invoked
|
||||
# for 32-bit. We pass in -m64 explicitly, and -mbig-endian and -mlittle-endian
|
||||
# are passed in from Kconfig, which takes care of toolchains defaulting to
|
||||
# other targets.
|
||||
|
||||
# Test whether the compile option -mprofile-kernel exists and generates
|
||||
# profiling code (ie. a call to _mcount()).
|
||||
echo "int func() { return 0; }" | \
|
||||
$* -m64 -mlittle-endian -S -x c -O2 -p -mprofile-kernel - -o - \
|
||||
$* -m64 -S -x c -O2 -p -mprofile-kernel - -o - \
|
||||
2> /dev/null | grep -q "_mcount"
|
||||
|
||||
# Test whether the notrace attribute correctly suppresses calls to _mcount().
|
||||
|
||||
echo -e "#include <linux/compiler.h>\nnotrace int func() { return 0; }" | \
|
||||
$* -m64 -mlittle-endian -S -x c -O2 -p -mprofile-kernel - -o - \
|
||||
$* -m64 -S -x c -O2 -p -mprofile-kernel - -o - \
|
||||
2> /dev/null | grep -q "_mcount" && \
|
||||
exit 1
|
||||
|
||||
|
@ -9,5 +9,3 @@ lib-y += memcopy.o memset.o checksum.o \
|
||||
usercopy.o strnlen_user.o
|
||||
lib-$(CONFIG_ARCH_HAS_STRNCPY_FROM_USER) += strncpy_user.o
|
||||
lib-$(CONFIG_PCI) += pci-auto.o
|
||||
lib-$(CONFIG_KCSAN) += kcsan-stubs.o
|
||||
KCSAN_SANITIZE_kcsan-stubs.o := n
|
||||
|
@ -1,54 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
|
||||
#include <linux/bug.h>
|
||||
#include <linux/types.h>
|
||||
|
||||
void __atomic_store_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_load_8(const volatile void *p, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_exchange_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
bool __atomic_compare_exchange_8(volatile void *p1, void *p2, u64 v, bool b, int i1, int i2)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_add_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_sub_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_and_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_or_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_xor_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
||||
|
||||
u64 __atomic_fetch_nand_8(volatile void *p, u64 v, int i)
|
||||
{
|
||||
BUG();
|
||||
}
|
@ -69,7 +69,7 @@ static struct i2c_driver ams_i2c_driver = {
|
||||
.driver = {
|
||||
.name = "ams",
|
||||
},
|
||||
.probe_new = ams_i2c_probe,
|
||||
.probe = ams_i2c_probe,
|
||||
.remove = ams_i2c_remove,
|
||||
.id_table = ams_id,
|
||||
};
|
||||
|
@ -33,7 +33,8 @@
|
||||
#include <linux/delay.h>
|
||||
#include <linux/poll.h>
|
||||
#include <linux/mutex.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_address.h>
|
||||
#include <linux/of_irq.h>
|
||||
#include <linux/of_platform.h>
|
||||
#include <linux/slab.h>
|
||||
@ -470,7 +471,7 @@ EXPORT_SYMBOL(smu_present);
|
||||
int __init smu_init (void)
|
||||
{
|
||||
struct device_node *np;
|
||||
const u32 *data;
|
||||
u64 data;
|
||||
int ret = 0;
|
||||
|
||||
np = of_find_node_by_type(NULL, "smu");
|
||||
@ -514,8 +515,7 @@ int __init smu_init (void)
|
||||
ret = -ENXIO;
|
||||
goto fail_bootmem;
|
||||
}
|
||||
data = of_get_property(smu->db_node, "reg", NULL);
|
||||
if (data == NULL) {
|
||||
if (of_property_read_reg(smu->db_node, 0, &data, NULL)) {
|
||||
printk(KERN_ERR "SMU: Can't find doorbell GPIO address !\n");
|
||||
ret = -ENXIO;
|
||||
goto fail_db_node;
|
||||
@ -525,7 +525,7 @@ int __init smu_init (void)
|
||||
* and ack. GPIOs are at 0x50, best would be to find that out
|
||||
* in the device-tree though.
|
||||
*/
|
||||
smu->doorbell = *data;
|
||||
smu->doorbell = data;
|
||||
if (smu->doorbell < 0x50)
|
||||
smu->doorbell += 0x50;
|
||||
|
||||
@ -534,13 +534,12 @@ int __init smu_init (void)
|
||||
smu->msg_node = of_find_node_by_name(NULL, "smu-interrupt");
|
||||
if (smu->msg_node == NULL)
|
||||
break;
|
||||
data = of_get_property(smu->msg_node, "reg", NULL);
|
||||
if (data == NULL) {
|
||||
if (of_property_read_reg(smu->msg_node, 0, &data, NULL)) {
|
||||
of_node_put(smu->msg_node);
|
||||
smu->msg_node = NULL;
|
||||
break;
|
||||
}
|
||||
smu->msg = *data;
|
||||
smu->msg = data;
|
||||
if (smu->msg < 0x50)
|
||||
smu->msg += 0x50;
|
||||
} while(0);
|
||||
|
@ -598,7 +598,7 @@ static struct i2c_driver thermostat_driver = {
|
||||
.driver = {
|
||||
.name = "therm_adt746x",
|
||||
},
|
||||
.probe_new = probe_thermostat,
|
||||
.probe = probe_thermostat,
|
||||
.remove = remove_thermostat,
|
||||
.id_table = therm_adt746x_id,
|
||||
};
|
||||
|
@ -442,7 +442,7 @@ static struct i2c_driver g4fan_driver = {
|
||||
.driver = {
|
||||
.name = "therm_windtunnel",
|
||||
},
|
||||
.probe_new = do_probe,
|
||||
.probe = do_probe,
|
||||
.remove = do_remove,
|
||||
.id_table = therm_windtunnel_id,
|
||||
};
|
||||
|
@ -235,8 +235,7 @@ int __init find_via_cuda(void)
|
||||
int __init find_via_cuda(void)
|
||||
{
|
||||
struct adb_request req;
|
||||
phys_addr_t taddr;
|
||||
const u32 *reg;
|
||||
struct resource res;
|
||||
int err;
|
||||
|
||||
if (vias)
|
||||
@ -245,17 +244,12 @@ int __init find_via_cuda(void)
|
||||
if (!vias)
|
||||
return 0;
|
||||
|
||||
reg = of_get_property(vias, "reg", NULL);
|
||||
if (reg == NULL) {
|
||||
printk(KERN_ERR "via-cuda: No \"reg\" property !\n");
|
||||
err = of_address_to_resource(vias, 0, &res);
|
||||
if (err) {
|
||||
printk(KERN_ERR "via-cuda: Error getting \"reg\" property !\n");
|
||||
goto fail;
|
||||
}
|
||||
taddr = of_translate_address(vias, reg);
|
||||
if (taddr == 0) {
|
||||
printk(KERN_ERR "via-cuda: Can't translate address !\n");
|
||||
goto fail;
|
||||
}
|
||||
via = ioremap(taddr, 0x2000);
|
||||
via = ioremap(res.start, 0x2000);
|
||||
if (via == NULL) {
|
||||
printk(KERN_ERR "via-cuda: Can't map address !\n");
|
||||
goto fail;
|
||||
|
@ -286,8 +286,9 @@ static char *pbook_type[] = {
|
||||
int __init find_via_pmu(void)
|
||||
{
|
||||
#ifdef CONFIG_PPC_PMAC
|
||||
int err;
|
||||
u64 taddr;
|
||||
const u32 *reg;
|
||||
struct resource res;
|
||||
|
||||
if (pmu_state != uninitialized)
|
||||
return 1;
|
||||
@ -295,16 +296,12 @@ int __init find_via_pmu(void)
|
||||
if (vias == NULL)
|
||||
return 0;
|
||||
|
||||
reg = of_get_property(vias, "reg", NULL);
|
||||
if (reg == NULL) {
|
||||
printk(KERN_ERR "via-pmu: No \"reg\" property !\n");
|
||||
goto fail;
|
||||
}
|
||||
taddr = of_translate_address(vias, reg);
|
||||
if (taddr == OF_BAD_ADDR) {
|
||||
printk(KERN_ERR "via-pmu: Can't translate address !\n");
|
||||
err = of_address_to_resource(vias, 0, &res);
|
||||
if (err) {
|
||||
printk(KERN_ERR "via-pmu: Error getting \"reg\" property !\n");
|
||||
goto fail;
|
||||
}
|
||||
taddr = res.start;
|
||||
|
||||
pmu_has_adb = 1;
|
||||
|
||||
@ -324,7 +321,6 @@ int __init find_via_pmu(void)
|
||||
|| of_device_is_compatible(vias->parent, "K2-Keylargo")) {
|
||||
struct device_node *gpiop;
|
||||
struct device_node *adbp;
|
||||
u64 gaddr = OF_BAD_ADDR;
|
||||
|
||||
pmu_kind = PMU_KEYLARGO_BASED;
|
||||
adbp = of_find_node_by_type(NULL, "adb");
|
||||
@ -338,11 +334,8 @@ int __init find_via_pmu(void)
|
||||
|
||||
gpiop = of_find_node_by_name(NULL, "gpio");
|
||||
if (gpiop) {
|
||||
reg = of_get_property(gpiop, "reg", NULL);
|
||||
if (reg)
|
||||
gaddr = of_translate_address(gpiop, reg);
|
||||
if (gaddr != OF_BAD_ADDR)
|
||||
gpio_reg = ioremap(gaddr, 0x10);
|
||||
if (!of_address_to_resource(gpiop, 0, &res))
|
||||
gpio_reg = ioremap(res.start, 0x10);
|
||||
of_node_put(gpiop);
|
||||
}
|
||||
if (gpio_reg == NULL) {
|
||||
|
@ -320,7 +320,7 @@ static struct i2c_driver wf_ad7417_driver = {
|
||||
.name = "wf_ad7417",
|
||||
.of_match_table = wf_ad7417_of_id,
|
||||
},
|
||||
.probe_new = wf_ad7417_probe,
|
||||
.probe = wf_ad7417_probe,
|
||||
.remove = wf_ad7417_remove,
|
||||
.id_table = wf_ad7417_id,
|
||||
};
|
||||
|
@ -589,7 +589,7 @@ static struct i2c_driver wf_fcu_driver = {
|
||||
.name = "wf_fcu",
|
||||
.of_match_table = wf_fcu_of_id,
|
||||
},
|
||||
.probe_new = wf_fcu_probe,
|
||||
.probe = wf_fcu_probe,
|
||||
.remove = wf_fcu_remove,
|
||||
.id_table = wf_fcu_id,
|
||||
};
|
||||
|
@ -177,7 +177,7 @@ static struct i2c_driver wf_lm75_driver = {
|
||||
.name = "wf_lm75",
|
||||
.of_match_table = wf_lm75_of_id,
|
||||
},
|
||||
.probe_new = wf_lm75_probe,
|
||||
.probe = wf_lm75_probe,
|
||||
.remove = wf_lm75_remove,
|
||||
.id_table = wf_lm75_id,
|
||||
};
|
||||
|
@ -172,7 +172,7 @@ static struct i2c_driver wf_lm87_driver = {
|
||||
.name = "wf_lm87",
|
||||
.of_match_table = wf_lm87_of_id,
|
||||
},
|
||||
.probe_new = wf_lm87_probe,
|
||||
.probe = wf_lm87_probe,
|
||||
.remove = wf_lm87_remove,
|
||||
.id_table = wf_lm87_id,
|
||||
};
|
||||
|
@ -128,7 +128,7 @@ static struct i2c_driver wf_max6690_driver = {
|
||||
.name = "wf_max6690",
|
||||
.of_match_table = wf_max6690_of_id,
|
||||
},
|
||||
.probe_new = wf_max6690_probe,
|
||||
.probe = wf_max6690_probe,
|
||||
.remove = wf_max6690_remove,
|
||||
.id_table = wf_max6690_id,
|
||||
};
|
||||
|
@ -349,7 +349,7 @@ static struct i2c_driver wf_sat_driver = {
|
||||
.name = "wf_smu_sat",
|
||||
.of_match_table = wf_sat_of_id,
|
||||
},
|
||||
.probe_new = wf_sat_probe,
|
||||
.probe = wf_sat_probe,
|
||||
.remove = wf_sat_remove,
|
||||
.id_table = wf_sat_id,
|
||||
};
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user