Linux 3.6-rc6
-----BEGIN PGP SIGNATURE----- Version: GnuPG v2.0.18 (GNU/Linux) iQEcBAABAgAGBQJQVkutAAoJEHm+PkMAQRiGW8sH/36FVQ3zI75QH16AmR++2nMZ BRJGoxcRFMssrXTYVdkMyzygf8b7MZbNEn1qt2g63MNzGaJucPlw5NVL4GLzR+zr x/EglLrTEPCD5el9wJ3ls9iC1soudKQTvC2BjcdUjpoSwHrDM/7GKfbOacE54Wqc C1VHCcg5DWOD7F0RnYT2SQEVCeDODNmcyFdk7Oi4cUicTPJoYWJ9O9MGfBDBok0N M+dXxa9nvsl7EeEKpBKH9vo4TfXn3Gsj6LCRdedvI15ilZjfo8jdHYbSn7KBfQuZ JIKRnqkaQ1JfMFt+M/JJZ1b/+Wrd4HLMmmn5oUmrGGIvhpi32nJfi/97+nSy8iU= =c5gW -----END PGP SIGNATURE----- Merge tag 'v3.6-rc6' into for-3.7 Linux 3.6-rc6 has all our bug fixes. Conflicts (trivial overlap): sound/soc/omap/am3517evm.c
This commit is contained in:
commit
ddfb43f388
|
@ -210,3 +210,15 @@ Users:
|
||||||
firmware assigned instance number of the PCI
|
firmware assigned instance number of the PCI
|
||||||
device that can help in understanding the firmware
|
device that can help in understanding the firmware
|
||||||
intended order of the PCI device.
|
intended order of the PCI device.
|
||||||
|
|
||||||
|
What: /sys/bus/pci/devices/.../d3cold_allowed
|
||||||
|
Date: July 2012
|
||||||
|
Contact: Huang Ying <ying.huang@intel.com>
|
||||||
|
Description:
|
||||||
|
d3cold_allowed is bit to control whether the corresponding PCI
|
||||||
|
device can be put into D3Cold state. If it is cleared, the
|
||||||
|
device will never be put into D3Cold state. If it is set, the
|
||||||
|
device may be put into D3Cold state if other requirements are
|
||||||
|
satisfied too. Reading this attribute will show the current
|
||||||
|
value of d3cold_allowed bit. Writing this attribute will set
|
||||||
|
the value of d3cold_allowed bit.
|
||||||
|
|
|
@ -579,7 +579,7 @@ Why: KVM tracepoints provide mostly equivalent information in a much more
|
||||||
----------------------------
|
----------------------------
|
||||||
|
|
||||||
What: at91-mci driver ("CONFIG_MMC_AT91")
|
What: at91-mci driver ("CONFIG_MMC_AT91")
|
||||||
When: 3.7
|
When: 3.8
|
||||||
Why: There are two mci drivers: at91-mci and atmel-mci. The PDC support
|
Why: There are two mci drivers: at91-mci and atmel-mci. The PDC support
|
||||||
was added to atmel-mci as a first step to support more chips.
|
was added to atmel-mci as a first step to support more chips.
|
||||||
Then at91-mci was kept only for old IP versions (on at91rm9200 and
|
Then at91-mci was kept only for old IP versions (on at91rm9200 and
|
||||||
|
|
|
@ -21,6 +21,7 @@ Supported adapters:
|
||||||
* Intel DH89xxCC (PCH)
|
* Intel DH89xxCC (PCH)
|
||||||
* Intel Panther Point (PCH)
|
* Intel Panther Point (PCH)
|
||||||
* Intel Lynx Point (PCH)
|
* Intel Lynx Point (PCH)
|
||||||
|
* Intel Lynx Point-LP (PCH)
|
||||||
Datasheets: Publicly available at the Intel website
|
Datasheets: Publicly available at the Intel website
|
||||||
|
|
||||||
On Intel Patsburg and later chipsets, both the normal host SMBus controller
|
On Intel Patsburg and later chipsets, both the normal host SMBus controller
|
||||||
|
|
|
@ -3388,7 +3388,7 @@ M: "Wolfram Sang (embedded platforms)" <w.sang@pengutronix.de>
|
||||||
L: linux-i2c@vger.kernel.org
|
L: linux-i2c@vger.kernel.org
|
||||||
W: http://i2c.wiki.kernel.org/
|
W: http://i2c.wiki.kernel.org/
|
||||||
T: quilt kernel.org/pub/linux/kernel/people/jdelvare/linux-2.6/jdelvare-i2c/
|
T: quilt kernel.org/pub/linux/kernel/people/jdelvare/linux-2.6/jdelvare-i2c/
|
||||||
T: git git://git.fluff.org/bjdooks/linux.git
|
T: git git://git.pengutronix.de/git/wsa/linux.git
|
||||||
S: Maintained
|
S: Maintained
|
||||||
F: Documentation/i2c/
|
F: Documentation/i2c/
|
||||||
F: drivers/i2c/
|
F: drivers/i2c/
|
||||||
|
|
2
Makefile
2
Makefile
|
@ -1,7 +1,7 @@
|
||||||
VERSION = 3
|
VERSION = 3
|
||||||
PATCHLEVEL = 6
|
PATCHLEVEL = 6
|
||||||
SUBLEVEL = 0
|
SUBLEVEL = 0
|
||||||
EXTRAVERSION = -rc4
|
EXTRAVERSION = -rc6
|
||||||
NAME = Saber-toothed Squirrel
|
NAME = Saber-toothed Squirrel
|
||||||
|
|
||||||
# *DOCUMENTATION*
|
# *DOCUMENTATION*
|
||||||
|
|
|
@ -6,7 +6,7 @@ config ARM
|
||||||
select HAVE_DMA_API_DEBUG
|
select HAVE_DMA_API_DEBUG
|
||||||
select HAVE_IDE if PCI || ISA || PCMCIA
|
select HAVE_IDE if PCI || ISA || PCMCIA
|
||||||
select HAVE_DMA_ATTRS
|
select HAVE_DMA_ATTRS
|
||||||
select HAVE_DMA_CONTIGUOUS if (CPU_V6 || CPU_V6K || CPU_V7)
|
select HAVE_DMA_CONTIGUOUS if MMU
|
||||||
select HAVE_MEMBLOCK
|
select HAVE_MEMBLOCK
|
||||||
select RTC_LIB
|
select RTC_LIB
|
||||||
select SYS_SUPPORTS_APM_EMULATION
|
select SYS_SUPPORTS_APM_EMULATION
|
||||||
|
|
|
@ -356,15 +356,15 @@ choice
|
||||||
is nothing connected to read from the DCC.
|
is nothing connected to read from the DCC.
|
||||||
|
|
||||||
config DEBUG_SEMIHOSTING
|
config DEBUG_SEMIHOSTING
|
||||||
bool "Kernel low-level debug output via semihosting I"
|
bool "Kernel low-level debug output via semihosting I/O"
|
||||||
help
|
help
|
||||||
Semihosting enables code running on an ARM target to use
|
Semihosting enables code running on an ARM target to use
|
||||||
the I/O facilities on a host debugger/emulator through a
|
the I/O facilities on a host debugger/emulator through a
|
||||||
simple SVC calls. The host debugger or emulator must have
|
simple SVC call. The host debugger or emulator must have
|
||||||
semihosting enabled for the special svc call to be trapped
|
semihosting enabled for the special svc call to be trapped
|
||||||
otherwise the kernel will crash.
|
otherwise the kernel will crash.
|
||||||
|
|
||||||
This is known to work with OpenOCD, as wellas
|
This is known to work with OpenOCD, as well as
|
||||||
ARM's Fast Models, or any other controlling environment
|
ARM's Fast Models, or any other controlling environment
|
||||||
that implements semihosting.
|
that implements semihosting.
|
||||||
|
|
||||||
|
|
|
@ -284,10 +284,10 @@ zImage Image xipImage bootpImage uImage: vmlinux
|
||||||
zinstall uinstall install: vmlinux
|
zinstall uinstall install: vmlinux
|
||||||
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $@
|
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $@
|
||||||
|
|
||||||
%.dtb:
|
%.dtb: scripts
|
||||||
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $(boot)/$@
|
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $(boot)/$@
|
||||||
|
|
||||||
dtbs:
|
dtbs: scripts
|
||||||
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $(boot)/$@
|
$(Q)$(MAKE) $(build)=$(boot) MACHINE=$(MACHINE) $(boot)/$@
|
||||||
|
|
||||||
# We use MRPROPER_FILES and CLEAN_FILES now
|
# We use MRPROPER_FILES and CLEAN_FILES now
|
||||||
|
|
|
@ -659,10 +659,14 @@ __armv7_mmu_cache_on:
|
||||||
#ifdef CONFIG_CPU_ENDIAN_BE8
|
#ifdef CONFIG_CPU_ENDIAN_BE8
|
||||||
orr r0, r0, #1 << 25 @ big-endian page tables
|
orr r0, r0, #1 << 25 @ big-endian page tables
|
||||||
#endif
|
#endif
|
||||||
|
mrcne p15, 0, r6, c2, c0, 2 @ read ttb control reg
|
||||||
orrne r0, r0, #1 @ MMU enabled
|
orrne r0, r0, #1 @ MMU enabled
|
||||||
movne r1, #0xfffffffd @ domain 0 = client
|
movne r1, #0xfffffffd @ domain 0 = client
|
||||||
|
bic r6, r6, #1 << 31 @ 32-bit translation system
|
||||||
|
bic r6, r6, #3 << 0 @ use only ttbr0
|
||||||
mcrne p15, 0, r3, c2, c0, 0 @ load page table pointer
|
mcrne p15, 0, r3, c2, c0, 0 @ load page table pointer
|
||||||
mcrne p15, 0, r1, c3, c0, 0 @ load domain access control
|
mcrne p15, 0, r1, c3, c0, 0 @ load domain access control
|
||||||
|
mcrne p15, 0, r6, c2, c0, 2 @ load ttb control
|
||||||
#endif
|
#endif
|
||||||
mcr p15, 0, r0, c7, c5, 4 @ ISB
|
mcr p15, 0, r0, c7, c5, 4 @ ISB
|
||||||
mcr p15, 0, r0, c1, c0, 0 @ load control register
|
mcr p15, 0, r0, c1, c0, 0 @ load control register
|
||||||
|
|
|
@ -15,7 +15,7 @@
|
||||||
compatible = "atmel,at91sam9g25ek", "atmel,at91sam9x5ek", "atmel,at91sam9x5", "atmel,at91sam9";
|
compatible = "atmel,at91sam9g25ek", "atmel,at91sam9x5ek", "atmel,at91sam9x5", "atmel,at91sam9";
|
||||||
|
|
||||||
chosen {
|
chosen {
|
||||||
bootargs = "128M console=ttyS0,115200 root=/dev/mtdblock1 rw rootfstype=ubifs ubi.mtd=1 root=ubi0:rootfs";
|
bootargs = "console=ttyS0,115200 root=/dev/mtdblock1 rw rootfstype=ubifs ubi.mtd=1 root=ubi0:rootfs";
|
||||||
};
|
};
|
||||||
|
|
||||||
ahb {
|
ahb {
|
||||||
|
|
|
@ -33,7 +33,7 @@ CONFIG_AEABI=y
|
||||||
CONFIG_FORCE_MAX_ZONEORDER=13
|
CONFIG_FORCE_MAX_ZONEORDER=13
|
||||||
CONFIG_ZBOOT_ROM_TEXT=0x0
|
CONFIG_ZBOOT_ROM_TEXT=0x0
|
||||||
CONFIG_ZBOOT_ROM_BSS=0x0
|
CONFIG_ZBOOT_ROM_BSS=0x0
|
||||||
CONFIG_CMDLINE="console=tty0 console=ttySC1,115200 earlyprintk=sh-sci.1,115200 ignore_loglevel root=/dev/nfs ip=dhcp nfsroot=,rsize=4096,wsize=4096"
|
CONFIG_CMDLINE="console=tty0 console=ttySC1,115200 earlyprintk=sh-sci.1,115200 ignore_loglevel root=/dev/nfs ip=dhcp nfsroot=,rsize=4096,wsize=4096 rw"
|
||||||
CONFIG_CMDLINE_FORCE=y
|
CONFIG_CMDLINE_FORCE=y
|
||||||
CONFIG_KEXEC=y
|
CONFIG_KEXEC=y
|
||||||
CONFIG_VFP=y
|
CONFIG_VFP=y
|
||||||
|
|
|
@ -320,4 +320,12 @@
|
||||||
.size \name , . - \name
|
.size \name , . - \name
|
||||||
.endm
|
.endm
|
||||||
|
|
||||||
|
.macro check_uaccess, addr:req, size:req, limit:req, tmp:req, bad:req
|
||||||
|
#ifndef CONFIG_CPU_USE_DOMAINS
|
||||||
|
adds \tmp, \addr, #\size - 1
|
||||||
|
sbcccs \tmp, \tmp, \limit
|
||||||
|
bcs \bad
|
||||||
|
#endif
|
||||||
|
.endm
|
||||||
|
|
||||||
#endif /* __ASM_ASSEMBLER_H__ */
|
#endif /* __ASM_ASSEMBLER_H__ */
|
||||||
|
|
|
@ -202,6 +202,13 @@ static inline void dma_free_writecombine(struct device *dev, size_t size,
|
||||||
return dma_free_attrs(dev, size, cpu_addr, dma_handle, &attrs);
|
return dma_free_attrs(dev, size, cpu_addr, dma_handle, &attrs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This can be called during early boot to increase the size of the atomic
|
||||||
|
* coherent DMA pool above the default value of 256KiB. It must be called
|
||||||
|
* before postcore_initcall.
|
||||||
|
*/
|
||||||
|
extern void __init init_dma_coherent_pool_size(unsigned long size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This can be called during boot to increase the size of the consistent
|
* This can be called during boot to increase the size of the consistent
|
||||||
* DMA region above it's default value of 2MB. It must be called before the
|
* DMA region above it's default value of 2MB. It must be called before the
|
||||||
|
|
|
@ -187,6 +187,7 @@ static inline unsigned long __phys_to_virt(unsigned long x)
|
||||||
#define __phys_to_virt(x) ((x) - PHYS_OFFSET + PAGE_OFFSET)
|
#define __phys_to_virt(x) ((x) - PHYS_OFFSET + PAGE_OFFSET)
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
#endif /* __ASSEMBLY__ */
|
||||||
|
|
||||||
#ifndef PHYS_OFFSET
|
#ifndef PHYS_OFFSET
|
||||||
#ifdef PLAT_PHYS_OFFSET
|
#ifdef PLAT_PHYS_OFFSET
|
||||||
|
@ -196,6 +197,8 @@ static inline unsigned long __phys_to_virt(unsigned long x)
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef __ASSEMBLY__
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* PFNs are used to describe any physical page; this means
|
* PFNs are used to describe any physical page; this means
|
||||||
* PFN 0 == physical address 0.
|
* PFN 0 == physical address 0.
|
||||||
|
|
|
@ -199,6 +199,9 @@ static inline void __pte_free_tlb(struct mmu_gather *tlb, pgtable_t pte,
|
||||||
{
|
{
|
||||||
pgtable_page_dtor(pte);
|
pgtable_page_dtor(pte);
|
||||||
|
|
||||||
|
#ifdef CONFIG_ARM_LPAE
|
||||||
|
tlb_add_flush(tlb, addr);
|
||||||
|
#else
|
||||||
/*
|
/*
|
||||||
* With the classic ARM MMU, a pte page has two corresponding pmd
|
* With the classic ARM MMU, a pte page has two corresponding pmd
|
||||||
* entries, each covering 1MB.
|
* entries, each covering 1MB.
|
||||||
|
@ -206,6 +209,7 @@ static inline void __pte_free_tlb(struct mmu_gather *tlb, pgtable_t pte,
|
||||||
addr &= PMD_MASK;
|
addr &= PMD_MASK;
|
||||||
tlb_add_flush(tlb, addr + SZ_1M - PAGE_SIZE);
|
tlb_add_flush(tlb, addr + SZ_1M - PAGE_SIZE);
|
||||||
tlb_add_flush(tlb, addr + SZ_1M);
|
tlb_add_flush(tlb, addr + SZ_1M);
|
||||||
|
#endif
|
||||||
|
|
||||||
tlb_remove_page(tlb, pte);
|
tlb_remove_page(tlb, pte);
|
||||||
}
|
}
|
||||||
|
|
|
@ -101,28 +101,39 @@ extern int __get_user_1(void *);
|
||||||
extern int __get_user_2(void *);
|
extern int __get_user_2(void *);
|
||||||
extern int __get_user_4(void *);
|
extern int __get_user_4(void *);
|
||||||
|
|
||||||
#define __get_user_x(__r2,__p,__e,__s,__i...) \
|
#define __GUP_CLOBBER_1 "lr", "cc"
|
||||||
|
#ifdef CONFIG_CPU_USE_DOMAINS
|
||||||
|
#define __GUP_CLOBBER_2 "ip", "lr", "cc"
|
||||||
|
#else
|
||||||
|
#define __GUP_CLOBBER_2 "lr", "cc"
|
||||||
|
#endif
|
||||||
|
#define __GUP_CLOBBER_4 "lr", "cc"
|
||||||
|
|
||||||
|
#define __get_user_x(__r2,__p,__e,__l,__s) \
|
||||||
__asm__ __volatile__ ( \
|
__asm__ __volatile__ ( \
|
||||||
__asmeq("%0", "r0") __asmeq("%1", "r2") \
|
__asmeq("%0", "r0") __asmeq("%1", "r2") \
|
||||||
|
__asmeq("%3", "r1") \
|
||||||
"bl __get_user_" #__s \
|
"bl __get_user_" #__s \
|
||||||
: "=&r" (__e), "=r" (__r2) \
|
: "=&r" (__e), "=r" (__r2) \
|
||||||
: "0" (__p) \
|
: "0" (__p), "r" (__l) \
|
||||||
: __i, "cc")
|
: __GUP_CLOBBER_##__s)
|
||||||
|
|
||||||
#define get_user(x,p) \
|
#define __get_user_check(x,p) \
|
||||||
({ \
|
({ \
|
||||||
|
unsigned long __limit = current_thread_info()->addr_limit - 1; \
|
||||||
register const typeof(*(p)) __user *__p asm("r0") = (p);\
|
register const typeof(*(p)) __user *__p asm("r0") = (p);\
|
||||||
register unsigned long __r2 asm("r2"); \
|
register unsigned long __r2 asm("r2"); \
|
||||||
|
register unsigned long __l asm("r1") = __limit; \
|
||||||
register int __e asm("r0"); \
|
register int __e asm("r0"); \
|
||||||
switch (sizeof(*(__p))) { \
|
switch (sizeof(*(__p))) { \
|
||||||
case 1: \
|
case 1: \
|
||||||
__get_user_x(__r2, __p, __e, 1, "lr"); \
|
__get_user_x(__r2, __p, __e, __l, 1); \
|
||||||
break; \
|
break; \
|
||||||
case 2: \
|
case 2: \
|
||||||
__get_user_x(__r2, __p, __e, 2, "r3", "lr"); \
|
__get_user_x(__r2, __p, __e, __l, 2); \
|
||||||
break; \
|
break; \
|
||||||
case 4: \
|
case 4: \
|
||||||
__get_user_x(__r2, __p, __e, 4, "lr"); \
|
__get_user_x(__r2, __p, __e, __l, 4); \
|
||||||
break; \
|
break; \
|
||||||
default: __e = __get_user_bad(); break; \
|
default: __e = __get_user_bad(); break; \
|
||||||
} \
|
} \
|
||||||
|
@ -130,42 +141,57 @@ extern int __get_user_4(void *);
|
||||||
__e; \
|
__e; \
|
||||||
})
|
})
|
||||||
|
|
||||||
|
#define get_user(x,p) \
|
||||||
|
({ \
|
||||||
|
might_fault(); \
|
||||||
|
__get_user_check(x,p); \
|
||||||
|
})
|
||||||
|
|
||||||
extern int __put_user_1(void *, unsigned int);
|
extern int __put_user_1(void *, unsigned int);
|
||||||
extern int __put_user_2(void *, unsigned int);
|
extern int __put_user_2(void *, unsigned int);
|
||||||
extern int __put_user_4(void *, unsigned int);
|
extern int __put_user_4(void *, unsigned int);
|
||||||
extern int __put_user_8(void *, unsigned long long);
|
extern int __put_user_8(void *, unsigned long long);
|
||||||
|
|
||||||
#define __put_user_x(__r2,__p,__e,__s) \
|
#define __put_user_x(__r2,__p,__e,__l,__s) \
|
||||||
__asm__ __volatile__ ( \
|
__asm__ __volatile__ ( \
|
||||||
__asmeq("%0", "r0") __asmeq("%2", "r2") \
|
__asmeq("%0", "r0") __asmeq("%2", "r2") \
|
||||||
|
__asmeq("%3", "r1") \
|
||||||
"bl __put_user_" #__s \
|
"bl __put_user_" #__s \
|
||||||
: "=&r" (__e) \
|
: "=&r" (__e) \
|
||||||
: "0" (__p), "r" (__r2) \
|
: "0" (__p), "r" (__r2), "r" (__l) \
|
||||||
: "ip", "lr", "cc")
|
: "ip", "lr", "cc")
|
||||||
|
|
||||||
#define put_user(x,p) \
|
#define __put_user_check(x,p) \
|
||||||
({ \
|
({ \
|
||||||
|
unsigned long __limit = current_thread_info()->addr_limit - 1; \
|
||||||
register const typeof(*(p)) __r2 asm("r2") = (x); \
|
register const typeof(*(p)) __r2 asm("r2") = (x); \
|
||||||
register const typeof(*(p)) __user *__p asm("r0") = (p);\
|
register const typeof(*(p)) __user *__p asm("r0") = (p);\
|
||||||
|
register unsigned long __l asm("r1") = __limit; \
|
||||||
register int __e asm("r0"); \
|
register int __e asm("r0"); \
|
||||||
switch (sizeof(*(__p))) { \
|
switch (sizeof(*(__p))) { \
|
||||||
case 1: \
|
case 1: \
|
||||||
__put_user_x(__r2, __p, __e, 1); \
|
__put_user_x(__r2, __p, __e, __l, 1); \
|
||||||
break; \
|
break; \
|
||||||
case 2: \
|
case 2: \
|
||||||
__put_user_x(__r2, __p, __e, 2); \
|
__put_user_x(__r2, __p, __e, __l, 2); \
|
||||||
break; \
|
break; \
|
||||||
case 4: \
|
case 4: \
|
||||||
__put_user_x(__r2, __p, __e, 4); \
|
__put_user_x(__r2, __p, __e, __l, 4); \
|
||||||
break; \
|
break; \
|
||||||
case 8: \
|
case 8: \
|
||||||
__put_user_x(__r2, __p, __e, 8); \
|
__put_user_x(__r2, __p, __e, __l, 8); \
|
||||||
break; \
|
break; \
|
||||||
default: __e = __put_user_bad(); break; \
|
default: __e = __put_user_bad(); break; \
|
||||||
} \
|
} \
|
||||||
__e; \
|
__e; \
|
||||||
})
|
})
|
||||||
|
|
||||||
|
#define put_user(x,p) \
|
||||||
|
({ \
|
||||||
|
might_fault(); \
|
||||||
|
__put_user_check(x,p); \
|
||||||
|
})
|
||||||
|
|
||||||
#else /* CONFIG_MMU */
|
#else /* CONFIG_MMU */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -219,6 +245,7 @@ do { \
|
||||||
unsigned long __gu_addr = (unsigned long)(ptr); \
|
unsigned long __gu_addr = (unsigned long)(ptr); \
|
||||||
unsigned long __gu_val; \
|
unsigned long __gu_val; \
|
||||||
__chk_user_ptr(ptr); \
|
__chk_user_ptr(ptr); \
|
||||||
|
might_fault(); \
|
||||||
switch (sizeof(*(ptr))) { \
|
switch (sizeof(*(ptr))) { \
|
||||||
case 1: __get_user_asm_byte(__gu_val,__gu_addr,err); break; \
|
case 1: __get_user_asm_byte(__gu_val,__gu_addr,err); break; \
|
||||||
case 2: __get_user_asm_half(__gu_val,__gu_addr,err); break; \
|
case 2: __get_user_asm_half(__gu_val,__gu_addr,err); break; \
|
||||||
|
@ -300,6 +327,7 @@ do { \
|
||||||
unsigned long __pu_addr = (unsigned long)(ptr); \
|
unsigned long __pu_addr = (unsigned long)(ptr); \
|
||||||
__typeof__(*(ptr)) __pu_val = (x); \
|
__typeof__(*(ptr)) __pu_val = (x); \
|
||||||
__chk_user_ptr(ptr); \
|
__chk_user_ptr(ptr); \
|
||||||
|
might_fault(); \
|
||||||
switch (sizeof(*(ptr))) { \
|
switch (sizeof(*(ptr))) { \
|
||||||
case 1: __put_user_asm_byte(__pu_val,__pu_addr,err); break; \
|
case 1: __put_user_asm_byte(__pu_val,__pu_addr,err); break; \
|
||||||
case 2: __put_user_asm_half(__pu_val,__pu_addr,err); break; \
|
case 2: __put_user_asm_half(__pu_val,__pu_addr,err); break; \
|
||||||
|
|
|
@ -159,6 +159,12 @@ static int debug_arch_supported(void)
|
||||||
arch >= ARM_DEBUG_ARCH_V7_1;
|
arch >= ARM_DEBUG_ARCH_V7_1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Can we determine the watchpoint access type from the fsr? */
|
||||||
|
static int debug_exception_updates_fsr(void)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* Determine number of WRP registers available. */
|
/* Determine number of WRP registers available. */
|
||||||
static int get_num_wrp_resources(void)
|
static int get_num_wrp_resources(void)
|
||||||
{
|
{
|
||||||
|
@ -604,13 +610,14 @@ int arch_validate_hwbkpt_settings(struct perf_event *bp)
|
||||||
/* Aligned */
|
/* Aligned */
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
/* Allow single byte watchpoint. */
|
|
||||||
if (info->ctrl.len == ARM_BREAKPOINT_LEN_1)
|
|
||||||
break;
|
|
||||||
case 2:
|
case 2:
|
||||||
/* Allow halfword watchpoints and breakpoints. */
|
/* Allow halfword watchpoints and breakpoints. */
|
||||||
if (info->ctrl.len == ARM_BREAKPOINT_LEN_2)
|
if (info->ctrl.len == ARM_BREAKPOINT_LEN_2)
|
||||||
break;
|
break;
|
||||||
|
case 3:
|
||||||
|
/* Allow single byte watchpoint. */
|
||||||
|
if (info->ctrl.len == ARM_BREAKPOINT_LEN_1)
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -619,18 +626,35 @@ int arch_validate_hwbkpt_settings(struct perf_event *bp)
|
||||||
info->address &= ~alignment_mask;
|
info->address &= ~alignment_mask;
|
||||||
info->ctrl.len <<= offset;
|
info->ctrl.len <<= offset;
|
||||||
|
|
||||||
/*
|
if (!bp->overflow_handler) {
|
||||||
* Currently we rely on an overflow handler to take
|
/*
|
||||||
* care of single-stepping the breakpoint when it fires.
|
* Mismatch breakpoints are required for single-stepping
|
||||||
* In the case of userspace breakpoints on a core with V7 debug,
|
* breakpoints.
|
||||||
* we can use the mismatch feature as a poor-man's hardware
|
*/
|
||||||
* single-step, but this only works for per-task breakpoints.
|
if (!core_has_mismatch_brps())
|
||||||
*/
|
return -EINVAL;
|
||||||
if (!bp->overflow_handler && (arch_check_bp_in_kernelspace(bp) ||
|
|
||||||
!core_has_mismatch_brps() || !bp->hw.bp_target)) {
|
/* We don't allow mismatch breakpoints in kernel space. */
|
||||||
pr_warning("overflow handler required but none found\n");
|
if (arch_check_bp_in_kernelspace(bp))
|
||||||
ret = -EINVAL;
|
return -EPERM;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Per-cpu breakpoints are not supported by our stepping
|
||||||
|
* mechanism.
|
||||||
|
*/
|
||||||
|
if (!bp->hw.bp_target)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We only support specific access types if the fsr
|
||||||
|
* reports them.
|
||||||
|
*/
|
||||||
|
if (!debug_exception_updates_fsr() &&
|
||||||
|
(info->ctrl.type == ARM_BREAKPOINT_LOAD ||
|
||||||
|
info->ctrl.type == ARM_BREAKPOINT_STORE))
|
||||||
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -706,10 +730,12 @@ static void watchpoint_handler(unsigned long addr, unsigned int fsr,
|
||||||
goto unlock;
|
goto unlock;
|
||||||
|
|
||||||
/* Check that the access type matches. */
|
/* Check that the access type matches. */
|
||||||
access = (fsr & ARM_FSR_ACCESS_MASK) ? HW_BREAKPOINT_W :
|
if (debug_exception_updates_fsr()) {
|
||||||
HW_BREAKPOINT_R;
|
access = (fsr & ARM_FSR_ACCESS_MASK) ?
|
||||||
if (!(access & hw_breakpoint_type(wp)))
|
HW_BREAKPOINT_W : HW_BREAKPOINT_R;
|
||||||
goto unlock;
|
if (!(access & hw_breakpoint_type(wp)))
|
||||||
|
goto unlock;
|
||||||
|
}
|
||||||
|
|
||||||
/* We have a winner. */
|
/* We have a winner. */
|
||||||
info->trigger = addr;
|
info->trigger = addr;
|
||||||
|
|
|
@ -420,20 +420,23 @@ asmlinkage void __exception do_undefinstr(struct pt_regs *regs)
|
||||||
#endif
|
#endif
|
||||||
instr = *(u32 *) pc;
|
instr = *(u32 *) pc;
|
||||||
} else if (thumb_mode(regs)) {
|
} else if (thumb_mode(regs)) {
|
||||||
get_user(instr, (u16 __user *)pc);
|
if (get_user(instr, (u16 __user *)pc))
|
||||||
|
goto die_sig;
|
||||||
if (is_wide_instruction(instr)) {
|
if (is_wide_instruction(instr)) {
|
||||||
unsigned int instr2;
|
unsigned int instr2;
|
||||||
get_user(instr2, (u16 __user *)pc+1);
|
if (get_user(instr2, (u16 __user *)pc+1))
|
||||||
|
goto die_sig;
|
||||||
instr <<= 16;
|
instr <<= 16;
|
||||||
instr |= instr2;
|
instr |= instr2;
|
||||||
}
|
}
|
||||||
} else {
|
} else if (get_user(instr, (u32 __user *)pc)) {
|
||||||
get_user(instr, (u32 __user *)pc);
|
goto die_sig;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (call_undef_hook(regs, instr) == 0)
|
if (call_undef_hook(regs, instr) == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
die_sig:
|
||||||
#ifdef CONFIG_DEBUG_USER
|
#ifdef CONFIG_DEBUG_USER
|
||||||
if (user_debug & UDBG_UNDEFINED) {
|
if (user_debug & UDBG_UNDEFINED) {
|
||||||
printk(KERN_INFO "%s (%d): undefined instruction: pc=%p\n",
|
printk(KERN_INFO "%s (%d): undefined instruction: pc=%p\n",
|
||||||
|
|
|
@ -59,6 +59,7 @@ void __init init_current_timer_delay(unsigned long freq)
|
||||||
{
|
{
|
||||||
pr_info("Switching to timer-based delay loop\n");
|
pr_info("Switching to timer-based delay loop\n");
|
||||||
lpj_fine = freq / HZ;
|
lpj_fine = freq / HZ;
|
||||||
|
loops_per_jiffy = lpj_fine;
|
||||||
arm_delay_ops.delay = __timer_delay;
|
arm_delay_ops.delay = __timer_delay;
|
||||||
arm_delay_ops.const_udelay = __timer_const_udelay;
|
arm_delay_ops.const_udelay = __timer_const_udelay;
|
||||||
arm_delay_ops.udelay = __timer_udelay;
|
arm_delay_ops.udelay = __timer_udelay;
|
||||||
|
|
|
@ -16,8 +16,9 @@
|
||||||
* __get_user_X
|
* __get_user_X
|
||||||
*
|
*
|
||||||
* Inputs: r0 contains the address
|
* Inputs: r0 contains the address
|
||||||
|
* r1 contains the address limit, which must be preserved
|
||||||
* Outputs: r0 is the error code
|
* Outputs: r0 is the error code
|
||||||
* r2, r3 contains the zero-extended value
|
* r2 contains the zero-extended value
|
||||||
* lr corrupted
|
* lr corrupted
|
||||||
*
|
*
|
||||||
* No other registers must be altered. (see <asm/uaccess.h>
|
* No other registers must be altered. (see <asm/uaccess.h>
|
||||||
|
@ -27,33 +28,39 @@
|
||||||
* Note also that it is intended that __get_user_bad is not global.
|
* Note also that it is intended that __get_user_bad is not global.
|
||||||
*/
|
*/
|
||||||
#include <linux/linkage.h>
|
#include <linux/linkage.h>
|
||||||
|
#include <asm/assembler.h>
|
||||||
#include <asm/errno.h>
|
#include <asm/errno.h>
|
||||||
#include <asm/domain.h>
|
#include <asm/domain.h>
|
||||||
|
|
||||||
ENTRY(__get_user_1)
|
ENTRY(__get_user_1)
|
||||||
|
check_uaccess r0, 1, r1, r2, __get_user_bad
|
||||||
1: TUSER(ldrb) r2, [r0]
|
1: TUSER(ldrb) r2, [r0]
|
||||||
mov r0, #0
|
mov r0, #0
|
||||||
mov pc, lr
|
mov pc, lr
|
||||||
ENDPROC(__get_user_1)
|
ENDPROC(__get_user_1)
|
||||||
|
|
||||||
ENTRY(__get_user_2)
|
ENTRY(__get_user_2)
|
||||||
#ifdef CONFIG_THUMB2_KERNEL
|
check_uaccess r0, 2, r1, r2, __get_user_bad
|
||||||
2: TUSER(ldrb) r2, [r0]
|
#ifdef CONFIG_CPU_USE_DOMAINS
|
||||||
3: TUSER(ldrb) r3, [r0, #1]
|
rb .req ip
|
||||||
|
2: ldrbt r2, [r0], #1
|
||||||
|
3: ldrbt rb, [r0], #0
|
||||||
#else
|
#else
|
||||||
2: TUSER(ldrb) r2, [r0], #1
|
rb .req r0
|
||||||
3: TUSER(ldrb) r3, [r0]
|
2: ldrb r2, [r0]
|
||||||
|
3: ldrb rb, [r0, #1]
|
||||||
#endif
|
#endif
|
||||||
#ifndef __ARMEB__
|
#ifndef __ARMEB__
|
||||||
orr r2, r2, r3, lsl #8
|
orr r2, r2, rb, lsl #8
|
||||||
#else
|
#else
|
||||||
orr r2, r3, r2, lsl #8
|
orr r2, rb, r2, lsl #8
|
||||||
#endif
|
#endif
|
||||||
mov r0, #0
|
mov r0, #0
|
||||||
mov pc, lr
|
mov pc, lr
|
||||||
ENDPROC(__get_user_2)
|
ENDPROC(__get_user_2)
|
||||||
|
|
||||||
ENTRY(__get_user_4)
|
ENTRY(__get_user_4)
|
||||||
|
check_uaccess r0, 4, r1, r2, __get_user_bad
|
||||||
4: TUSER(ldr) r2, [r0]
|
4: TUSER(ldr) r2, [r0]
|
||||||
mov r0, #0
|
mov r0, #0
|
||||||
mov pc, lr
|
mov pc, lr
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
* __put_user_X
|
* __put_user_X
|
||||||
*
|
*
|
||||||
* Inputs: r0 contains the address
|
* Inputs: r0 contains the address
|
||||||
|
* r1 contains the address limit, which must be preserved
|
||||||
* r2, r3 contains the value
|
* r2, r3 contains the value
|
||||||
* Outputs: r0 is the error code
|
* Outputs: r0 is the error code
|
||||||
* lr corrupted
|
* lr corrupted
|
||||||
|
@ -27,16 +28,19 @@
|
||||||
* Note also that it is intended that __put_user_bad is not global.
|
* Note also that it is intended that __put_user_bad is not global.
|
||||||
*/
|
*/
|
||||||
#include <linux/linkage.h>
|
#include <linux/linkage.h>
|
||||||
|
#include <asm/assembler.h>
|
||||||
#include <asm/errno.h>
|
#include <asm/errno.h>
|
||||||
#include <asm/domain.h>
|
#include <asm/domain.h>
|
||||||
|
|
||||||
ENTRY(__put_user_1)
|
ENTRY(__put_user_1)
|
||||||
|
check_uaccess r0, 1, r1, ip, __put_user_bad
|
||||||
1: TUSER(strb) r2, [r0]
|
1: TUSER(strb) r2, [r0]
|
||||||
mov r0, #0
|
mov r0, #0
|
||||||
mov pc, lr
|
mov pc, lr
|
||||||
ENDPROC(__put_user_1)
|
ENDPROC(__put_user_1)
|
||||||
|
|
||||||
ENTRY(__put_user_2)
|
ENTRY(__put_user_2)
|
||||||
|
check_uaccess r0, 2, r1, ip, __put_user_bad
|
||||||
mov ip, r2, lsr #8
|
mov ip, r2, lsr #8
|
||||||
#ifdef CONFIG_THUMB2_KERNEL
|
#ifdef CONFIG_THUMB2_KERNEL
|
||||||
#ifndef __ARMEB__
|
#ifndef __ARMEB__
|
||||||
|
@ -60,12 +64,14 @@ ENTRY(__put_user_2)
|
||||||
ENDPROC(__put_user_2)
|
ENDPROC(__put_user_2)
|
||||||
|
|
||||||
ENTRY(__put_user_4)
|
ENTRY(__put_user_4)
|
||||||
|
check_uaccess r0, 4, r1, ip, __put_user_bad
|
||||||
4: TUSER(str) r2, [r0]
|
4: TUSER(str) r2, [r0]
|
||||||
mov r0, #0
|
mov r0, #0
|
||||||
mov pc, lr
|
mov pc, lr
|
||||||
ENDPROC(__put_user_4)
|
ENDPROC(__put_user_4)
|
||||||
|
|
||||||
ENTRY(__put_user_8)
|
ENTRY(__put_user_8)
|
||||||
|
check_uaccess r0, 8, r1, ip, __put_user_bad
|
||||||
#ifdef CONFIG_THUMB2_KERNEL
|
#ifdef CONFIG_THUMB2_KERNEL
|
||||||
5: TUSER(str) r2, [r0]
|
5: TUSER(str) r2, [r0]
|
||||||
6: TUSER(str) r3, [r0, #4]
|
6: TUSER(str) r3, [r0, #4]
|
||||||
|
|
|
@ -197,7 +197,7 @@ void __init at91rm9200_timer_init(void)
|
||||||
at91_st_read(AT91_ST_SR);
|
at91_st_read(AT91_ST_SR);
|
||||||
|
|
||||||
/* Make IRQs happen for the system timer */
|
/* Make IRQs happen for the system timer */
|
||||||
setup_irq(AT91_ID_SYS, &at91rm9200_timer_irq);
|
setup_irq(NR_IRQS_LEGACY + AT91_ID_SYS, &at91rm9200_timer_irq);
|
||||||
|
|
||||||
/* The 32KiHz "Slow Clock" (tick every 30517.58 nanoseconds) is used
|
/* The 32KiHz "Slow Clock" (tick every 30517.58 nanoseconds) is used
|
||||||
* directly for the clocksource and all clockevents, after adjusting
|
* directly for the clocksource and all clockevents, after adjusting
|
||||||
|
|
|
@ -726,6 +726,8 @@ static struct resource rtt_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -744,10 +746,12 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
* The second resource is needed:
|
* The second resource is needed:
|
||||||
* GPBR will serve as the storage for RTC time offset
|
* GPBR will serve as the storage for RTC time offset
|
||||||
*/
|
*/
|
||||||
at91sam9260_rtt_device.num_resources = 2;
|
at91sam9260_rtt_device.num_resources = 3;
|
||||||
rtt_resources[1].start = AT91SAM9260_BASE_GPBR +
|
rtt_resources[1].start = AT91SAM9260_BASE_GPBR +
|
||||||
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
||||||
rtt_resources[1].end = rtt_resources[1].start + 3;
|
rtt_resources[1].end = rtt_resources[1].start + 3;
|
||||||
|
rtt_resources[2].start = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
|
rtt_resources[2].end = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static void __init at91_add_device_rtt_rtc(void)
|
static void __init at91_add_device_rtt_rtc(void)
|
||||||
|
|
|
@ -609,6 +609,8 @@ static struct resource rtt_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -626,10 +628,12 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
* The second resource is needed:
|
* The second resource is needed:
|
||||||
* GPBR will serve as the storage for RTC time offset
|
* GPBR will serve as the storage for RTC time offset
|
||||||
*/
|
*/
|
||||||
at91sam9261_rtt_device.num_resources = 2;
|
at91sam9261_rtt_device.num_resources = 3;
|
||||||
rtt_resources[1].start = AT91SAM9261_BASE_GPBR +
|
rtt_resources[1].start = AT91SAM9261_BASE_GPBR +
|
||||||
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
||||||
rtt_resources[1].end = rtt_resources[1].start + 3;
|
rtt_resources[1].end = rtt_resources[1].start + 3;
|
||||||
|
rtt_resources[2].start = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
|
rtt_resources[2].end = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static void __init at91_add_device_rtt_rtc(void)
|
static void __init at91_add_device_rtt_rtc(void)
|
||||||
|
|
|
@ -990,6 +990,8 @@ static struct resource rtt0_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1006,6 +1008,8 @@ static struct resource rtt1_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1027,14 +1031,14 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
* The second resource is needed only for the chosen RTT:
|
* The second resource is needed only for the chosen RTT:
|
||||||
* GPBR will serve as the storage for RTC time offset
|
* GPBR will serve as the storage for RTC time offset
|
||||||
*/
|
*/
|
||||||
at91sam9263_rtt0_device.num_resources = 2;
|
at91sam9263_rtt0_device.num_resources = 3;
|
||||||
at91sam9263_rtt1_device.num_resources = 1;
|
at91sam9263_rtt1_device.num_resources = 1;
|
||||||
pdev = &at91sam9263_rtt0_device;
|
pdev = &at91sam9263_rtt0_device;
|
||||||
r = rtt0_resources;
|
r = rtt0_resources;
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
at91sam9263_rtt0_device.num_resources = 1;
|
at91sam9263_rtt0_device.num_resources = 1;
|
||||||
at91sam9263_rtt1_device.num_resources = 2;
|
at91sam9263_rtt1_device.num_resources = 3;
|
||||||
pdev = &at91sam9263_rtt1_device;
|
pdev = &at91sam9263_rtt1_device;
|
||||||
r = rtt1_resources;
|
r = rtt1_resources;
|
||||||
break;
|
break;
|
||||||
|
@ -1047,6 +1051,8 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
pdev->name = "rtc-at91sam9";
|
pdev->name = "rtc-at91sam9";
|
||||||
r[1].start = AT91SAM9263_BASE_GPBR + 4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
r[1].start = AT91SAM9263_BASE_GPBR + 4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
||||||
r[1].end = r[1].start + 3;
|
r[1].end = r[1].start + 3;
|
||||||
|
r[2].start = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
|
r[2].end = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static void __init at91_add_device_rtt_rtc(void)
|
static void __init at91_add_device_rtt_rtc(void)
|
||||||
|
|
|
@ -1293,6 +1293,8 @@ static struct resource rtt_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1310,10 +1312,12 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
* The second resource is needed:
|
* The second resource is needed:
|
||||||
* GPBR will serve as the storage for RTC time offset
|
* GPBR will serve as the storage for RTC time offset
|
||||||
*/
|
*/
|
||||||
at91sam9g45_rtt_device.num_resources = 2;
|
at91sam9g45_rtt_device.num_resources = 3;
|
||||||
rtt_resources[1].start = AT91SAM9G45_BASE_GPBR +
|
rtt_resources[1].start = AT91SAM9G45_BASE_GPBR +
|
||||||
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
||||||
rtt_resources[1].end = rtt_resources[1].start + 3;
|
rtt_resources[1].end = rtt_resources[1].start + 3;
|
||||||
|
rtt_resources[2].start = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
|
rtt_resources[2].end = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static void __init at91_add_device_rtt_rtc(void)
|
static void __init at91_add_device_rtt_rtc(void)
|
||||||
|
|
|
@ -688,6 +688,8 @@ static struct resource rtt_resources[] = {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
}, {
|
}, {
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
|
}, {
|
||||||
|
.flags = IORESOURCE_IRQ,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -705,10 +707,12 @@ static void __init at91_add_device_rtt_rtc(void)
|
||||||
* The second resource is needed:
|
* The second resource is needed:
|
||||||
* GPBR will serve as the storage for RTC time offset
|
* GPBR will serve as the storage for RTC time offset
|
||||||
*/
|
*/
|
||||||
at91sam9rl_rtt_device.num_resources = 2;
|
at91sam9rl_rtt_device.num_resources = 3;
|
||||||
rtt_resources[1].start = AT91SAM9RL_BASE_GPBR +
|
rtt_resources[1].start = AT91SAM9RL_BASE_GPBR +
|
||||||
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
4 * CONFIG_RTC_DRV_AT91SAM9_GPBR;
|
||||||
rtt_resources[1].end = rtt_resources[1].start + 3;
|
rtt_resources[1].end = rtt_resources[1].start + 3;
|
||||||
|
rtt_resources[2].start = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
|
rtt_resources[2].end = NR_IRQS_LEGACY + AT91_ID_SYS;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static void __init at91_add_device_rtt_rtc(void)
|
static void __init at91_add_device_rtt_rtc(void)
|
||||||
|
|
|
@ -63,6 +63,12 @@ EXPORT_SYMBOL_GPL(at91_pmc_base);
|
||||||
|
|
||||||
#define cpu_has_300M_plla() (cpu_is_at91sam9g10())
|
#define cpu_has_300M_plla() (cpu_is_at91sam9g10())
|
||||||
|
|
||||||
|
#define cpu_has_240M_plla() (cpu_is_at91sam9261() \
|
||||||
|
|| cpu_is_at91sam9263() \
|
||||||
|
|| cpu_is_at91sam9rl())
|
||||||
|
|
||||||
|
#define cpu_has_210M_plla() (cpu_is_at91sam9260())
|
||||||
|
|
||||||
#define cpu_has_pllb() (!(cpu_is_at91sam9rl() \
|
#define cpu_has_pllb() (!(cpu_is_at91sam9rl() \
|
||||||
|| cpu_is_at91sam9g45() \
|
|| cpu_is_at91sam9g45() \
|
||||||
|| cpu_is_at91sam9x5() \
|
|| cpu_is_at91sam9x5() \
|
||||||
|
@ -706,6 +712,12 @@ static int __init at91_pmc_init(unsigned long main_clock)
|
||||||
} else if (cpu_has_800M_plla()) {
|
} else if (cpu_has_800M_plla()) {
|
||||||
if (plla.rate_hz > 800000000)
|
if (plla.rate_hz > 800000000)
|
||||||
pll_overclock = true;
|
pll_overclock = true;
|
||||||
|
} else if (cpu_has_240M_plla()) {
|
||||||
|
if (plla.rate_hz > 240000000)
|
||||||
|
pll_overclock = true;
|
||||||
|
} else if (cpu_has_210M_plla()) {
|
||||||
|
if (plla.rate_hz > 210000000)
|
||||||
|
pll_overclock = true;
|
||||||
} else {
|
} else {
|
||||||
if (plla.rate_hz > 209000000)
|
if (plla.rate_hz > 209000000)
|
||||||
pll_overclock = true;
|
pll_overclock = true;
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <asm/irq.h>
|
#include <asm/irq.h>
|
||||||
#include <asm/mach/irq.h>
|
#include <asm/mach/irq.h>
|
||||||
|
#include <asm/system_misc.h>
|
||||||
#include <mach/hardware.h>
|
#include <mach/hardware.h>
|
||||||
|
|
||||||
#define IRQ_SOURCE(base_addr) (base_addr + 0x00)
|
#define IRQ_SOURCE(base_addr) (base_addr + 0x00)
|
||||||
|
|
|
@ -222,10 +222,8 @@ int __init mx25_clocks_init(void)
|
||||||
clk_register_clkdev(clk[lcdc_ipg], "ipg", "imx-fb.0");
|
clk_register_clkdev(clk[lcdc_ipg], "ipg", "imx-fb.0");
|
||||||
clk_register_clkdev(clk[lcdc_ahb], "ahb", "imx-fb.0");
|
clk_register_clkdev(clk[lcdc_ahb], "ahb", "imx-fb.0");
|
||||||
clk_register_clkdev(clk[wdt_ipg], NULL, "imx2-wdt.0");
|
clk_register_clkdev(clk[wdt_ipg], NULL, "imx2-wdt.0");
|
||||||
clk_register_clkdev(clk[ssi1_ipg_per], "per", "imx-ssi.0");
|
clk_register_clkdev(clk[ssi1_ipg], NULL, "imx-ssi.0");
|
||||||
clk_register_clkdev(clk[ssi1_ipg], "ipg", "imx-ssi.0");
|
clk_register_clkdev(clk[ssi2_ipg], NULL, "imx-ssi.1");
|
||||||
clk_register_clkdev(clk[ssi2_ipg_per], "per", "imx-ssi.1");
|
|
||||||
clk_register_clkdev(clk[ssi2_ipg], "ipg", "imx-ssi.1");
|
|
||||||
clk_register_clkdev(clk[esdhc1_ipg_per], "per", "sdhci-esdhc-imx25.0");
|
clk_register_clkdev(clk[esdhc1_ipg_per], "per", "sdhci-esdhc-imx25.0");
|
||||||
clk_register_clkdev(clk[esdhc1_ipg], "ipg", "sdhci-esdhc-imx25.0");
|
clk_register_clkdev(clk[esdhc1_ipg], "ipg", "sdhci-esdhc-imx25.0");
|
||||||
clk_register_clkdev(clk[esdhc1_ahb], "ahb", "sdhci-esdhc-imx25.0");
|
clk_register_clkdev(clk[esdhc1_ahb], "ahb", "sdhci-esdhc-imx25.0");
|
||||||
|
|
|
@ -230,10 +230,8 @@ int __init mx35_clocks_init()
|
||||||
clk_register_clkdev(clk[ipu_gate], NULL, "mx3_sdc_fb");
|
clk_register_clkdev(clk[ipu_gate], NULL, "mx3_sdc_fb");
|
||||||
clk_register_clkdev(clk[owire_gate], NULL, "mxc_w1");
|
clk_register_clkdev(clk[owire_gate], NULL, "mxc_w1");
|
||||||
clk_register_clkdev(clk[sdma_gate], NULL, "imx35-sdma");
|
clk_register_clkdev(clk[sdma_gate], NULL, "imx35-sdma");
|
||||||
clk_register_clkdev(clk[ipg], "ipg", "imx-ssi.0");
|
clk_register_clkdev(clk[ssi1_gate], NULL, "imx-ssi.0");
|
||||||
clk_register_clkdev(clk[ssi1_div_post], "per", "imx-ssi.0");
|
clk_register_clkdev(clk[ssi2_gate], NULL, "imx-ssi.1");
|
||||||
clk_register_clkdev(clk[ipg], "ipg", "imx-ssi.1");
|
|
||||||
clk_register_clkdev(clk[ssi2_div_post], "per", "imx-ssi.1");
|
|
||||||
/* i.mx35 has the i.mx21 type uart */
|
/* i.mx35 has the i.mx21 type uart */
|
||||||
clk_register_clkdev(clk[uart1_gate], "per", "imx21-uart.0");
|
clk_register_clkdev(clk[uart1_gate], "per", "imx21-uart.0");
|
||||||
clk_register_clkdev(clk[ipg], "ipg", "imx21-uart.0");
|
clk_register_clkdev(clk[ipg], "ipg", "imx21-uart.0");
|
||||||
|
|
|
@ -517,6 +517,13 @@ void __init kirkwood_wdt_init(void)
|
||||||
void __init kirkwood_init_early(void)
|
void __init kirkwood_init_early(void)
|
||||||
{
|
{
|
||||||
orion_time_set_base(TIMER_VIRT_BASE);
|
orion_time_set_base(TIMER_VIRT_BASE);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Some Kirkwood devices allocate their coherent buffers from atomic
|
||||||
|
* context. Increase size of atomic coherent pool to make sure such
|
||||||
|
* the allocations won't fail.
|
||||||
|
*/
|
||||||
|
init_dma_coherent_pool_size(SZ_1M);
|
||||||
}
|
}
|
||||||
|
|
||||||
int kirkwood_tclk;
|
int kirkwood_tclk;
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
|
#include <linux/sizes.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
#include <linux/mtd/partitions.h>
|
#include <linux/mtd/partitions.h>
|
||||||
#include <linux/ata_platform.h>
|
#include <linux/ata_platform.h>
|
||||||
|
|
|
@ -232,10 +232,11 @@ config MACH_OMAP3_PANDORA
|
||||||
select OMAP_PACKAGE_CBB
|
select OMAP_PACKAGE_CBB
|
||||||
select REGULATOR_FIXED_VOLTAGE if REGULATOR
|
select REGULATOR_FIXED_VOLTAGE if REGULATOR
|
||||||
|
|
||||||
config MACH_OMAP3_TOUCHBOOK
|
config MACH_TOUCHBOOK
|
||||||
bool "OMAP3 Touch Book"
|
bool "OMAP3 Touch Book"
|
||||||
depends on ARCH_OMAP3
|
depends on ARCH_OMAP3
|
||||||
default y
|
default y
|
||||||
|
select OMAP_PACKAGE_CBB
|
||||||
|
|
||||||
config MACH_OMAP_3430SDP
|
config MACH_OMAP_3430SDP
|
||||||
bool "OMAP 3430 SDP board"
|
bool "OMAP 3430 SDP board"
|
||||||
|
|
|
@ -255,7 +255,7 @@ obj-$(CONFIG_MACH_OMAP_3630SDP) += board-zoom-display.o
|
||||||
obj-$(CONFIG_MACH_CM_T35) += board-cm-t35.o
|
obj-$(CONFIG_MACH_CM_T35) += board-cm-t35.o
|
||||||
obj-$(CONFIG_MACH_CM_T3517) += board-cm-t3517.o
|
obj-$(CONFIG_MACH_CM_T3517) += board-cm-t3517.o
|
||||||
obj-$(CONFIG_MACH_IGEP0020) += board-igep0020.o
|
obj-$(CONFIG_MACH_IGEP0020) += board-igep0020.o
|
||||||
obj-$(CONFIG_MACH_OMAP3_TOUCHBOOK) += board-omap3touchbook.o
|
obj-$(CONFIG_MACH_TOUCHBOOK) += board-omap3touchbook.o
|
||||||
obj-$(CONFIG_MACH_OMAP_4430SDP) += board-4430sdp.o
|
obj-$(CONFIG_MACH_OMAP_4430SDP) += board-4430sdp.o
|
||||||
obj-$(CONFIG_MACH_OMAP4_PANDA) += board-omap4panda.o
|
obj-$(CONFIG_MACH_OMAP4_PANDA) += board-omap4panda.o
|
||||||
|
|
||||||
|
|
|
@ -1036,13 +1036,13 @@ static struct omap_clk am33xx_clks[] = {
|
||||||
CLK(NULL, "mmu_fck", &mmu_fck, CK_AM33XX),
|
CLK(NULL, "mmu_fck", &mmu_fck, CK_AM33XX),
|
||||||
CLK(NULL, "smartreflex0_fck", &smartreflex0_fck, CK_AM33XX),
|
CLK(NULL, "smartreflex0_fck", &smartreflex0_fck, CK_AM33XX),
|
||||||
CLK(NULL, "smartreflex1_fck", &smartreflex1_fck, CK_AM33XX),
|
CLK(NULL, "smartreflex1_fck", &smartreflex1_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt1_fck", &timer1_fck, CK_AM33XX),
|
CLK(NULL, "timer1_fck", &timer1_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt2_fck", &timer2_fck, CK_AM33XX),
|
CLK(NULL, "timer2_fck", &timer2_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt3_fck", &timer3_fck, CK_AM33XX),
|
CLK(NULL, "timer3_fck", &timer3_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt4_fck", &timer4_fck, CK_AM33XX),
|
CLK(NULL, "timer4_fck", &timer4_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt5_fck", &timer5_fck, CK_AM33XX),
|
CLK(NULL, "timer5_fck", &timer5_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt6_fck", &timer6_fck, CK_AM33XX),
|
CLK(NULL, "timer6_fck", &timer6_fck, CK_AM33XX),
|
||||||
CLK(NULL, "gpt7_fck", &timer7_fck, CK_AM33XX),
|
CLK(NULL, "timer7_fck", &timer7_fck, CK_AM33XX),
|
||||||
CLK(NULL, "usbotg_fck", &usbotg_fck, CK_AM33XX),
|
CLK(NULL, "usbotg_fck", &usbotg_fck, CK_AM33XX),
|
||||||
CLK(NULL, "ieee5000_fck", &ieee5000_fck, CK_AM33XX),
|
CLK(NULL, "ieee5000_fck", &ieee5000_fck, CK_AM33XX),
|
||||||
CLK(NULL, "wdt1_fck", &wdt1_fck, CK_AM33XX),
|
CLK(NULL, "wdt1_fck", &wdt1_fck, CK_AM33XX),
|
||||||
|
|
|
@ -241,6 +241,52 @@ static void omap3_clkdm_deny_idle(struct clockdomain *clkdm)
|
||||||
_clkdm_del_autodeps(clkdm);
|
_clkdm_del_autodeps(clkdm);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int omap3xxx_clkdm_clk_enable(struct clockdomain *clkdm)
|
||||||
|
{
|
||||||
|
bool hwsup = false;
|
||||||
|
|
||||||
|
if (!clkdm->clktrctrl_mask)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
hwsup = omap2_cm_is_clkdm_in_hwsup(clkdm->pwrdm.ptr->prcm_offs,
|
||||||
|
clkdm->clktrctrl_mask);
|
||||||
|
|
||||||
|
if (hwsup) {
|
||||||
|
/* Disable HW transitions when we are changing deps */
|
||||||
|
_disable_hwsup(clkdm);
|
||||||
|
_clkdm_add_autodeps(clkdm);
|
||||||
|
_enable_hwsup(clkdm);
|
||||||
|
} else {
|
||||||
|
if (clkdm->flags & CLKDM_CAN_FORCE_WAKEUP)
|
||||||
|
omap3_clkdm_wakeup(clkdm);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int omap3xxx_clkdm_clk_disable(struct clockdomain *clkdm)
|
||||||
|
{
|
||||||
|
bool hwsup = false;
|
||||||
|
|
||||||
|
if (!clkdm->clktrctrl_mask)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
hwsup = omap2_cm_is_clkdm_in_hwsup(clkdm->pwrdm.ptr->prcm_offs,
|
||||||
|
clkdm->clktrctrl_mask);
|
||||||
|
|
||||||
|
if (hwsup) {
|
||||||
|
/* Disable HW transitions when we are changing deps */
|
||||||
|
_disable_hwsup(clkdm);
|
||||||
|
_clkdm_del_autodeps(clkdm);
|
||||||
|
_enable_hwsup(clkdm);
|
||||||
|
} else {
|
||||||
|
if (clkdm->flags & CLKDM_CAN_FORCE_SLEEP)
|
||||||
|
omap3_clkdm_sleep(clkdm);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
struct clkdm_ops omap2_clkdm_operations = {
|
struct clkdm_ops omap2_clkdm_operations = {
|
||||||
.clkdm_add_wkdep = omap2_clkdm_add_wkdep,
|
.clkdm_add_wkdep = omap2_clkdm_add_wkdep,
|
||||||
.clkdm_del_wkdep = omap2_clkdm_del_wkdep,
|
.clkdm_del_wkdep = omap2_clkdm_del_wkdep,
|
||||||
|
@ -267,6 +313,6 @@ struct clkdm_ops omap3_clkdm_operations = {
|
||||||
.clkdm_wakeup = omap3_clkdm_wakeup,
|
.clkdm_wakeup = omap3_clkdm_wakeup,
|
||||||
.clkdm_allow_idle = omap3_clkdm_allow_idle,
|
.clkdm_allow_idle = omap3_clkdm_allow_idle,
|
||||||
.clkdm_deny_idle = omap3_clkdm_deny_idle,
|
.clkdm_deny_idle = omap3_clkdm_deny_idle,
|
||||||
.clkdm_clk_enable = omap2_clkdm_clk_enable,
|
.clkdm_clk_enable = omap3xxx_clkdm_clk_enable,
|
||||||
.clkdm_clk_disable = omap2_clkdm_clk_disable,
|
.clkdm_clk_disable = omap3xxx_clkdm_clk_disable,
|
||||||
};
|
};
|
||||||
|
|
|
@ -67,6 +67,7 @@
|
||||||
#define OMAP3430_EN_IVA2_DPLL_MASK (0x7 << 0)
|
#define OMAP3430_EN_IVA2_DPLL_MASK (0x7 << 0)
|
||||||
|
|
||||||
/* CM_IDLEST_IVA2 */
|
/* CM_IDLEST_IVA2 */
|
||||||
|
#define OMAP3430_ST_IVA2_SHIFT 0
|
||||||
#define OMAP3430_ST_IVA2_MASK (1 << 0)
|
#define OMAP3430_ST_IVA2_MASK (1 << 0)
|
||||||
|
|
||||||
/* CM_IDLEST_PLL_IVA2 */
|
/* CM_IDLEST_PLL_IVA2 */
|
||||||
|
|
|
@ -46,7 +46,7 @@
|
||||||
static void __iomem *wakeupgen_base;
|
static void __iomem *wakeupgen_base;
|
||||||
static void __iomem *sar_base;
|
static void __iomem *sar_base;
|
||||||
static DEFINE_SPINLOCK(wakeupgen_lock);
|
static DEFINE_SPINLOCK(wakeupgen_lock);
|
||||||
static unsigned int irq_target_cpu[NR_IRQS];
|
static unsigned int irq_target_cpu[MAX_IRQS];
|
||||||
static unsigned int irq_banks = MAX_NR_REG_BANKS;
|
static unsigned int irq_banks = MAX_NR_REG_BANKS;
|
||||||
static unsigned int max_irqs = MAX_IRQS;
|
static unsigned int max_irqs = MAX_IRQS;
|
||||||
static unsigned int omap_secure_apis;
|
static unsigned int omap_secure_apis;
|
||||||
|
|
|
@ -1889,6 +1889,7 @@ static int _enable(struct omap_hwmod *oh)
|
||||||
_enable_sysc(oh);
|
_enable_sysc(oh);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
_omap4_disable_module(oh);
|
||||||
_disable_clocks(oh);
|
_disable_clocks(oh);
|
||||||
pr_debug("omap_hwmod: %s: _wait_target_ready: %d\n",
|
pr_debug("omap_hwmod: %s: _wait_target_ready: %d\n",
|
||||||
oh->name, r);
|
oh->name, r);
|
||||||
|
|
|
@ -100,9 +100,9 @@ static struct omap_hwmod omap3xxx_mpu_hwmod = {
|
||||||
|
|
||||||
/* IVA2 (IVA2) */
|
/* IVA2 (IVA2) */
|
||||||
static struct omap_hwmod_rst_info omap3xxx_iva_resets[] = {
|
static struct omap_hwmod_rst_info omap3xxx_iva_resets[] = {
|
||||||
{ .name = "logic", .rst_shift = 0 },
|
{ .name = "logic", .rst_shift = 0, .st_shift = 8 },
|
||||||
{ .name = "seq0", .rst_shift = 1 },
|
{ .name = "seq0", .rst_shift = 1, .st_shift = 9 },
|
||||||
{ .name = "seq1", .rst_shift = 2 },
|
{ .name = "seq1", .rst_shift = 2, .st_shift = 10 },
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct omap_hwmod omap3xxx_iva_hwmod = {
|
static struct omap_hwmod omap3xxx_iva_hwmod = {
|
||||||
|
@ -112,6 +112,15 @@ static struct omap_hwmod omap3xxx_iva_hwmod = {
|
||||||
.rst_lines = omap3xxx_iva_resets,
|
.rst_lines = omap3xxx_iva_resets,
|
||||||
.rst_lines_cnt = ARRAY_SIZE(omap3xxx_iva_resets),
|
.rst_lines_cnt = ARRAY_SIZE(omap3xxx_iva_resets),
|
||||||
.main_clk = "iva2_ck",
|
.main_clk = "iva2_ck",
|
||||||
|
.prcm = {
|
||||||
|
.omap2 = {
|
||||||
|
.module_offs = OMAP3430_IVA2_MOD,
|
||||||
|
.prcm_reg_id = 1,
|
||||||
|
.module_bit = OMAP3430_CM_FCLKEN_IVA2_EN_IVA2_SHIFT,
|
||||||
|
.idlest_reg_id = 1,
|
||||||
|
.idlest_idle_bit = OMAP3430_ST_IVA2_SHIFT,
|
||||||
|
}
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* timer class */
|
/* timer class */
|
||||||
|
|
|
@ -4210,7 +4210,7 @@ static struct omap_hwmod_ocp_if omap44xx_dsp__iva = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* dsp -> sl2if */
|
/* dsp -> sl2if */
|
||||||
static struct omap_hwmod_ocp_if omap44xx_dsp__sl2if = {
|
static struct omap_hwmod_ocp_if __maybe_unused omap44xx_dsp__sl2if = {
|
||||||
.master = &omap44xx_dsp_hwmod,
|
.master = &omap44xx_dsp_hwmod,
|
||||||
.slave = &omap44xx_sl2if_hwmod,
|
.slave = &omap44xx_sl2if_hwmod,
|
||||||
.clk = "dpll_iva_m5x2_ck",
|
.clk = "dpll_iva_m5x2_ck",
|
||||||
|
@ -4828,7 +4828,7 @@ static struct omap_hwmod_ocp_if omap44xx_l3_main_2__iss = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* iva -> sl2if */
|
/* iva -> sl2if */
|
||||||
static struct omap_hwmod_ocp_if omap44xx_iva__sl2if = {
|
static struct omap_hwmod_ocp_if __maybe_unused omap44xx_iva__sl2if = {
|
||||||
.master = &omap44xx_iva_hwmod,
|
.master = &omap44xx_iva_hwmod,
|
||||||
.slave = &omap44xx_sl2if_hwmod,
|
.slave = &omap44xx_sl2if_hwmod,
|
||||||
.clk = "dpll_iva_m5x2_ck",
|
.clk = "dpll_iva_m5x2_ck",
|
||||||
|
@ -5364,7 +5364,7 @@ static struct omap_hwmod_ocp_if omap44xx_l4_wkup__scrm = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* l3_main_2 -> sl2if */
|
/* l3_main_2 -> sl2if */
|
||||||
static struct omap_hwmod_ocp_if omap44xx_l3_main_2__sl2if = {
|
static struct omap_hwmod_ocp_if __maybe_unused omap44xx_l3_main_2__sl2if = {
|
||||||
.master = &omap44xx_l3_main_2_hwmod,
|
.master = &omap44xx_l3_main_2_hwmod,
|
||||||
.slave = &omap44xx_sl2if_hwmod,
|
.slave = &omap44xx_sl2if_hwmod,
|
||||||
.clk = "l3_div_ck",
|
.clk = "l3_div_ck",
|
||||||
|
@ -6034,7 +6034,7 @@ static struct omap_hwmod_ocp_if *omap44xx_hwmod_ocp_ifs[] __initdata = {
|
||||||
&omap44xx_l4_abe__dmic,
|
&omap44xx_l4_abe__dmic,
|
||||||
&omap44xx_l4_abe__dmic_dma,
|
&omap44xx_l4_abe__dmic_dma,
|
||||||
&omap44xx_dsp__iva,
|
&omap44xx_dsp__iva,
|
||||||
&omap44xx_dsp__sl2if,
|
/* &omap44xx_dsp__sl2if, */
|
||||||
&omap44xx_l4_cfg__dsp,
|
&omap44xx_l4_cfg__dsp,
|
||||||
&omap44xx_l3_main_2__dss,
|
&omap44xx_l3_main_2__dss,
|
||||||
&omap44xx_l4_per__dss,
|
&omap44xx_l4_per__dss,
|
||||||
|
@ -6070,7 +6070,7 @@ static struct omap_hwmod_ocp_if *omap44xx_hwmod_ocp_ifs[] __initdata = {
|
||||||
&omap44xx_l4_per__i2c4,
|
&omap44xx_l4_per__i2c4,
|
||||||
&omap44xx_l3_main_2__ipu,
|
&omap44xx_l3_main_2__ipu,
|
||||||
&omap44xx_l3_main_2__iss,
|
&omap44xx_l3_main_2__iss,
|
||||||
&omap44xx_iva__sl2if,
|
/* &omap44xx_iva__sl2if, */
|
||||||
&omap44xx_l3_main_2__iva,
|
&omap44xx_l3_main_2__iva,
|
||||||
&omap44xx_l4_wkup__kbd,
|
&omap44xx_l4_wkup__kbd,
|
||||||
&omap44xx_l4_cfg__mailbox,
|
&omap44xx_l4_cfg__mailbox,
|
||||||
|
@ -6101,7 +6101,7 @@ static struct omap_hwmod_ocp_if *omap44xx_hwmod_ocp_ifs[] __initdata = {
|
||||||
&omap44xx_l4_cfg__cm_core,
|
&omap44xx_l4_cfg__cm_core,
|
||||||
&omap44xx_l4_wkup__prm,
|
&omap44xx_l4_wkup__prm,
|
||||||
&omap44xx_l4_wkup__scrm,
|
&omap44xx_l4_wkup__scrm,
|
||||||
&omap44xx_l3_main_2__sl2if,
|
/* &omap44xx_l3_main_2__sl2if, */
|
||||||
&omap44xx_l4_abe__slimbus1,
|
&omap44xx_l4_abe__slimbus1,
|
||||||
&omap44xx_l4_abe__slimbus1_dma,
|
&omap44xx_l4_abe__slimbus1_dma,
|
||||||
&omap44xx_l4_per__slimbus2,
|
&omap44xx_l4_per__slimbus2,
|
||||||
|
|
|
@ -260,6 +260,7 @@ static u32 notrace dmtimer_read_sched_clock(void)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_OMAP_32K_TIMER
|
||||||
/* Setup free-running counter for clocksource */
|
/* Setup free-running counter for clocksource */
|
||||||
static int __init omap2_sync32k_clocksource_init(void)
|
static int __init omap2_sync32k_clocksource_init(void)
|
||||||
{
|
{
|
||||||
|
@ -299,6 +300,12 @@ static int __init omap2_sync32k_clocksource_init(void)
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
static inline int omap2_sync32k_clocksource_init(void)
|
||||||
|
{
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static void __init omap2_gptimer_clocksource_init(int gptimer_id,
|
static void __init omap2_gptimer_clocksource_init(int gptimer_id,
|
||||||
const char *fck_source)
|
const char *fck_source)
|
||||||
|
|
|
@ -520,13 +520,14 @@ static struct platform_device hdmi_lcdc_device = {
|
||||||
};
|
};
|
||||||
|
|
||||||
/* GPIO KEY */
|
/* GPIO KEY */
|
||||||
#define GPIO_KEY(c, g, d) { .code = c, .gpio = g, .desc = d, .active_low = 1 }
|
#define GPIO_KEY(c, g, d, ...) \
|
||||||
|
{ .code = c, .gpio = g, .desc = d, .active_low = 1, __VA_ARGS__ }
|
||||||
|
|
||||||
static struct gpio_keys_button gpio_buttons[] = {
|
static struct gpio_keys_button gpio_buttons[] = {
|
||||||
GPIO_KEY(KEY_POWER, GPIO_PORT99, "SW1"),
|
GPIO_KEY(KEY_POWER, GPIO_PORT99, "SW3", .wakeup = 1),
|
||||||
GPIO_KEY(KEY_BACK, GPIO_PORT100, "SW2"),
|
GPIO_KEY(KEY_BACK, GPIO_PORT100, "SW4"),
|
||||||
GPIO_KEY(KEY_MENU, GPIO_PORT97, "SW3"),
|
GPIO_KEY(KEY_MENU, GPIO_PORT97, "SW5"),
|
||||||
GPIO_KEY(KEY_HOME, GPIO_PORT98, "SW4"),
|
GPIO_KEY(KEY_HOME, GPIO_PORT98, "SW6"),
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct gpio_keys_platform_data gpio_key_info = {
|
static struct gpio_keys_platform_data gpio_key_info = {
|
||||||
|
@ -901,8 +902,8 @@ static struct platform_device *eva_devices[] __initdata = {
|
||||||
&camera_device,
|
&camera_device,
|
||||||
&ceu0_device,
|
&ceu0_device,
|
||||||
&fsi_device,
|
&fsi_device,
|
||||||
&fsi_hdmi_device,
|
|
||||||
&fsi_wm8978_device,
|
&fsi_wm8978_device,
|
||||||
|
&fsi_hdmi_device,
|
||||||
};
|
};
|
||||||
|
|
||||||
static void __init eva_clock_init(void)
|
static void __init eva_clock_init(void)
|
||||||
|
|
|
@ -695,6 +695,7 @@ static struct platform_device usbhs0_device = {
|
||||||
* - J30 "open"
|
* - J30 "open"
|
||||||
* - modify usbhs1_get_id() USBHS_HOST -> USBHS_GADGET
|
* - modify usbhs1_get_id() USBHS_HOST -> USBHS_GADGET
|
||||||
* - add .get_vbus = usbhs_get_vbus in usbhs1_private
|
* - add .get_vbus = usbhs_get_vbus in usbhs1_private
|
||||||
|
* - check usbhs0_device(pio)/usbhs1_device(irq) order in mackerel_devices.
|
||||||
*/
|
*/
|
||||||
#define IRQ8 evt2irq(0x0300)
|
#define IRQ8 evt2irq(0x0300)
|
||||||
#define USB_PHY_MODE (1 << 4)
|
#define USB_PHY_MODE (1 << 4)
|
||||||
|
@ -1325,8 +1326,8 @@ static struct platform_device *mackerel_devices[] __initdata = {
|
||||||
&nor_flash_device,
|
&nor_flash_device,
|
||||||
&smc911x_device,
|
&smc911x_device,
|
||||||
&lcdc_device,
|
&lcdc_device,
|
||||||
&usbhs1_device,
|
|
||||||
&usbhs0_device,
|
&usbhs0_device,
|
||||||
|
&usbhs1_device,
|
||||||
&leds_device,
|
&leds_device,
|
||||||
&fsi_device,
|
&fsi_device,
|
||||||
&fsi_ak4643_device,
|
&fsi_ak4643_device,
|
||||||
|
|
|
@ -67,7 +67,7 @@ static struct smsc911x_platform_config smsc911x_platdata = {
|
||||||
|
|
||||||
static struct platform_device eth_device = {
|
static struct platform_device eth_device = {
|
||||||
.name = "smsc911x",
|
.name = "smsc911x",
|
||||||
.id = 0,
|
.id = -1,
|
||||||
.dev = {
|
.dev = {
|
||||||
.platform_data = &smsc911x_platdata,
|
.platform_data = &smsc911x_platdata,
|
||||||
},
|
},
|
||||||
|
|
|
@ -259,9 +259,9 @@ static int sh73a0_set_wake(struct irq_data *data, unsigned int on)
|
||||||
return 0; /* always allow wakeup */
|
return 0; /* always allow wakeup */
|
||||||
}
|
}
|
||||||
|
|
||||||
#define RELOC_BASE 0x1000
|
#define RELOC_BASE 0x1200
|
||||||
|
|
||||||
/* INTCA IRQ pins at INTCS + 0x1000 to make space for GIC+INTC handling */
|
/* INTCA IRQ pins at INTCS + RELOC_BASE to make space for GIC+INTC handling */
|
||||||
#define INTCS_VECT_RELOC(n, vect) INTCS_VECT((n), (vect) + RELOC_BASE)
|
#define INTCS_VECT_RELOC(n, vect) INTCS_VECT((n), (vect) + RELOC_BASE)
|
||||||
|
|
||||||
INTC_IRQ_PINS_32(intca_irq_pins, 0xe6900000,
|
INTC_IRQ_PINS_32(intca_irq_pins, 0xe6900000,
|
||||||
|
|
|
@ -63,10 +63,11 @@ static int contextidr_notifier(struct notifier_block *unused, unsigned long cmd,
|
||||||
pid = task_pid_nr(thread->task) << ASID_BITS;
|
pid = task_pid_nr(thread->task) << ASID_BITS;
|
||||||
asm volatile(
|
asm volatile(
|
||||||
" mrc p15, 0, %0, c13, c0, 1\n"
|
" mrc p15, 0, %0, c13, c0, 1\n"
|
||||||
" bfi %1, %0, #0, %2\n"
|
" and %0, %0, %2\n"
|
||||||
" mcr p15, 0, %1, c13, c0, 1\n"
|
" orr %0, %0, %1\n"
|
||||||
|
" mcr p15, 0, %0, c13, c0, 1\n"
|
||||||
: "=r" (contextidr), "+r" (pid)
|
: "=r" (contextidr), "+r" (pid)
|
||||||
: "I" (ASID_BITS));
|
: "I" (~ASID_MASK));
|
||||||
isb();
|
isb();
|
||||||
|
|
||||||
return NOTIFY_OK;
|
return NOTIFY_OK;
|
||||||
|
|
|
@ -267,17 +267,19 @@ static void __dma_free_remap(void *cpu_addr, size_t size)
|
||||||
vunmap(cpu_addr);
|
vunmap(cpu_addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define DEFAULT_DMA_COHERENT_POOL_SIZE SZ_256K
|
||||||
|
|
||||||
struct dma_pool {
|
struct dma_pool {
|
||||||
size_t size;
|
size_t size;
|
||||||
spinlock_t lock;
|
spinlock_t lock;
|
||||||
unsigned long *bitmap;
|
unsigned long *bitmap;
|
||||||
unsigned long nr_pages;
|
unsigned long nr_pages;
|
||||||
void *vaddr;
|
void *vaddr;
|
||||||
struct page *page;
|
struct page **pages;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct dma_pool atomic_pool = {
|
static struct dma_pool atomic_pool = {
|
||||||
.size = SZ_256K,
|
.size = DEFAULT_DMA_COHERENT_POOL_SIZE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int __init early_coherent_pool(char *p)
|
static int __init early_coherent_pool(char *p)
|
||||||
|
@ -287,6 +289,21 @@ static int __init early_coherent_pool(char *p)
|
||||||
}
|
}
|
||||||
early_param("coherent_pool", early_coherent_pool);
|
early_param("coherent_pool", early_coherent_pool);
|
||||||
|
|
||||||
|
void __init init_dma_coherent_pool_size(unsigned long size)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Catch any attempt to set the pool size too late.
|
||||||
|
*/
|
||||||
|
BUG_ON(atomic_pool.vaddr);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set architecture specific coherent pool size only if
|
||||||
|
* it has not been changed by kernel command line parameter.
|
||||||
|
*/
|
||||||
|
if (atomic_pool.size == DEFAULT_DMA_COHERENT_POOL_SIZE)
|
||||||
|
atomic_pool.size = size;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialise the coherent pool for atomic allocations.
|
* Initialise the coherent pool for atomic allocations.
|
||||||
*/
|
*/
|
||||||
|
@ -297,6 +314,7 @@ static int __init atomic_pool_init(void)
|
||||||
unsigned long nr_pages = pool->size >> PAGE_SHIFT;
|
unsigned long nr_pages = pool->size >> PAGE_SHIFT;
|
||||||
unsigned long *bitmap;
|
unsigned long *bitmap;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
struct page **pages;
|
||||||
void *ptr;
|
void *ptr;
|
||||||
int bitmap_size = BITS_TO_LONGS(nr_pages) * sizeof(long);
|
int bitmap_size = BITS_TO_LONGS(nr_pages) * sizeof(long);
|
||||||
|
|
||||||
|
@ -304,21 +322,31 @@ static int __init atomic_pool_init(void)
|
||||||
if (!bitmap)
|
if (!bitmap)
|
||||||
goto no_bitmap;
|
goto no_bitmap;
|
||||||
|
|
||||||
|
pages = kzalloc(nr_pages * sizeof(struct page *), GFP_KERNEL);
|
||||||
|
if (!pages)
|
||||||
|
goto no_pages;
|
||||||
|
|
||||||
if (IS_ENABLED(CONFIG_CMA))
|
if (IS_ENABLED(CONFIG_CMA))
|
||||||
ptr = __alloc_from_contiguous(NULL, pool->size, prot, &page);
|
ptr = __alloc_from_contiguous(NULL, pool->size, prot, &page);
|
||||||
else
|
else
|
||||||
ptr = __alloc_remap_buffer(NULL, pool->size, GFP_KERNEL, prot,
|
ptr = __alloc_remap_buffer(NULL, pool->size, GFP_KERNEL, prot,
|
||||||
&page, NULL);
|
&page, NULL);
|
||||||
if (ptr) {
|
if (ptr) {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < nr_pages; i++)
|
||||||
|
pages[i] = page + i;
|
||||||
|
|
||||||
spin_lock_init(&pool->lock);
|
spin_lock_init(&pool->lock);
|
||||||
pool->vaddr = ptr;
|
pool->vaddr = ptr;
|
||||||
pool->page = page;
|
pool->pages = pages;
|
||||||
pool->bitmap = bitmap;
|
pool->bitmap = bitmap;
|
||||||
pool->nr_pages = nr_pages;
|
pool->nr_pages = nr_pages;
|
||||||
pr_info("DMA: preallocated %u KiB pool for atomic coherent allocations\n",
|
pr_info("DMA: preallocated %u KiB pool for atomic coherent allocations\n",
|
||||||
(unsigned)pool->size / 1024);
|
(unsigned)pool->size / 1024);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
no_pages:
|
||||||
kfree(bitmap);
|
kfree(bitmap);
|
||||||
no_bitmap:
|
no_bitmap:
|
||||||
pr_err("DMA: failed to allocate %u KiB pool for atomic coherent allocation\n",
|
pr_err("DMA: failed to allocate %u KiB pool for atomic coherent allocation\n",
|
||||||
|
@ -443,27 +471,45 @@ static void *__alloc_from_pool(size_t size, struct page **ret_page)
|
||||||
if (pageno < pool->nr_pages) {
|
if (pageno < pool->nr_pages) {
|
||||||
bitmap_set(pool->bitmap, pageno, count);
|
bitmap_set(pool->bitmap, pageno, count);
|
||||||
ptr = pool->vaddr + PAGE_SIZE * pageno;
|
ptr = pool->vaddr + PAGE_SIZE * pageno;
|
||||||
*ret_page = pool->page + pageno;
|
*ret_page = pool->pages[pageno];
|
||||||
|
} else {
|
||||||
|
pr_err_once("ERROR: %u KiB atomic DMA coherent pool is too small!\n"
|
||||||
|
"Please increase it with coherent_pool= kernel parameter!\n",
|
||||||
|
(unsigned)pool->size / 1024);
|
||||||
}
|
}
|
||||||
spin_unlock_irqrestore(&pool->lock, flags);
|
spin_unlock_irqrestore(&pool->lock, flags);
|
||||||
|
|
||||||
return ptr;
|
return ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool __in_atomic_pool(void *start, size_t size)
|
||||||
|
{
|
||||||
|
struct dma_pool *pool = &atomic_pool;
|
||||||
|
void *end = start + size;
|
||||||
|
void *pool_start = pool->vaddr;
|
||||||
|
void *pool_end = pool->vaddr + pool->size;
|
||||||
|
|
||||||
|
if (start < pool_start || start >= pool_end)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (end <= pool_end)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
WARN(1, "Wrong coherent size(%p-%p) from atomic pool(%p-%p)\n",
|
||||||
|
start, end - 1, pool_start, pool_end - 1);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
static int __free_from_pool(void *start, size_t size)
|
static int __free_from_pool(void *start, size_t size)
|
||||||
{
|
{
|
||||||
struct dma_pool *pool = &atomic_pool;
|
struct dma_pool *pool = &atomic_pool;
|
||||||
unsigned long pageno, count;
|
unsigned long pageno, count;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
if (start < pool->vaddr || start > pool->vaddr + pool->size)
|
if (!__in_atomic_pool(start, size))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (start + size > pool->vaddr + pool->size) {
|
|
||||||
WARN(1, "freeing wrong coherent size from pool\n");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
pageno = (start - pool->vaddr) >> PAGE_SHIFT;
|
pageno = (start - pool->vaddr) >> PAGE_SHIFT;
|
||||||
count = size >> PAGE_SHIFT;
|
count = size >> PAGE_SHIFT;
|
||||||
|
|
||||||
|
@ -1090,10 +1136,22 @@ static int __iommu_remove_mapping(struct device *dev, dma_addr_t iova, size_t si
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static struct page **__atomic_get_pages(void *addr)
|
||||||
|
{
|
||||||
|
struct dma_pool *pool = &atomic_pool;
|
||||||
|
struct page **pages = pool->pages;
|
||||||
|
int offs = (addr - pool->vaddr) >> PAGE_SHIFT;
|
||||||
|
|
||||||
|
return pages + offs;
|
||||||
|
}
|
||||||
|
|
||||||
static struct page **__iommu_get_pages(void *cpu_addr, struct dma_attrs *attrs)
|
static struct page **__iommu_get_pages(void *cpu_addr, struct dma_attrs *attrs)
|
||||||
{
|
{
|
||||||
struct vm_struct *area;
|
struct vm_struct *area;
|
||||||
|
|
||||||
|
if (__in_atomic_pool(cpu_addr, PAGE_SIZE))
|
||||||
|
return __atomic_get_pages(cpu_addr);
|
||||||
|
|
||||||
if (dma_get_attr(DMA_ATTR_NO_KERNEL_MAPPING, attrs))
|
if (dma_get_attr(DMA_ATTR_NO_KERNEL_MAPPING, attrs))
|
||||||
return cpu_addr;
|
return cpu_addr;
|
||||||
|
|
||||||
|
@ -1103,6 +1161,34 @@ static struct page **__iommu_get_pages(void *cpu_addr, struct dma_attrs *attrs)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void *__iommu_alloc_atomic(struct device *dev, size_t size,
|
||||||
|
dma_addr_t *handle)
|
||||||
|
{
|
||||||
|
struct page *page;
|
||||||
|
void *addr;
|
||||||
|
|
||||||
|
addr = __alloc_from_pool(size, &page);
|
||||||
|
if (!addr)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
*handle = __iommu_create_mapping(dev, &page, size);
|
||||||
|
if (*handle == DMA_ERROR_CODE)
|
||||||
|
goto err_mapping;
|
||||||
|
|
||||||
|
return addr;
|
||||||
|
|
||||||
|
err_mapping:
|
||||||
|
__free_from_pool(addr, size);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void __iommu_free_atomic(struct device *dev, struct page **pages,
|
||||||
|
dma_addr_t handle, size_t size)
|
||||||
|
{
|
||||||
|
__iommu_remove_mapping(dev, handle, size);
|
||||||
|
__free_from_pool(page_address(pages[0]), size);
|
||||||
|
}
|
||||||
|
|
||||||
static void *arm_iommu_alloc_attrs(struct device *dev, size_t size,
|
static void *arm_iommu_alloc_attrs(struct device *dev, size_t size,
|
||||||
dma_addr_t *handle, gfp_t gfp, struct dma_attrs *attrs)
|
dma_addr_t *handle, gfp_t gfp, struct dma_attrs *attrs)
|
||||||
{
|
{
|
||||||
|
@ -1113,6 +1199,9 @@ static void *arm_iommu_alloc_attrs(struct device *dev, size_t size,
|
||||||
*handle = DMA_ERROR_CODE;
|
*handle = DMA_ERROR_CODE;
|
||||||
size = PAGE_ALIGN(size);
|
size = PAGE_ALIGN(size);
|
||||||
|
|
||||||
|
if (gfp & GFP_ATOMIC)
|
||||||
|
return __iommu_alloc_atomic(dev, size, handle);
|
||||||
|
|
||||||
pages = __iommu_alloc_buffer(dev, size, gfp);
|
pages = __iommu_alloc_buffer(dev, size, gfp);
|
||||||
if (!pages)
|
if (!pages)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -1179,6 +1268,11 @@ void arm_iommu_free_attrs(struct device *dev, size_t size, void *cpu_addr,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (__in_atomic_pool(cpu_addr, size)) {
|
||||||
|
__iommu_free_atomic(dev, pages, handle, size);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!dma_get_attr(DMA_ATTR_NO_KERNEL_MAPPING, attrs)) {
|
if (!dma_get_attr(DMA_ATTR_NO_KERNEL_MAPPING, attrs)) {
|
||||||
unmap_kernel_range((unsigned long)cpu_addr, size);
|
unmap_kernel_range((unsigned long)cpu_addr, size);
|
||||||
vunmap(cpu_addr);
|
vunmap(cpu_addr);
|
||||||
|
|
|
@ -55,6 +55,9 @@ extern void __flush_dcache_page(struct address_space *mapping, struct page *page
|
||||||
/* permanent static mappings from iotable_init() */
|
/* permanent static mappings from iotable_init() */
|
||||||
#define VM_ARM_STATIC_MAPPING 0x40000000
|
#define VM_ARM_STATIC_MAPPING 0x40000000
|
||||||
|
|
||||||
|
/* empty mapping */
|
||||||
|
#define VM_ARM_EMPTY_MAPPING 0x20000000
|
||||||
|
|
||||||
/* mapping type (attributes) for permanent static mappings */
|
/* mapping type (attributes) for permanent static mappings */
|
||||||
#define VM_ARM_MTYPE(mt) ((mt) << 20)
|
#define VM_ARM_MTYPE(mt) ((mt) << 20)
|
||||||
#define VM_ARM_MTYPE_MASK (0x1f << 20)
|
#define VM_ARM_MTYPE_MASK (0x1f << 20)
|
||||||
|
|
|
@ -807,7 +807,7 @@ static void __init pmd_empty_section_gap(unsigned long addr)
|
||||||
vm = early_alloc_aligned(sizeof(*vm), __alignof__(*vm));
|
vm = early_alloc_aligned(sizeof(*vm), __alignof__(*vm));
|
||||||
vm->addr = (void *)addr;
|
vm->addr = (void *)addr;
|
||||||
vm->size = SECTION_SIZE;
|
vm->size = SECTION_SIZE;
|
||||||
vm->flags = VM_IOREMAP | VM_ARM_STATIC_MAPPING;
|
vm->flags = VM_IOREMAP | VM_ARM_EMPTY_MAPPING;
|
||||||
vm->caller = pmd_empty_section_gap;
|
vm->caller = pmd_empty_section_gap;
|
||||||
vm_area_add_early(vm);
|
vm_area_add_early(vm);
|
||||||
}
|
}
|
||||||
|
@ -820,7 +820,7 @@ static void __init fill_pmd_gaps(void)
|
||||||
|
|
||||||
/* we're still single threaded hence no lock needed here */
|
/* we're still single threaded hence no lock needed here */
|
||||||
for (vm = vmlist; vm; vm = vm->next) {
|
for (vm = vmlist; vm; vm = vm->next) {
|
||||||
if (!(vm->flags & VM_ARM_STATIC_MAPPING))
|
if (!(vm->flags & (VM_ARM_STATIC_MAPPING | VM_ARM_EMPTY_MAPPING)))
|
||||||
continue;
|
continue;
|
||||||
addr = (unsigned long)vm->addr;
|
addr = (unsigned long)vm->addr;
|
||||||
if (addr < next)
|
if (addr < next)
|
||||||
|
@ -961,8 +961,8 @@ void __init sanity_check_meminfo(void)
|
||||||
* Check whether this memory bank would partially overlap
|
* Check whether this memory bank would partially overlap
|
||||||
* the vmalloc area.
|
* the vmalloc area.
|
||||||
*/
|
*/
|
||||||
if (__va(bank->start + bank->size) > vmalloc_min ||
|
if (__va(bank->start + bank->size - 1) >= vmalloc_min ||
|
||||||
__va(bank->start + bank->size) < __va(bank->start)) {
|
__va(bank->start + bank->size - 1) <= __va(bank->start)) {
|
||||||
unsigned long newsize = vmalloc_min - __va(bank->start);
|
unsigned long newsize = vmalloc_min - __va(bank->start);
|
||||||
printk(KERN_NOTICE "Truncating RAM at %.8llx-%.8llx "
|
printk(KERN_NOTICE "Truncating RAM at %.8llx-%.8llx "
|
||||||
"to -%.8llx (vmalloc region overlap).\n",
|
"to -%.8llx (vmalloc region overlap).\n",
|
||||||
|
|
|
@ -68,6 +68,7 @@
|
||||||
|
|
||||||
static unsigned long omap_sram_start;
|
static unsigned long omap_sram_start;
|
||||||
static void __iomem *omap_sram_base;
|
static void __iomem *omap_sram_base;
|
||||||
|
static unsigned long omap_sram_skip;
|
||||||
static unsigned long omap_sram_size;
|
static unsigned long omap_sram_size;
|
||||||
static void __iomem *omap_sram_ceil;
|
static void __iomem *omap_sram_ceil;
|
||||||
|
|
||||||
|
@ -106,6 +107,7 @@ static int is_sram_locked(void)
|
||||||
*/
|
*/
|
||||||
static void __init omap_detect_sram(void)
|
static void __init omap_detect_sram(void)
|
||||||
{
|
{
|
||||||
|
omap_sram_skip = SRAM_BOOTLOADER_SZ;
|
||||||
if (cpu_class_is_omap2()) {
|
if (cpu_class_is_omap2()) {
|
||||||
if (is_sram_locked()) {
|
if (is_sram_locked()) {
|
||||||
if (cpu_is_omap34xx()) {
|
if (cpu_is_omap34xx()) {
|
||||||
|
@ -113,6 +115,7 @@ static void __init omap_detect_sram(void)
|
||||||
if ((omap_type() == OMAP2_DEVICE_TYPE_EMU) ||
|
if ((omap_type() == OMAP2_DEVICE_TYPE_EMU) ||
|
||||||
(omap_type() == OMAP2_DEVICE_TYPE_SEC)) {
|
(omap_type() == OMAP2_DEVICE_TYPE_SEC)) {
|
||||||
omap_sram_size = 0x7000; /* 28K */
|
omap_sram_size = 0x7000; /* 28K */
|
||||||
|
omap_sram_skip += SZ_16K;
|
||||||
} else {
|
} else {
|
||||||
omap_sram_size = 0x8000; /* 32K */
|
omap_sram_size = 0x8000; /* 32K */
|
||||||
}
|
}
|
||||||
|
@ -175,8 +178,10 @@ static void __init omap_map_sram(void)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
#ifdef CONFIG_OMAP4_ERRATA_I688
|
#ifdef CONFIG_OMAP4_ERRATA_I688
|
||||||
|
if (cpu_is_omap44xx()) {
|
||||||
omap_sram_start += PAGE_SIZE;
|
omap_sram_start += PAGE_SIZE;
|
||||||
omap_sram_size -= SZ_16K;
|
omap_sram_size -= SZ_16K;
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
if (cpu_is_omap34xx()) {
|
if (cpu_is_omap34xx()) {
|
||||||
/*
|
/*
|
||||||
|
@ -203,8 +208,8 @@ static void __init omap_map_sram(void)
|
||||||
* Looks like we need to preserve some bootloader code at the
|
* Looks like we need to preserve some bootloader code at the
|
||||||
* beginning of SRAM for jumping to flash for reboot to work...
|
* beginning of SRAM for jumping to flash for reboot to work...
|
||||||
*/
|
*/
|
||||||
memset_io(omap_sram_base + SRAM_BOOTLOADER_SZ, 0,
|
memset_io(omap_sram_base + omap_sram_skip, 0,
|
||||||
omap_sram_size - SRAM_BOOTLOADER_SZ);
|
omap_sram_size - omap_sram_skip);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -218,7 +223,7 @@ void *omap_sram_push_address(unsigned long size)
|
||||||
{
|
{
|
||||||
unsigned long available, new_ceil = (unsigned long)omap_sram_ceil;
|
unsigned long available, new_ceil = (unsigned long)omap_sram_ceil;
|
||||||
|
|
||||||
available = omap_sram_ceil - (omap_sram_base + SRAM_BOOTLOADER_SZ);
|
available = omap_sram_ceil - (omap_sram_base + omap_sram_skip);
|
||||||
|
|
||||||
if (size > available) {
|
if (size > available) {
|
||||||
pr_err("Not enough space in SRAM\n");
|
pr_err("Not enough space in SRAM\n");
|
||||||
|
|
|
@ -38,6 +38,7 @@ config BLACKFIN
|
||||||
select GENERIC_ATOMIC64
|
select GENERIC_ATOMIC64
|
||||||
select GENERIC_IRQ_PROBE
|
select GENERIC_IRQ_PROBE
|
||||||
select IRQ_PER_CPU if SMP
|
select IRQ_PER_CPU if SMP
|
||||||
|
select USE_GENERIC_SMP_HELPERS if SMP
|
||||||
select HAVE_NMI_WATCHDOG if NMI_WATCHDOG
|
select HAVE_NMI_WATCHDOG if NMI_WATCHDOG
|
||||||
select GENERIC_SMP_IDLE_THREAD
|
select GENERIC_SMP_IDLE_THREAD
|
||||||
select ARCH_USES_GETTIMEOFFSET if !GENERIC_CLOCKEVENTS
|
select ARCH_USES_GETTIMEOFFSET if !GENERIC_CLOCKEVENTS
|
||||||
|
|
|
@ -20,7 +20,6 @@ endif
|
||||||
KBUILD_AFLAGS += $(call cc-option,-mno-fdpic)
|
KBUILD_AFLAGS += $(call cc-option,-mno-fdpic)
|
||||||
KBUILD_CFLAGS_MODULE += -mlong-calls
|
KBUILD_CFLAGS_MODULE += -mlong-calls
|
||||||
LDFLAGS += -m elf32bfin
|
LDFLAGS += -m elf32bfin
|
||||||
KALLSYMS += --symbol-prefix=_
|
|
||||||
|
|
||||||
KBUILD_DEFCONFIG := BF537-STAMP_defconfig
|
KBUILD_DEFCONFIG := BF537-STAMP_defconfig
|
||||||
|
|
||||||
|
|
|
@ -18,6 +18,8 @@
|
||||||
#define raw_smp_processor_id() blackfin_core_id()
|
#define raw_smp_processor_id() blackfin_core_id()
|
||||||
|
|
||||||
extern void bfin_relocate_coreb_l1_mem(void);
|
extern void bfin_relocate_coreb_l1_mem(void);
|
||||||
|
extern void arch_send_call_function_single_ipi(int cpu);
|
||||||
|
extern void arch_send_call_function_ipi_mask(const struct cpumask *mask);
|
||||||
|
|
||||||
#if defined(CONFIG_SMP) && defined(CONFIG_ICACHE_FLUSH_L1)
|
#if defined(CONFIG_SMP) && defined(CONFIG_ICACHE_FLUSH_L1)
|
||||||
asmlinkage void blackfin_icache_flush_range_l1(unsigned long *ptr);
|
asmlinkage void blackfin_icache_flush_range_l1(unsigned long *ptr);
|
||||||
|
|
|
@ -48,10 +48,13 @@ unsigned long blackfin_iflush_l1_entry[NR_CPUS];
|
||||||
|
|
||||||
struct blackfin_initial_pda __cpuinitdata initial_pda_coreb;
|
struct blackfin_initial_pda __cpuinitdata initial_pda_coreb;
|
||||||
|
|
||||||
#define BFIN_IPI_TIMER 0
|
enum ipi_message_type {
|
||||||
#define BFIN_IPI_RESCHEDULE 1
|
BFIN_IPI_TIMER,
|
||||||
#define BFIN_IPI_CALL_FUNC 2
|
BFIN_IPI_RESCHEDULE,
|
||||||
#define BFIN_IPI_CPU_STOP 3
|
BFIN_IPI_CALL_FUNC,
|
||||||
|
BFIN_IPI_CALL_FUNC_SINGLE,
|
||||||
|
BFIN_IPI_CPU_STOP,
|
||||||
|
};
|
||||||
|
|
||||||
struct blackfin_flush_data {
|
struct blackfin_flush_data {
|
||||||
unsigned long start;
|
unsigned long start;
|
||||||
|
@ -60,35 +63,20 @@ struct blackfin_flush_data {
|
||||||
|
|
||||||
void *secondary_stack;
|
void *secondary_stack;
|
||||||
|
|
||||||
|
|
||||||
struct smp_call_struct {
|
|
||||||
void (*func)(void *info);
|
|
||||||
void *info;
|
|
||||||
int wait;
|
|
||||||
cpumask_t *waitmask;
|
|
||||||
};
|
|
||||||
|
|
||||||
static struct blackfin_flush_data smp_flush_data;
|
static struct blackfin_flush_data smp_flush_data;
|
||||||
|
|
||||||
static DEFINE_SPINLOCK(stop_lock);
|
static DEFINE_SPINLOCK(stop_lock);
|
||||||
|
|
||||||
struct ipi_message {
|
|
||||||
unsigned long type;
|
|
||||||
struct smp_call_struct call_struct;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* A magic number - stress test shows this is safe for common cases */
|
/* A magic number - stress test shows this is safe for common cases */
|
||||||
#define BFIN_IPI_MSGQ_LEN 5
|
#define BFIN_IPI_MSGQ_LEN 5
|
||||||
|
|
||||||
/* Simple FIFO buffer, overflow leads to panic */
|
/* Simple FIFO buffer, overflow leads to panic */
|
||||||
struct ipi_message_queue {
|
struct ipi_data {
|
||||||
spinlock_t lock;
|
|
||||||
unsigned long count;
|
unsigned long count;
|
||||||
unsigned long head; /* head of the queue */
|
unsigned long bits;
|
||||||
struct ipi_message ipi_message[BFIN_IPI_MSGQ_LEN];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static DEFINE_PER_CPU(struct ipi_message_queue, ipi_msg_queue);
|
static DEFINE_PER_CPU(struct ipi_data, bfin_ipi);
|
||||||
|
|
||||||
static void ipi_cpu_stop(unsigned int cpu)
|
static void ipi_cpu_stop(unsigned int cpu)
|
||||||
{
|
{
|
||||||
|
@ -129,28 +117,6 @@ static void ipi_flush_icache(void *info)
|
||||||
blackfin_icache_flush_range(fdata->start, fdata->end);
|
blackfin_icache_flush_range(fdata->start, fdata->end);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ipi_call_function(unsigned int cpu, struct ipi_message *msg)
|
|
||||||
{
|
|
||||||
int wait;
|
|
||||||
void (*func)(void *info);
|
|
||||||
void *info;
|
|
||||||
func = msg->call_struct.func;
|
|
||||||
info = msg->call_struct.info;
|
|
||||||
wait = msg->call_struct.wait;
|
|
||||||
func(info);
|
|
||||||
if (wait) {
|
|
||||||
#ifdef __ARCH_SYNC_CORE_DCACHE
|
|
||||||
/*
|
|
||||||
* 'wait' usually means synchronization between CPUs.
|
|
||||||
* Invalidate D cache in case shared data was changed
|
|
||||||
* by func() to ensure cache coherence.
|
|
||||||
*/
|
|
||||||
resync_core_dcache();
|
|
||||||
#endif
|
|
||||||
cpumask_clear_cpu(cpu, msg->call_struct.waitmask);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Use IRQ_SUPPLE_0 to request reschedule.
|
/* Use IRQ_SUPPLE_0 to request reschedule.
|
||||||
* When returning from interrupt to user space,
|
* When returning from interrupt to user space,
|
||||||
* there is chance to reschedule */
|
* there is chance to reschedule */
|
||||||
|
@ -172,152 +138,95 @@ void ipi_timer(void)
|
||||||
|
|
||||||
static irqreturn_t ipi_handler_int1(int irq, void *dev_instance)
|
static irqreturn_t ipi_handler_int1(int irq, void *dev_instance)
|
||||||
{
|
{
|
||||||
struct ipi_message *msg;
|
struct ipi_data *bfin_ipi_data;
|
||||||
struct ipi_message_queue *msg_queue;
|
|
||||||
unsigned int cpu = smp_processor_id();
|
unsigned int cpu = smp_processor_id();
|
||||||
unsigned long flags;
|
unsigned long pending;
|
||||||
|
unsigned long msg;
|
||||||
|
|
||||||
platform_clear_ipi(cpu, IRQ_SUPPLE_1);
|
platform_clear_ipi(cpu, IRQ_SUPPLE_1);
|
||||||
|
|
||||||
msg_queue = &__get_cpu_var(ipi_msg_queue);
|
bfin_ipi_data = &__get_cpu_var(bfin_ipi);
|
||||||
|
|
||||||
spin_lock_irqsave(&msg_queue->lock, flags);
|
while ((pending = xchg(&bfin_ipi_data->bits, 0)) != 0) {
|
||||||
|
msg = 0;
|
||||||
|
do {
|
||||||
|
msg = find_next_bit(&pending, BITS_PER_LONG, msg + 1);
|
||||||
|
switch (msg) {
|
||||||
|
case BFIN_IPI_TIMER:
|
||||||
|
ipi_timer();
|
||||||
|
break;
|
||||||
|
case BFIN_IPI_RESCHEDULE:
|
||||||
|
scheduler_ipi();
|
||||||
|
break;
|
||||||
|
case BFIN_IPI_CALL_FUNC:
|
||||||
|
generic_smp_call_function_interrupt();
|
||||||
|
break;
|
||||||
|
|
||||||
while (msg_queue->count) {
|
case BFIN_IPI_CALL_FUNC_SINGLE:
|
||||||
msg = &msg_queue->ipi_message[msg_queue->head];
|
generic_smp_call_function_single_interrupt();
|
||||||
switch (msg->type) {
|
break;
|
||||||
case BFIN_IPI_TIMER:
|
|
||||||
ipi_timer();
|
case BFIN_IPI_CPU_STOP:
|
||||||
break;
|
ipi_cpu_stop(cpu);
|
||||||
case BFIN_IPI_RESCHEDULE:
|
break;
|
||||||
scheduler_ipi();
|
}
|
||||||
break;
|
} while (msg < BITS_PER_LONG);
|
||||||
case BFIN_IPI_CALL_FUNC:
|
|
||||||
ipi_call_function(cpu, msg);
|
smp_mb();
|
||||||
break;
|
|
||||||
case BFIN_IPI_CPU_STOP:
|
|
||||||
ipi_cpu_stop(cpu);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
printk(KERN_CRIT "CPU%u: Unknown IPI message 0x%lx\n",
|
|
||||||
cpu, msg->type);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
msg_queue->head++;
|
|
||||||
msg_queue->head %= BFIN_IPI_MSGQ_LEN;
|
|
||||||
msg_queue->count--;
|
|
||||||
}
|
}
|
||||||
spin_unlock_irqrestore(&msg_queue->lock, flags);
|
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ipi_queue_init(void)
|
static void bfin_ipi_init(void)
|
||||||
{
|
{
|
||||||
unsigned int cpu;
|
unsigned int cpu;
|
||||||
struct ipi_message_queue *msg_queue;
|
struct ipi_data *bfin_ipi_data;
|
||||||
for_each_possible_cpu(cpu) {
|
for_each_possible_cpu(cpu) {
|
||||||
msg_queue = &per_cpu(ipi_msg_queue, cpu);
|
bfin_ipi_data = &per_cpu(bfin_ipi, cpu);
|
||||||
spin_lock_init(&msg_queue->lock);
|
bfin_ipi_data->bits = 0;
|
||||||
msg_queue->count = 0;
|
bfin_ipi_data->count = 0;
|
||||||
msg_queue->head = 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void smp_send_message(cpumask_t callmap, unsigned long type,
|
void send_ipi(const struct cpumask *cpumask, enum ipi_message_type msg)
|
||||||
void (*func) (void *info), void *info, int wait)
|
|
||||||
{
|
{
|
||||||
unsigned int cpu;
|
unsigned int cpu;
|
||||||
struct ipi_message_queue *msg_queue;
|
struct ipi_data *bfin_ipi_data;
|
||||||
struct ipi_message *msg;
|
unsigned long flags;
|
||||||
unsigned long flags, next_msg;
|
|
||||||
cpumask_t waitmask; /* waitmask is shared by all cpus */
|
|
||||||
|
|
||||||
cpumask_copy(&waitmask, &callmap);
|
local_irq_save(flags);
|
||||||
for_each_cpu(cpu, &callmap) {
|
|
||||||
msg_queue = &per_cpu(ipi_msg_queue, cpu);
|
for_each_cpu(cpu, cpumask) {
|
||||||
spin_lock_irqsave(&msg_queue->lock, flags);
|
bfin_ipi_data = &per_cpu(bfin_ipi, cpu);
|
||||||
if (msg_queue->count < BFIN_IPI_MSGQ_LEN) {
|
smp_mb();
|
||||||
next_msg = (msg_queue->head + msg_queue->count)
|
set_bit(msg, &bfin_ipi_data->bits);
|
||||||
% BFIN_IPI_MSGQ_LEN;
|
bfin_ipi_data->count++;
|
||||||
msg = &msg_queue->ipi_message[next_msg];
|
|
||||||
msg->type = type;
|
|
||||||
if (type == BFIN_IPI_CALL_FUNC) {
|
|
||||||
msg->call_struct.func = func;
|
|
||||||
msg->call_struct.info = info;
|
|
||||||
msg->call_struct.wait = wait;
|
|
||||||
msg->call_struct.waitmask = &waitmask;
|
|
||||||
}
|
|
||||||
msg_queue->count++;
|
|
||||||
} else
|
|
||||||
panic("IPI message queue overflow\n");
|
|
||||||
spin_unlock_irqrestore(&msg_queue->lock, flags);
|
|
||||||
platform_send_ipi_cpu(cpu, IRQ_SUPPLE_1);
|
platform_send_ipi_cpu(cpu, IRQ_SUPPLE_1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (wait) {
|
local_irq_restore(flags);
|
||||||
while (!cpumask_empty(&waitmask))
|
|
||||||
blackfin_dcache_invalidate_range(
|
|
||||||
(unsigned long)(&waitmask),
|
|
||||||
(unsigned long)(&waitmask));
|
|
||||||
#ifdef __ARCH_SYNC_CORE_DCACHE
|
|
||||||
/*
|
|
||||||
* Invalidate D cache in case shared data was changed by
|
|
||||||
* other processors to ensure cache coherence.
|
|
||||||
*/
|
|
||||||
resync_core_dcache();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int smp_call_function(void (*func)(void *info), void *info, int wait)
|
void arch_send_call_function_single_ipi(int cpu)
|
||||||
{
|
{
|
||||||
cpumask_t callmap;
|
send_ipi(cpumask_of(cpu), BFIN_IPI_CALL_FUNC_SINGLE);
|
||||||
|
|
||||||
preempt_disable();
|
|
||||||
cpumask_copy(&callmap, cpu_online_mask);
|
|
||||||
cpumask_clear_cpu(smp_processor_id(), &callmap);
|
|
||||||
if (!cpumask_empty(&callmap))
|
|
||||||
smp_send_message(callmap, BFIN_IPI_CALL_FUNC, func, info, wait);
|
|
||||||
|
|
||||||
preempt_enable();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(smp_call_function);
|
|
||||||
|
|
||||||
int smp_call_function_single(int cpuid, void (*func) (void *info), void *info,
|
void arch_send_call_function_ipi_mask(const struct cpumask *mask)
|
||||||
int wait)
|
|
||||||
{
|
{
|
||||||
unsigned int cpu = cpuid;
|
send_ipi(mask, BFIN_IPI_CALL_FUNC);
|
||||||
cpumask_t callmap;
|
|
||||||
|
|
||||||
if (cpu_is_offline(cpu))
|
|
||||||
return 0;
|
|
||||||
cpumask_clear(&callmap);
|
|
||||||
cpumask_set_cpu(cpu, &callmap);
|
|
||||||
|
|
||||||
smp_send_message(callmap, BFIN_IPI_CALL_FUNC, func, info, wait);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(smp_call_function_single);
|
|
||||||
|
|
||||||
void smp_send_reschedule(int cpu)
|
void smp_send_reschedule(int cpu)
|
||||||
{
|
{
|
||||||
cpumask_t callmap;
|
send_ipi(cpumask_of(cpu), BFIN_IPI_RESCHEDULE);
|
||||||
/* simply trigger an ipi */
|
|
||||||
|
|
||||||
cpumask_clear(&callmap);
|
|
||||||
cpumask_set_cpu(cpu, &callmap);
|
|
||||||
|
|
||||||
smp_send_message(callmap, BFIN_IPI_RESCHEDULE, NULL, NULL, 0);
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
void smp_send_msg(const struct cpumask *mask, unsigned long type)
|
void smp_send_msg(const struct cpumask *mask, unsigned long type)
|
||||||
{
|
{
|
||||||
smp_send_message(*mask, type, NULL, NULL, 0);
|
send_ipi(mask, type);
|
||||||
}
|
}
|
||||||
|
|
||||||
void smp_timer_broadcast(const struct cpumask *mask)
|
void smp_timer_broadcast(const struct cpumask *mask)
|
||||||
|
@ -333,7 +242,7 @@ void smp_send_stop(void)
|
||||||
cpumask_copy(&callmap, cpu_online_mask);
|
cpumask_copy(&callmap, cpu_online_mask);
|
||||||
cpumask_clear_cpu(smp_processor_id(), &callmap);
|
cpumask_clear_cpu(smp_processor_id(), &callmap);
|
||||||
if (!cpumask_empty(&callmap))
|
if (!cpumask_empty(&callmap))
|
||||||
smp_send_message(callmap, BFIN_IPI_CPU_STOP, NULL, NULL, 0);
|
send_ipi(&callmap, BFIN_IPI_CPU_STOP);
|
||||||
|
|
||||||
preempt_enable();
|
preempt_enable();
|
||||||
|
|
||||||
|
@ -436,7 +345,7 @@ void __init smp_prepare_boot_cpu(void)
|
||||||
void __init smp_prepare_cpus(unsigned int max_cpus)
|
void __init smp_prepare_cpus(unsigned int max_cpus)
|
||||||
{
|
{
|
||||||
platform_prepare_cpus(max_cpus);
|
platform_prepare_cpus(max_cpus);
|
||||||
ipi_queue_init();
|
bfin_ipi_init();
|
||||||
platform_request_ipi(IRQ_SUPPLE_0, ipi_handler_int0);
|
platform_request_ipi(IRQ_SUPPLE_0, ipi_handler_int0);
|
||||||
platform_request_ipi(IRQ_SUPPLE_1, ipi_handler_int1);
|
platform_request_ipi(IRQ_SUPPLE_1, ipi_handler_int1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -386,6 +386,7 @@ extern unsigned long cpuidle_disable;
|
||||||
enum idle_boot_override {IDLE_NO_OVERRIDE = 0, IDLE_POWERSAVE_OFF};
|
enum idle_boot_override {IDLE_NO_OVERRIDE = 0, IDLE_POWERSAVE_OFF};
|
||||||
|
|
||||||
extern int powersave_nap; /* set if nap mode can be used in idle loop */
|
extern int powersave_nap; /* set if nap mode can be used in idle loop */
|
||||||
|
extern void power7_nap(void);
|
||||||
|
|
||||||
#ifdef CONFIG_PSERIES_IDLE
|
#ifdef CONFIG_PSERIES_IDLE
|
||||||
extern void update_smt_snooze_delay(int snooze);
|
extern void update_smt_snooze_delay(int snooze);
|
||||||
|
|
|
@ -76,6 +76,7 @@ int main(void)
|
||||||
DEFINE(SIGSEGV, SIGSEGV);
|
DEFINE(SIGSEGV, SIGSEGV);
|
||||||
DEFINE(NMI_MASK, NMI_MASK);
|
DEFINE(NMI_MASK, NMI_MASK);
|
||||||
DEFINE(THREAD_DSCR, offsetof(struct thread_struct, dscr));
|
DEFINE(THREAD_DSCR, offsetof(struct thread_struct, dscr));
|
||||||
|
DEFINE(THREAD_DSCR_INHERIT, offsetof(struct thread_struct, dscr_inherit));
|
||||||
#else
|
#else
|
||||||
DEFINE(THREAD_INFO, offsetof(struct task_struct, stack));
|
DEFINE(THREAD_INFO, offsetof(struct task_struct, stack));
|
||||||
#endif /* CONFIG_PPC64 */
|
#endif /* CONFIG_PPC64 */
|
||||||
|
|
|
@ -28,6 +28,8 @@ void doorbell_setup_this_cpu(void)
|
||||||
|
|
||||||
void doorbell_cause_ipi(int cpu, unsigned long data)
|
void doorbell_cause_ipi(int cpu, unsigned long data)
|
||||||
{
|
{
|
||||||
|
/* Order previous accesses vs. msgsnd, which is treated as a store */
|
||||||
|
mb();
|
||||||
ppc_msgsnd(PPC_DBELL, 0, data);
|
ppc_msgsnd(PPC_DBELL, 0, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -370,6 +370,12 @@ _GLOBAL(ret_from_fork)
|
||||||
li r3,0
|
li r3,0
|
||||||
b syscall_exit
|
b syscall_exit
|
||||||
|
|
||||||
|
.section ".toc","aw"
|
||||||
|
DSCR_DEFAULT:
|
||||||
|
.tc dscr_default[TC],dscr_default
|
||||||
|
|
||||||
|
.section ".text"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This routine switches between two different tasks. The process
|
* This routine switches between two different tasks. The process
|
||||||
* state of one is saved on its kernel stack. Then the state
|
* state of one is saved on its kernel stack. Then the state
|
||||||
|
@ -509,9 +515,6 @@ END_MMU_FTR_SECTION_IFSET(MMU_FTR_1T_SEGMENT)
|
||||||
mr r1,r8 /* start using new stack pointer */
|
mr r1,r8 /* start using new stack pointer */
|
||||||
std r7,PACAKSAVE(r13)
|
std r7,PACAKSAVE(r13)
|
||||||
|
|
||||||
ld r6,_CCR(r1)
|
|
||||||
mtcrf 0xFF,r6
|
|
||||||
|
|
||||||
#ifdef CONFIG_ALTIVEC
|
#ifdef CONFIG_ALTIVEC
|
||||||
BEGIN_FTR_SECTION
|
BEGIN_FTR_SECTION
|
||||||
ld r0,THREAD_VRSAVE(r4)
|
ld r0,THREAD_VRSAVE(r4)
|
||||||
|
@ -520,14 +523,22 @@ END_FTR_SECTION_IFSET(CPU_FTR_ALTIVEC)
|
||||||
#endif /* CONFIG_ALTIVEC */
|
#endif /* CONFIG_ALTIVEC */
|
||||||
#ifdef CONFIG_PPC64
|
#ifdef CONFIG_PPC64
|
||||||
BEGIN_FTR_SECTION
|
BEGIN_FTR_SECTION
|
||||||
|
lwz r6,THREAD_DSCR_INHERIT(r4)
|
||||||
|
ld r7,DSCR_DEFAULT@toc(2)
|
||||||
ld r0,THREAD_DSCR(r4)
|
ld r0,THREAD_DSCR(r4)
|
||||||
cmpd r0,r25
|
cmpwi r6,0
|
||||||
beq 1f
|
bne 1f
|
||||||
|
ld r0,0(r7)
|
||||||
|
1: cmpd r0,r25
|
||||||
|
beq 2f
|
||||||
mtspr SPRN_DSCR,r0
|
mtspr SPRN_DSCR,r0
|
||||||
1:
|
2:
|
||||||
END_FTR_SECTION_IFSET(CPU_FTR_DSCR)
|
END_FTR_SECTION_IFSET(CPU_FTR_DSCR)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
ld r6,_CCR(r1)
|
||||||
|
mtcrf 0xFF,r6
|
||||||
|
|
||||||
/* r3-r13 are destroyed -- Cort */
|
/* r3-r13 are destroyed -- Cort */
|
||||||
REST_8GPRS(14, r1)
|
REST_8GPRS(14, r1)
|
||||||
REST_10GPRS(22, r1)
|
REST_10GPRS(22, r1)
|
||||||
|
|
|
@ -186,7 +186,7 @@ hardware_interrupt_hv:
|
||||||
KVM_HANDLER_PR(PACA_EXGEN, EXC_STD, 0x800)
|
KVM_HANDLER_PR(PACA_EXGEN, EXC_STD, 0x800)
|
||||||
|
|
||||||
MASKABLE_EXCEPTION_PSERIES(0x900, 0x900, decrementer)
|
MASKABLE_EXCEPTION_PSERIES(0x900, 0x900, decrementer)
|
||||||
MASKABLE_EXCEPTION_HV(0x980, 0x982, decrementer)
|
STD_EXCEPTION_HV(0x980, 0x982, hdecrementer)
|
||||||
|
|
||||||
STD_EXCEPTION_PSERIES(0xa00, 0xa00, trap_0a)
|
STD_EXCEPTION_PSERIES(0xa00, 0xa00, trap_0a)
|
||||||
KVM_HANDLER_PR(PACA_EXGEN, EXC_STD, 0xa00)
|
KVM_HANDLER_PR(PACA_EXGEN, EXC_STD, 0xa00)
|
||||||
|
@ -486,6 +486,7 @@ machine_check_common:
|
||||||
|
|
||||||
STD_EXCEPTION_COMMON_ASYNC(0x500, hardware_interrupt, do_IRQ)
|
STD_EXCEPTION_COMMON_ASYNC(0x500, hardware_interrupt, do_IRQ)
|
||||||
STD_EXCEPTION_COMMON_ASYNC(0x900, decrementer, .timer_interrupt)
|
STD_EXCEPTION_COMMON_ASYNC(0x900, decrementer, .timer_interrupt)
|
||||||
|
STD_EXCEPTION_COMMON(0x980, hdecrementer, .hdec_interrupt)
|
||||||
STD_EXCEPTION_COMMON(0xa00, trap_0a, .unknown_exception)
|
STD_EXCEPTION_COMMON(0xa00, trap_0a, .unknown_exception)
|
||||||
STD_EXCEPTION_COMMON(0xb00, trap_0b, .unknown_exception)
|
STD_EXCEPTION_COMMON(0xb00, trap_0b, .unknown_exception)
|
||||||
STD_EXCEPTION_COMMON(0xd00, single_step, .single_step_exception)
|
STD_EXCEPTION_COMMON(0xd00, single_step, .single_step_exception)
|
||||||
|
|
|
@ -28,7 +28,9 @@ _GLOBAL(power7_idle)
|
||||||
lwz r4,ADDROFF(powersave_nap)(r3)
|
lwz r4,ADDROFF(powersave_nap)(r3)
|
||||||
cmpwi 0,r4,0
|
cmpwi 0,r4,0
|
||||||
beqlr
|
beqlr
|
||||||
|
/* fall through */
|
||||||
|
|
||||||
|
_GLOBAL(power7_nap)
|
||||||
/* NAP is a state loss, we create a regs frame on the
|
/* NAP is a state loss, we create a regs frame on the
|
||||||
* stack, fill it up with the state we care about and
|
* stack, fill it up with the state we care about and
|
||||||
* stick a pointer to it in PACAR1. We really only
|
* stick a pointer to it in PACAR1. We really only
|
||||||
|
|
|
@ -802,16 +802,8 @@ int copy_thread(unsigned long clone_flags, unsigned long usp,
|
||||||
#endif /* CONFIG_PPC_STD_MMU_64 */
|
#endif /* CONFIG_PPC_STD_MMU_64 */
|
||||||
#ifdef CONFIG_PPC64
|
#ifdef CONFIG_PPC64
|
||||||
if (cpu_has_feature(CPU_FTR_DSCR)) {
|
if (cpu_has_feature(CPU_FTR_DSCR)) {
|
||||||
if (current->thread.dscr_inherit) {
|
p->thread.dscr_inherit = current->thread.dscr_inherit;
|
||||||
p->thread.dscr_inherit = 1;
|
p->thread.dscr = current->thread.dscr;
|
||||||
p->thread.dscr = current->thread.dscr;
|
|
||||||
} else if (0 != dscr_default) {
|
|
||||||
p->thread.dscr_inherit = 1;
|
|
||||||
p->thread.dscr = dscr_default;
|
|
||||||
} else {
|
|
||||||
p->thread.dscr_inherit = 0;
|
|
||||||
p->thread.dscr = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -198,8 +198,15 @@ void smp_muxed_ipi_message_pass(int cpu, int msg)
|
||||||
struct cpu_messages *info = &per_cpu(ipi_message, cpu);
|
struct cpu_messages *info = &per_cpu(ipi_message, cpu);
|
||||||
char *message = (char *)&info->messages;
|
char *message = (char *)&info->messages;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Order previous accesses before accesses in the IPI handler.
|
||||||
|
*/
|
||||||
|
smp_mb();
|
||||||
message[msg] = 1;
|
message[msg] = 1;
|
||||||
mb();
|
/*
|
||||||
|
* cause_ipi functions are required to include a full barrier
|
||||||
|
* before doing whatever causes the IPI.
|
||||||
|
*/
|
||||||
smp_ops->cause_ipi(cpu, info->data);
|
smp_ops->cause_ipi(cpu, info->data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -211,7 +218,7 @@ irqreturn_t smp_ipi_demux(void)
|
||||||
mb(); /* order any irq clear */
|
mb(); /* order any irq clear */
|
||||||
|
|
||||||
do {
|
do {
|
||||||
all = xchg_local(&info->messages, 0);
|
all = xchg(&info->messages, 0);
|
||||||
|
|
||||||
#ifdef __BIG_ENDIAN
|
#ifdef __BIG_ENDIAN
|
||||||
if (all & (1 << (24 - 8 * PPC_MSG_CALL_FUNCTION)))
|
if (all & (1 << (24 - 8 * PPC_MSG_CALL_FUNCTION)))
|
||||||
|
|
|
@ -194,6 +194,14 @@ static ssize_t show_dscr_default(struct device *dev,
|
||||||
return sprintf(buf, "%lx\n", dscr_default);
|
return sprintf(buf, "%lx\n", dscr_default);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void update_dscr(void *dummy)
|
||||||
|
{
|
||||||
|
if (!current->thread.dscr_inherit) {
|
||||||
|
current->thread.dscr = dscr_default;
|
||||||
|
mtspr(SPRN_DSCR, dscr_default);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static ssize_t __used store_dscr_default(struct device *dev,
|
static ssize_t __used store_dscr_default(struct device *dev,
|
||||||
struct device_attribute *attr, const char *buf,
|
struct device_attribute *attr, const char *buf,
|
||||||
size_t count)
|
size_t count)
|
||||||
|
@ -206,6 +214,8 @@ static ssize_t __used store_dscr_default(struct device *dev,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
dscr_default = val;
|
dscr_default = val;
|
||||||
|
|
||||||
|
on_each_cpu(update_dscr, NULL, 1);
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -535,6 +535,15 @@ void timer_interrupt(struct pt_regs * regs)
|
||||||
trace_timer_interrupt_exit(regs);
|
trace_timer_interrupt_exit(regs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Hypervisor decrementer interrupts shouldn't occur but are sometimes
|
||||||
|
* left pending on exit from a KVM guest. We don't need to do anything
|
||||||
|
* to clear them, as they are edge-triggered.
|
||||||
|
*/
|
||||||
|
void hdec_interrupt(struct pt_regs *regs)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_SUSPEND
|
#ifdef CONFIG_SUSPEND
|
||||||
static void generic_suspend_disable_irqs(void)
|
static void generic_suspend_disable_irqs(void)
|
||||||
{
|
{
|
||||||
|
|
|
@ -972,8 +972,9 @@ static int emulate_instruction(struct pt_regs *regs)
|
||||||
cpu_has_feature(CPU_FTR_DSCR)) {
|
cpu_has_feature(CPU_FTR_DSCR)) {
|
||||||
PPC_WARN_EMULATED(mtdscr, regs);
|
PPC_WARN_EMULATED(mtdscr, regs);
|
||||||
rd = (instword >> 21) & 0x1f;
|
rd = (instword >> 21) & 0x1f;
|
||||||
mtspr(SPRN_DSCR, regs->gpr[rd]);
|
current->thread.dscr = regs->gpr[rd];
|
||||||
current->thread.dscr_inherit = 1;
|
current->thread.dscr_inherit = 1;
|
||||||
|
mtspr(SPRN_DSCR, current->thread.dscr);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -20,7 +20,7 @@ int patch_instruction(unsigned int *addr, unsigned int instr)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = __put_user(instr, addr);
|
__put_user_size(instr, addr, 4, err);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
asm ("dcbst 0, %0; sync; icbi 0,%0; sync; isync" : : "r" (addr));
|
asm ("dcbst 0, %0; sync; icbi 0,%0; sync; isync" : : "r" (addr));
|
||||||
|
|
|
@ -1436,11 +1436,11 @@ static long vphn_get_associativity(unsigned long cpu,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Update the node maps and sysfs entries for each cpu whose home node
|
* Update the node maps and sysfs entries for each cpu whose home node
|
||||||
* has changed.
|
* has changed. Returns 1 when the topology has changed, and 0 otherwise.
|
||||||
*/
|
*/
|
||||||
int arch_update_cpu_topology(void)
|
int arch_update_cpu_topology(void)
|
||||||
{
|
{
|
||||||
int cpu, nid, old_nid;
|
int cpu, nid, old_nid, changed = 0;
|
||||||
unsigned int associativity[VPHN_ASSOC_BUFSIZE] = {0};
|
unsigned int associativity[VPHN_ASSOC_BUFSIZE] = {0};
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
|
|
||||||
|
@ -1466,9 +1466,10 @@ int arch_update_cpu_topology(void)
|
||||||
dev = get_cpu_device(cpu);
|
dev = get_cpu_device(cpu);
|
||||||
if (dev)
|
if (dev)
|
||||||
kobject_uevent(&dev->kobj, KOBJ_CHANGE);
|
kobject_uevent(&dev->kobj, KOBJ_CHANGE);
|
||||||
|
changed = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return changed;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void topology_work_fn(struct work_struct *work)
|
static void topology_work_fn(struct work_struct *work)
|
||||||
|
|
|
@ -106,14 +106,6 @@ static void pnv_smp_cpu_kill_self(void)
|
||||||
{
|
{
|
||||||
unsigned int cpu;
|
unsigned int cpu;
|
||||||
|
|
||||||
/* If powersave_nap is enabled, use NAP mode, else just
|
|
||||||
* spin aimlessly
|
|
||||||
*/
|
|
||||||
if (!powersave_nap) {
|
|
||||||
generic_mach_cpu_die();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Standard hot unplug procedure */
|
/* Standard hot unplug procedure */
|
||||||
local_irq_disable();
|
local_irq_disable();
|
||||||
idle_task_exit();
|
idle_task_exit();
|
||||||
|
@ -128,7 +120,7 @@ static void pnv_smp_cpu_kill_self(void)
|
||||||
*/
|
*/
|
||||||
mtspr(SPRN_LPCR, mfspr(SPRN_LPCR) & ~(u64)LPCR_PECE1);
|
mtspr(SPRN_LPCR, mfspr(SPRN_LPCR) & ~(u64)LPCR_PECE1);
|
||||||
while (!generic_check_cpu_restart(cpu)) {
|
while (!generic_check_cpu_restart(cpu)) {
|
||||||
power7_idle();
|
power7_nap();
|
||||||
if (!generic_check_cpu_restart(cpu)) {
|
if (!generic_check_cpu_restart(cpu)) {
|
||||||
DBG("CPU%d Unexpected exit while offline !\n", cpu);
|
DBG("CPU%d Unexpected exit while offline !\n", cpu);
|
||||||
/* We may be getting an IPI, so we re-enable
|
/* We may be getting an IPI, so we re-enable
|
||||||
|
|
|
@ -65,7 +65,11 @@ static inline void icp_hv_set_xirr(unsigned int value)
|
||||||
static inline void icp_hv_set_qirr(int n_cpu , u8 value)
|
static inline void icp_hv_set_qirr(int n_cpu , u8 value)
|
||||||
{
|
{
|
||||||
int hw_cpu = get_hard_smp_processor_id(n_cpu);
|
int hw_cpu = get_hard_smp_processor_id(n_cpu);
|
||||||
long rc = plpar_hcall_norets(H_IPI, hw_cpu, value);
|
long rc;
|
||||||
|
|
||||||
|
/* Make sure all previous accesses are ordered before IPI sending */
|
||||||
|
mb();
|
||||||
|
rc = plpar_hcall_norets(H_IPI, hw_cpu, value);
|
||||||
if (rc != H_SUCCESS) {
|
if (rc != H_SUCCESS) {
|
||||||
pr_err("%s: bad return code qirr cpu=%d hw_cpu=%d mfrr=0x%x "
|
pr_err("%s: bad return code qirr cpu=%d hw_cpu=%d mfrr=0x%x "
|
||||||
"returned %ld\n", __func__, n_cpu, hw_cpu, value, rc);
|
"returned %ld\n", __func__, n_cpu, hw_cpu, value, rc);
|
||||||
|
|
|
@ -169,7 +169,7 @@ static ssize_t hw_interval_write(struct file *file, char const __user *buf,
|
||||||
if (*offset)
|
if (*offset)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
||||||
if (retval)
|
if (retval <= 0)
|
||||||
return retval;
|
return retval;
|
||||||
if (val < oprofile_min_interval)
|
if (val < oprofile_min_interval)
|
||||||
oprofile_hw_interval = oprofile_min_interval;
|
oprofile_hw_interval = oprofile_min_interval;
|
||||||
|
@ -212,7 +212,7 @@ static ssize_t hwsampler_zero_write(struct file *file, char const __user *buf,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
||||||
if (retval)
|
if (retval <= 0)
|
||||||
return retval;
|
return retval;
|
||||||
if (val != 0)
|
if (val != 0)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -243,7 +243,7 @@ static ssize_t hwsampler_kernel_write(struct file *file, char const __user *buf,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
||||||
if (retval)
|
if (retval <= 0)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
if (val != 0 && val != 1)
|
if (val != 0 && val != 1)
|
||||||
|
@ -278,7 +278,7 @@ static ssize_t hwsampler_user_write(struct file *file, char const __user *buf,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
||||||
if (retval)
|
if (retval <= 0)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
if (val != 0 && val != 1)
|
if (val != 0 && val != 1)
|
||||||
|
@ -317,7 +317,7 @@ static ssize_t timer_enabled_write(struct file *file, char const __user *buf,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
retval = oprofilefs_ulong_from_user(&val, buf, count);
|
||||||
if (retval)
|
if (retval <= 0)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
if (val != 0 && val != 1)
|
if (val != 0 && val != 1)
|
||||||
|
|
|
@ -114,7 +114,7 @@ static void deliver_alarm(void)
|
||||||
skew += this_tick - last_tick;
|
skew += this_tick - last_tick;
|
||||||
|
|
||||||
while (skew >= one_tick) {
|
while (skew >= one_tick) {
|
||||||
alarm_handler(SIGVTALRM, NULL);
|
alarm_handler(SIGVTALRM, NULL, NULL);
|
||||||
skew -= one_tick;
|
skew -= one_tick;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2008,6 +2008,7 @@ __init int intel_pmu_init(void)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 28: /* Atom */
|
case 28: /* Atom */
|
||||||
|
case 54: /* Cedariew */
|
||||||
memcpy(hw_cache_event_ids, atom_hw_cache_event_ids,
|
memcpy(hw_cache_event_ids, atom_hw_cache_event_ids,
|
||||||
sizeof(hw_cache_event_ids));
|
sizeof(hw_cache_event_ids));
|
||||||
|
|
||||||
|
|
|
@ -686,7 +686,8 @@ void intel_pmu_lbr_init_atom(void)
|
||||||
* to have an operational LBR which can freeze
|
* to have an operational LBR which can freeze
|
||||||
* on PMU interrupt
|
* on PMU interrupt
|
||||||
*/
|
*/
|
||||||
if (boot_cpu_data.x86_mask < 10) {
|
if (boot_cpu_data.x86_model == 28
|
||||||
|
&& boot_cpu_data.x86_mask < 10) {
|
||||||
pr_cont("LBR disabled due to erratum");
|
pr_cont("LBR disabled due to erratum");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -225,6 +225,9 @@ static ssize_t microcode_write(struct file *file, const char __user *buf,
|
||||||
if (do_microcode_update(buf, len) == 0)
|
if (do_microcode_update(buf, len) == 0)
|
||||||
ret = (ssize_t)len;
|
ret = (ssize_t)len;
|
||||||
|
|
||||||
|
if (ret > 0)
|
||||||
|
perf_check_microcode();
|
||||||
|
|
||||||
mutex_unlock(µcode_mutex);
|
mutex_unlock(µcode_mutex);
|
||||||
put_online_cpus();
|
put_online_cpus();
|
||||||
|
|
||||||
|
|
|
@ -318,7 +318,7 @@ static void pic_ioport_write(void *opaque, u32 addr, u32 val)
|
||||||
if (val & 0x10) {
|
if (val & 0x10) {
|
||||||
u8 edge_irr = s->irr & ~s->elcr;
|
u8 edge_irr = s->irr & ~s->elcr;
|
||||||
int i;
|
int i;
|
||||||
bool found;
|
bool found = false;
|
||||||
struct kvm_vcpu *vcpu;
|
struct kvm_vcpu *vcpu;
|
||||||
|
|
||||||
s->init4 = val & 1;
|
s->init4 = val & 1;
|
||||||
|
|
|
@ -3619,6 +3619,7 @@ static void seg_setup(int seg)
|
||||||
|
|
||||||
static int alloc_apic_access_page(struct kvm *kvm)
|
static int alloc_apic_access_page(struct kvm *kvm)
|
||||||
{
|
{
|
||||||
|
struct page *page;
|
||||||
struct kvm_userspace_memory_region kvm_userspace_mem;
|
struct kvm_userspace_memory_region kvm_userspace_mem;
|
||||||
int r = 0;
|
int r = 0;
|
||||||
|
|
||||||
|
@ -3633,7 +3634,13 @@ static int alloc_apic_access_page(struct kvm *kvm)
|
||||||
if (r)
|
if (r)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
kvm->arch.apic_access_page = gfn_to_page(kvm, 0xfee00);
|
page = gfn_to_page(kvm, 0xfee00);
|
||||||
|
if (is_error_page(page)) {
|
||||||
|
r = -EFAULT;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
kvm->arch.apic_access_page = page;
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&kvm->slots_lock);
|
mutex_unlock(&kvm->slots_lock);
|
||||||
return r;
|
return r;
|
||||||
|
@ -3641,6 +3648,7 @@ out:
|
||||||
|
|
||||||
static int alloc_identity_pagetable(struct kvm *kvm)
|
static int alloc_identity_pagetable(struct kvm *kvm)
|
||||||
{
|
{
|
||||||
|
struct page *page;
|
||||||
struct kvm_userspace_memory_region kvm_userspace_mem;
|
struct kvm_userspace_memory_region kvm_userspace_mem;
|
||||||
int r = 0;
|
int r = 0;
|
||||||
|
|
||||||
|
@ -3656,8 +3664,13 @@ static int alloc_identity_pagetable(struct kvm *kvm)
|
||||||
if (r)
|
if (r)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
kvm->arch.ept_identity_pagetable = gfn_to_page(kvm,
|
page = gfn_to_page(kvm, kvm->arch.ept_identity_map_addr >> PAGE_SHIFT);
|
||||||
kvm->arch.ept_identity_map_addr >> PAGE_SHIFT);
|
if (is_error_page(page)) {
|
||||||
|
r = -EFAULT;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
kvm->arch.ept_identity_pagetable = page;
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&kvm->slots_lock);
|
mutex_unlock(&kvm->slots_lock);
|
||||||
return r;
|
return r;
|
||||||
|
@ -6575,7 +6588,7 @@ static void vmx_cpuid_update(struct kvm_vcpu *vcpu)
|
||||||
/* Exposing INVPCID only when PCID is exposed */
|
/* Exposing INVPCID only when PCID is exposed */
|
||||||
best = kvm_find_cpuid_entry(vcpu, 0x7, 0);
|
best = kvm_find_cpuid_entry(vcpu, 0x7, 0);
|
||||||
if (vmx_invpcid_supported() &&
|
if (vmx_invpcid_supported() &&
|
||||||
best && (best->ecx & bit(X86_FEATURE_INVPCID)) &&
|
best && (best->ebx & bit(X86_FEATURE_INVPCID)) &&
|
||||||
guest_cpuid_has_pcid(vcpu)) {
|
guest_cpuid_has_pcid(vcpu)) {
|
||||||
exec_control |= SECONDARY_EXEC_ENABLE_INVPCID;
|
exec_control |= SECONDARY_EXEC_ENABLE_INVPCID;
|
||||||
vmcs_write32(SECONDARY_VM_EXEC_CONTROL,
|
vmcs_write32(SECONDARY_VM_EXEC_CONTROL,
|
||||||
|
@ -6585,7 +6598,7 @@ static void vmx_cpuid_update(struct kvm_vcpu *vcpu)
|
||||||
vmcs_write32(SECONDARY_VM_EXEC_CONTROL,
|
vmcs_write32(SECONDARY_VM_EXEC_CONTROL,
|
||||||
exec_control);
|
exec_control);
|
||||||
if (best)
|
if (best)
|
||||||
best->ecx &= ~bit(X86_FEATURE_INVPCID);
|
best->ebx &= ~bit(X86_FEATURE_INVPCID);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5113,17 +5113,20 @@ static void post_kvm_run_save(struct kvm_vcpu *vcpu)
|
||||||
!kvm_event_needs_reinjection(vcpu);
|
!kvm_event_needs_reinjection(vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vapic_enter(struct kvm_vcpu *vcpu)
|
static int vapic_enter(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
struct kvm_lapic *apic = vcpu->arch.apic;
|
struct kvm_lapic *apic = vcpu->arch.apic;
|
||||||
struct page *page;
|
struct page *page;
|
||||||
|
|
||||||
if (!apic || !apic->vapic_addr)
|
if (!apic || !apic->vapic_addr)
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
page = gfn_to_page(vcpu->kvm, apic->vapic_addr >> PAGE_SHIFT);
|
page = gfn_to_page(vcpu->kvm, apic->vapic_addr >> PAGE_SHIFT);
|
||||||
|
if (is_error_page(page))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
vcpu->arch.apic->vapic_page = page;
|
vcpu->arch.apic->vapic_page = page;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vapic_exit(struct kvm_vcpu *vcpu)
|
static void vapic_exit(struct kvm_vcpu *vcpu)
|
||||||
|
@ -5430,7 +5433,11 @@ static int __vcpu_run(struct kvm_vcpu *vcpu)
|
||||||
}
|
}
|
||||||
|
|
||||||
vcpu->srcu_idx = srcu_read_lock(&kvm->srcu);
|
vcpu->srcu_idx = srcu_read_lock(&kvm->srcu);
|
||||||
vapic_enter(vcpu);
|
r = vapic_enter(vcpu);
|
||||||
|
if (r) {
|
||||||
|
srcu_read_unlock(&kvm->srcu, vcpu->srcu_idx);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
r = 1;
|
r = 1;
|
||||||
while (r > 0) {
|
while (r > 0) {
|
||||||
|
|
|
@ -1283,7 +1283,7 @@ static void xen_flush_tlb_others(const struct cpumask *cpus,
|
||||||
cpumask_clear_cpu(smp_processor_id(), to_cpumask(args->mask));
|
cpumask_clear_cpu(smp_processor_id(), to_cpumask(args->mask));
|
||||||
|
|
||||||
args->op.cmd = MMUEXT_TLB_FLUSH_MULTI;
|
args->op.cmd = MMUEXT_TLB_FLUSH_MULTI;
|
||||||
if (start != TLB_FLUSH_ALL && (end - start) <= PAGE_SIZE) {
|
if (end != TLB_FLUSH_ALL && (end - start) <= PAGE_SIZE) {
|
||||||
args->op.cmd = MMUEXT_INVLPG_MULTI;
|
args->op.cmd = MMUEXT_INVLPG_MULTI;
|
||||||
args->op.arg1.linear_addr = start;
|
args->op.arg1.linear_addr = start;
|
||||||
}
|
}
|
||||||
|
|
|
@ -599,7 +599,7 @@ bool __init early_can_reuse_p2m_middle(unsigned long set_pfn, unsigned long set_
|
||||||
if (p2m_index(set_pfn))
|
if (p2m_index(set_pfn))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
for (pfn = 0; pfn <= MAX_DOMAIN_PAGES; pfn += P2M_PER_PAGE) {
|
for (pfn = 0; pfn < MAX_DOMAIN_PAGES; pfn += P2M_PER_PAGE) {
|
||||||
topidx = p2m_top_index(pfn);
|
topidx = p2m_top_index(pfn);
|
||||||
|
|
||||||
if (!p2m_top[topidx])
|
if (!p2m_top[topidx])
|
||||||
|
|
|
@ -336,7 +336,7 @@ static int crypto_authenc_genicv(struct aead_request *req, u8 *iv,
|
||||||
cryptlen += ivsize;
|
cryptlen += ivsize;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sg_is_last(assoc)) {
|
if (req->assoclen && sg_is_last(assoc)) {
|
||||||
authenc_ahash_fn = crypto_authenc_ahash;
|
authenc_ahash_fn = crypto_authenc_ahash;
|
||||||
sg_init_table(asg, 2);
|
sg_init_table(asg, 2);
|
||||||
sg_set_page(asg, sg_page(assoc), assoc->length, assoc->offset);
|
sg_set_page(asg, sg_page(assoc), assoc->length, assoc->offset);
|
||||||
|
@ -490,7 +490,7 @@ static int crypto_authenc_iverify(struct aead_request *req, u8 *iv,
|
||||||
cryptlen += ivsize;
|
cryptlen += ivsize;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sg_is_last(assoc)) {
|
if (req->assoclen && sg_is_last(assoc)) {
|
||||||
authenc_ahash_fn = crypto_authenc_ahash;
|
authenc_ahash_fn = crypto_authenc_ahash;
|
||||||
sg_init_table(asg, 2);
|
sg_init_table(asg, 2);
|
||||||
sg_set_page(asg, sg_page(assoc), assoc->length, assoc->offset);
|
sg_set_page(asg, sg_page(assoc), assoc->length, assoc->offset);
|
||||||
|
|
|
@ -237,6 +237,16 @@ static int __acpi_bus_get_power(struct acpi_device *device, int *state)
|
||||||
} else if (result == ACPI_STATE_D3_HOT) {
|
} else if (result == ACPI_STATE_D3_HOT) {
|
||||||
result = ACPI_STATE_D3;
|
result = ACPI_STATE_D3;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we were unsure about the device parent's power state up to this
|
||||||
|
* point, the fact that the device is in D0 implies that the parent has
|
||||||
|
* to be in D0 too.
|
||||||
|
*/
|
||||||
|
if (device->parent && device->parent->power.state == ACPI_STATE_UNKNOWN
|
||||||
|
&& result == ACPI_STATE_D0)
|
||||||
|
device->parent->power.state = ACPI_STATE_D0;
|
||||||
|
|
||||||
*state = result;
|
*state = result;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
|
|
|
@ -107,6 +107,7 @@ struct acpi_power_resource {
|
||||||
|
|
||||||
/* List of devices relying on this power resource */
|
/* List of devices relying on this power resource */
|
||||||
struct acpi_power_resource_device *devices;
|
struct acpi_power_resource_device *devices;
|
||||||
|
struct mutex devices_lock;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct list_head acpi_power_resource_list;
|
static struct list_head acpi_power_resource_list;
|
||||||
|
@ -225,7 +226,6 @@ static void acpi_power_on_device(struct acpi_power_managed_device *device)
|
||||||
|
|
||||||
static int __acpi_power_on(struct acpi_power_resource *resource)
|
static int __acpi_power_on(struct acpi_power_resource *resource)
|
||||||
{
|
{
|
||||||
struct acpi_power_resource_device *device_list = resource->devices;
|
|
||||||
acpi_status status = AE_OK;
|
acpi_status status = AE_OK;
|
||||||
|
|
||||||
status = acpi_evaluate_object(resource->device->handle, "_ON", NULL, NULL);
|
status = acpi_evaluate_object(resource->device->handle, "_ON", NULL, NULL);
|
||||||
|
@ -238,19 +238,15 @@ static int __acpi_power_on(struct acpi_power_resource *resource)
|
||||||
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Power resource [%s] turned on\n",
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Power resource [%s] turned on\n",
|
||||||
resource->name));
|
resource->name));
|
||||||
|
|
||||||
while (device_list) {
|
|
||||||
acpi_power_on_device(device_list->device);
|
|
||||||
|
|
||||||
device_list = device_list->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int acpi_power_on(acpi_handle handle)
|
static int acpi_power_on(acpi_handle handle)
|
||||||
{
|
{
|
||||||
int result = 0;
|
int result = 0;
|
||||||
|
bool resume_device = false;
|
||||||
struct acpi_power_resource *resource = NULL;
|
struct acpi_power_resource *resource = NULL;
|
||||||
|
struct acpi_power_resource_device *device_list;
|
||||||
|
|
||||||
result = acpi_power_get_context(handle, &resource);
|
result = acpi_power_get_context(handle, &resource);
|
||||||
if (result)
|
if (result)
|
||||||
|
@ -266,10 +262,25 @@ static int acpi_power_on(acpi_handle handle)
|
||||||
result = __acpi_power_on(resource);
|
result = __acpi_power_on(resource);
|
||||||
if (result)
|
if (result)
|
||||||
resource->ref_count--;
|
resource->ref_count--;
|
||||||
|
else
|
||||||
|
resume_device = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_unlock(&resource->resource_lock);
|
mutex_unlock(&resource->resource_lock);
|
||||||
|
|
||||||
|
if (!resume_device)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
mutex_lock(&resource->devices_lock);
|
||||||
|
|
||||||
|
device_list = resource->devices;
|
||||||
|
while (device_list) {
|
||||||
|
acpi_power_on_device(device_list->device);
|
||||||
|
device_list = device_list->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
mutex_unlock(&resource->devices_lock);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -355,7 +366,7 @@ static void __acpi_power_resource_unregister_device(struct device *dev,
|
||||||
if (acpi_power_get_context(res_handle, &resource))
|
if (acpi_power_get_context(res_handle, &resource))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
mutex_lock(&resource->resource_lock);
|
mutex_lock(&resource->devices_lock);
|
||||||
prev = NULL;
|
prev = NULL;
|
||||||
curr = resource->devices;
|
curr = resource->devices;
|
||||||
while (curr) {
|
while (curr) {
|
||||||
|
@ -372,7 +383,7 @@ static void __acpi_power_resource_unregister_device(struct device *dev,
|
||||||
prev = curr;
|
prev = curr;
|
||||||
curr = curr->next;
|
curr = curr->next;
|
||||||
}
|
}
|
||||||
mutex_unlock(&resource->resource_lock);
|
mutex_unlock(&resource->devices_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Unlink dev from all power resources in _PR0 */
|
/* Unlink dev from all power resources in _PR0 */
|
||||||
|
@ -414,10 +425,10 @@ static int __acpi_power_resource_register_device(
|
||||||
|
|
||||||
power_resource_device->device = powered_device;
|
power_resource_device->device = powered_device;
|
||||||
|
|
||||||
mutex_lock(&resource->resource_lock);
|
mutex_lock(&resource->devices_lock);
|
||||||
power_resource_device->next = resource->devices;
|
power_resource_device->next = resource->devices;
|
||||||
resource->devices = power_resource_device;
|
resource->devices = power_resource_device;
|
||||||
mutex_unlock(&resource->resource_lock);
|
mutex_unlock(&resource->devices_lock);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -462,7 +473,7 @@ int acpi_power_resource_register_device(struct device *dev, acpi_handle handle)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
no_power_resource:
|
no_power_resource:
|
||||||
printk(KERN_WARNING PREFIX "Invalid Power Resource to register!");
|
printk(KERN_DEBUG PREFIX "Invalid Power Resource to register!");
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(acpi_power_resource_register_device);
|
EXPORT_SYMBOL_GPL(acpi_power_resource_register_device);
|
||||||
|
@ -721,6 +732,7 @@ static int acpi_power_add(struct acpi_device *device)
|
||||||
|
|
||||||
resource->device = device;
|
resource->device = device;
|
||||||
mutex_init(&resource->resource_lock);
|
mutex_init(&resource->resource_lock);
|
||||||
|
mutex_init(&resource->devices_lock);
|
||||||
strcpy(resource->name, device->pnp.bus_id);
|
strcpy(resource->name, device->pnp.bus_id);
|
||||||
strcpy(acpi_device_name(device), ACPI_POWER_DEVICE_NAME);
|
strcpy(acpi_device_name(device), ACPI_POWER_DEVICE_NAME);
|
||||||
strcpy(acpi_device_class(device), ACPI_POWER_CLASS);
|
strcpy(acpi_device_class(device), ACPI_POWER_CLASS);
|
||||||
|
|
|
@ -268,6 +268,9 @@ static const struct pci_device_id ahci_pci_tbl[] = {
|
||||||
/* JMicron 360/1/3/5/6, match class to avoid IDE function */
|
/* JMicron 360/1/3/5/6, match class to avoid IDE function */
|
||||||
{ PCI_VENDOR_ID_JMICRON, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID,
|
{ PCI_VENDOR_ID_JMICRON, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID,
|
||||||
PCI_CLASS_STORAGE_SATA_AHCI, 0xffffff, board_ahci_ign_iferr },
|
PCI_CLASS_STORAGE_SATA_AHCI, 0xffffff, board_ahci_ign_iferr },
|
||||||
|
/* JMicron 362B and 362C have an AHCI function with IDE class code */
|
||||||
|
{ PCI_VDEVICE(JMICRON, 0x2362), board_ahci_ign_iferr },
|
||||||
|
{ PCI_VDEVICE(JMICRON, 0x236f), board_ahci_ign_iferr },
|
||||||
|
|
||||||
/* ATI */
|
/* ATI */
|
||||||
{ PCI_VDEVICE(ATI, 0x4380), board_ahci_sb600 }, /* ATI SB600 */
|
{ PCI_VDEVICE(ATI, 0x4380), board_ahci_sb600 }, /* ATI SB600 */
|
||||||
|
@ -393,6 +396,8 @@ static const struct pci_device_id ahci_pci_tbl[] = {
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9125 */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9125 */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x917a),
|
{ PCI_DEVICE(0x1b4b, 0x917a),
|
||||||
.driver_data = board_ahci_yes_fbs }, /* 88se9172 */
|
.driver_data = board_ahci_yes_fbs }, /* 88se9172 */
|
||||||
|
{ PCI_DEVICE(0x1b4b, 0x9192),
|
||||||
|
.driver_data = board_ahci_yes_fbs }, /* 88se9172 on some Gigabyte */
|
||||||
{ PCI_DEVICE(0x1b4b, 0x91a3),
|
{ PCI_DEVICE(0x1b4b, 0x91a3),
|
||||||
.driver_data = board_ahci_yes_fbs },
|
.driver_data = board_ahci_yes_fbs },
|
||||||
|
|
||||||
|
@ -400,7 +405,10 @@ static const struct pci_device_id ahci_pci_tbl[] = {
|
||||||
{ PCI_VDEVICE(PROMISE, 0x3f20), board_ahci }, /* PDC42819 */
|
{ PCI_VDEVICE(PROMISE, 0x3f20), board_ahci }, /* PDC42819 */
|
||||||
|
|
||||||
/* Asmedia */
|
/* Asmedia */
|
||||||
{ PCI_VDEVICE(ASMEDIA, 0x0612), board_ahci }, /* ASM1061 */
|
{ PCI_VDEVICE(ASMEDIA, 0x0601), board_ahci }, /* ASM1060 */
|
||||||
|
{ PCI_VDEVICE(ASMEDIA, 0x0602), board_ahci }, /* ASM1060 */
|
||||||
|
{ PCI_VDEVICE(ASMEDIA, 0x0611), board_ahci }, /* ASM1061 */
|
||||||
|
{ PCI_VDEVICE(ASMEDIA, 0x0612), board_ahci }, /* ASM1062 */
|
||||||
|
|
||||||
/* Generic, PCI class code for AHCI */
|
/* Generic, PCI class code for AHCI */
|
||||||
{ PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID,
|
{ PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID,
|
||||||
|
|
|
@ -250,7 +250,7 @@ int __init dma_declare_contiguous(struct device *dev, unsigned long size,
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
/* Sanitise input arguments */
|
/* Sanitise input arguments */
|
||||||
alignment = PAGE_SIZE << max(MAX_ORDER, pageblock_order);
|
alignment = PAGE_SIZE << max(MAX_ORDER - 1, pageblock_order);
|
||||||
base = ALIGN(base, alignment);
|
base = ALIGN(base, alignment);
|
||||||
size = ALIGN(size, alignment);
|
size = ALIGN(size, alignment);
|
||||||
limit &= ~(alignment - 1);
|
limit &= ~(alignment - 1);
|
||||||
|
|
|
@ -86,6 +86,7 @@ static struct usb_device_id ath3k_table[] = {
|
||||||
|
|
||||||
/* Atheros AR5BBU22 with sflash firmware */
|
/* Atheros AR5BBU22 with sflash firmware */
|
||||||
{ USB_DEVICE(0x0489, 0xE03C) },
|
{ USB_DEVICE(0x0489, 0xE03C) },
|
||||||
|
{ USB_DEVICE(0x0489, 0xE036) },
|
||||||
|
|
||||||
{ } /* Terminating entry */
|
{ } /* Terminating entry */
|
||||||
};
|
};
|
||||||
|
@ -109,6 +110,7 @@ static struct usb_device_id ath3k_blist_tbl[] = {
|
||||||
|
|
||||||
/* Atheros AR5BBU22 with sflash firmware */
|
/* Atheros AR5BBU22 with sflash firmware */
|
||||||
{ USB_DEVICE(0x0489, 0xE03C), .driver_info = BTUSB_ATH3012 },
|
{ USB_DEVICE(0x0489, 0xE03C), .driver_info = BTUSB_ATH3012 },
|
||||||
|
{ USB_DEVICE(0x0489, 0xE036), .driver_info = BTUSB_ATH3012 },
|
||||||
|
|
||||||
{ } /* Terminating entry */
|
{ } /* Terminating entry */
|
||||||
};
|
};
|
||||||
|
|
|
@ -52,6 +52,9 @@ static struct usb_device_id btusb_table[] = {
|
||||||
/* Generic Bluetooth USB device */
|
/* Generic Bluetooth USB device */
|
||||||
{ USB_DEVICE_INFO(0xe0, 0x01, 0x01) },
|
{ USB_DEVICE_INFO(0xe0, 0x01, 0x01) },
|
||||||
|
|
||||||
|
/* Apple-specific (Broadcom) devices */
|
||||||
|
{ USB_VENDOR_AND_INTERFACE_INFO(0x05ac, 0xff, 0x01, 0x01) },
|
||||||
|
|
||||||
/* Broadcom SoftSailing reporting vendor specific */
|
/* Broadcom SoftSailing reporting vendor specific */
|
||||||
{ USB_DEVICE(0x0a5c, 0x21e1) },
|
{ USB_DEVICE(0x0a5c, 0x21e1) },
|
||||||
|
|
||||||
|
@ -94,16 +97,14 @@ static struct usb_device_id btusb_table[] = {
|
||||||
|
|
||||||
/* Broadcom BCM20702A0 */
|
/* Broadcom BCM20702A0 */
|
||||||
{ USB_DEVICE(0x0489, 0xe042) },
|
{ USB_DEVICE(0x0489, 0xe042) },
|
||||||
{ USB_DEVICE(0x0a5c, 0x21e3) },
|
|
||||||
{ USB_DEVICE(0x0a5c, 0x21e6) },
|
|
||||||
{ USB_DEVICE(0x0a5c, 0x21e8) },
|
|
||||||
{ USB_DEVICE(0x0a5c, 0x21f3) },
|
|
||||||
{ USB_DEVICE(0x0a5c, 0x21f4) },
|
|
||||||
{ USB_DEVICE(0x413c, 0x8197) },
|
{ USB_DEVICE(0x413c, 0x8197) },
|
||||||
|
|
||||||
/* Foxconn - Hon Hai */
|
/* Foxconn - Hon Hai */
|
||||||
{ USB_DEVICE(0x0489, 0xe033) },
|
{ USB_DEVICE(0x0489, 0xe033) },
|
||||||
|
|
||||||
|
/*Broadcom devices with vendor specific id */
|
||||||
|
{ USB_VENDOR_AND_INTERFACE_INFO(0x0a5c, 0xff, 0x01, 0x01) },
|
||||||
|
|
||||||
{ } /* Terminating entry */
|
{ } /* Terminating entry */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -141,6 +142,7 @@ static struct usb_device_id blacklist_table[] = {
|
||||||
|
|
||||||
/* Atheros AR5BBU12 with sflash firmware */
|
/* Atheros AR5BBU12 with sflash firmware */
|
||||||
{ USB_DEVICE(0x0489, 0xe03c), .driver_info = BTUSB_ATH3012 },
|
{ USB_DEVICE(0x0489, 0xe03c), .driver_info = BTUSB_ATH3012 },
|
||||||
|
{ USB_DEVICE(0x0489, 0xe036), .driver_info = BTUSB_ATH3012 },
|
||||||
|
|
||||||
/* Broadcom BCM2035 */
|
/* Broadcom BCM2035 */
|
||||||
{ USB_DEVICE(0x0a5c, 0x2035), .driver_info = BTUSB_WRONG_SCO_MTU },
|
{ USB_DEVICE(0x0a5c, 0x2035), .driver_info = BTUSB_WRONG_SCO_MTU },
|
||||||
|
|
|
@ -120,3 +120,4 @@ u32 gen_split_key(struct device *jrdev, u8 *key_out, int split_key_len,
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
EXPORT_SYMBOL(gen_split_key);
|
||||||
|
|
|
@ -669,13 +669,18 @@ static int __devinit max77693_muic_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
info->dev = &pdev->dev;
|
info->dev = &pdev->dev;
|
||||||
info->max77693 = max77693;
|
info->max77693 = max77693;
|
||||||
info->max77693->regmap_muic = regmap_init_i2c(info->max77693->muic,
|
if (info->max77693->regmap_muic)
|
||||||
&max77693_muic_regmap_config);
|
dev_dbg(&pdev->dev, "allocate register map\n");
|
||||||
if (IS_ERR(info->max77693->regmap_muic)) {
|
else {
|
||||||
ret = PTR_ERR(info->max77693->regmap_muic);
|
info->max77693->regmap_muic = devm_regmap_init_i2c(
|
||||||
dev_err(max77693->dev,
|
info->max77693->muic,
|
||||||
"failed to allocate register map: %d\n", ret);
|
&max77693_muic_regmap_config);
|
||||||
goto err_regmap;
|
if (IS_ERR(info->max77693->regmap_muic)) {
|
||||||
|
ret = PTR_ERR(info->max77693->regmap_muic);
|
||||||
|
dev_err(max77693->dev,
|
||||||
|
"failed to allocate register map: %d\n", ret);
|
||||||
|
goto err_regmap;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
platform_set_drvdata(pdev, info);
|
platform_set_drvdata(pdev, info);
|
||||||
mutex_init(&info->mutex);
|
mutex_init(&info->mutex);
|
||||||
|
|
|
@ -294,7 +294,7 @@ config GPIO_MAX732X_IRQ
|
||||||
|
|
||||||
config GPIO_MC9S08DZ60
|
config GPIO_MC9S08DZ60
|
||||||
bool "MX35 3DS BOARD MC9S08DZ60 GPIO functions"
|
bool "MX35 3DS BOARD MC9S08DZ60 GPIO functions"
|
||||||
depends on I2C && MACH_MX35_3DS
|
depends on I2C=y && MACH_MX35_3DS
|
||||||
help
|
help
|
||||||
Select this to enable the MC9S08DZ60 GPIO driver
|
Select this to enable the MC9S08DZ60 GPIO driver
|
||||||
|
|
||||||
|
|
|
@ -247,9 +247,9 @@ static int __devinit em_gio_irq_domain_init(struct em_gio_priv *p)
|
||||||
|
|
||||||
p->irq_base = irq_alloc_descs(pdata->irq_base, 0,
|
p->irq_base = irq_alloc_descs(pdata->irq_base, 0,
|
||||||
pdata->number_of_pins, numa_node_id());
|
pdata->number_of_pins, numa_node_id());
|
||||||
if (IS_ERR_VALUE(p->irq_base)) {
|
if (p->irq_base < 0) {
|
||||||
dev_err(&pdev->dev, "cannot get irq_desc\n");
|
dev_err(&pdev->dev, "cannot get irq_desc\n");
|
||||||
return -ENXIO;
|
return p->irq_base;
|
||||||
}
|
}
|
||||||
pr_debug("gio: hw base = %d, nr = %d, sw base = %d\n",
|
pr_debug("gio: hw base = %d, nr = %d, sw base = %d\n",
|
||||||
pdata->gpio_base, pdata->number_of_pins, p->irq_base);
|
pdata->gpio_base, pdata->number_of_pins, p->irq_base);
|
||||||
|
|
|
@ -170,6 +170,7 @@ static int __devinit rdc321x_gpio_probe(struct platform_device *pdev)
|
||||||
rdc321x_gpio_dev->reg2_data_base = r->start + 0x4;
|
rdc321x_gpio_dev->reg2_data_base = r->start + 0x4;
|
||||||
|
|
||||||
rdc321x_gpio_dev->chip.label = "rdc321x-gpio";
|
rdc321x_gpio_dev->chip.label = "rdc321x-gpio";
|
||||||
|
rdc321x_gpio_dev->chip.owner = THIS_MODULE;
|
||||||
rdc321x_gpio_dev->chip.direction_input = rdc_gpio_direction_input;
|
rdc321x_gpio_dev->chip.direction_input = rdc_gpio_direction_input;
|
||||||
rdc321x_gpio_dev->chip.direction_output = rdc_gpio_config;
|
rdc321x_gpio_dev->chip.direction_output = rdc_gpio_config;
|
||||||
rdc321x_gpio_dev->chip.get = rdc_gpio_get_value;
|
rdc321x_gpio_dev->chip.get = rdc_gpio_get_value;
|
||||||
|
|
|
@ -82,7 +82,7 @@ int of_get_named_gpio_flags(struct device_node *np, const char *propname,
|
||||||
gpiochip_find(&gg_data, of_gpiochip_find_and_xlate);
|
gpiochip_find(&gg_data, of_gpiochip_find_and_xlate);
|
||||||
|
|
||||||
of_node_put(gg_data.gpiospec.np);
|
of_node_put(gg_data.gpiospec.np);
|
||||||
pr_debug("%s exited with status %d\n", __func__, ret);
|
pr_debug("%s exited with status %d\n", __func__, gg_data.out_gpio);
|
||||||
return gg_data.out_gpio;
|
return gg_data.out_gpio;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(of_get_named_gpio_flags);
|
EXPORT_SYMBOL(of_get_named_gpio_flags);
|
||||||
|
|
|
@ -193,6 +193,9 @@ static const struct file_operations ast_fops = {
|
||||||
.mmap = ast_mmap,
|
.mmap = ast_mmap,
|
||||||
.poll = drm_poll,
|
.poll = drm_poll,
|
||||||
.fasync = drm_fasync,
|
.fasync = drm_fasync,
|
||||||
|
#ifdef CONFIG_COMPAT
|
||||||
|
.compat_ioctl = drm_compat_ioctl,
|
||||||
|
#endif
|
||||||
.read = drm_read,
|
.read = drm_read,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -841,7 +841,7 @@ int ast_cursor_init(struct drm_device *dev)
|
||||||
|
|
||||||
ast->cursor_cache = obj;
|
ast->cursor_cache = obj;
|
||||||
ast->cursor_cache_gpu_addr = gpu_addr;
|
ast->cursor_cache_gpu_addr = gpu_addr;
|
||||||
DRM_ERROR("pinned cursor cache at %llx\n", ast->cursor_cache_gpu_addr);
|
DRM_DEBUG_KMS("pinned cursor cache at %llx\n", ast->cursor_cache_gpu_addr);
|
||||||
return 0;
|
return 0;
|
||||||
fail:
|
fail:
|
||||||
return ret;
|
return ret;
|
||||||
|
|
|
@ -74,6 +74,9 @@ static const struct file_operations cirrus_driver_fops = {
|
||||||
.unlocked_ioctl = drm_ioctl,
|
.unlocked_ioctl = drm_ioctl,
|
||||||
.mmap = cirrus_mmap,
|
.mmap = cirrus_mmap,
|
||||||
.poll = drm_poll,
|
.poll = drm_poll,
|
||||||
|
#ifdef CONFIG_COMPAT
|
||||||
|
.compat_ioctl = drm_compat_ioctl,
|
||||||
|
#endif
|
||||||
.fasync = drm_fasync,
|
.fasync = drm_fasync,
|
||||||
};
|
};
|
||||||
static struct drm_driver driver = {
|
static struct drm_driver driver = {
|
||||||
|
|
|
@ -36,6 +36,6 @@ config DRM_EXYNOS_VIDI
|
||||||
|
|
||||||
config DRM_EXYNOS_G2D
|
config DRM_EXYNOS_G2D
|
||||||
bool "Exynos DRM G2D"
|
bool "Exynos DRM G2D"
|
||||||
depends on DRM_EXYNOS
|
depends on DRM_EXYNOS && !VIDEO_SAMSUNG_S5P_G2D
|
||||||
help
|
help
|
||||||
Choose this option if you want to use Exynos G2D for DRM.
|
Choose this option if you want to use Exynos G2D for DRM.
|
||||||
|
|
|
@ -163,6 +163,12 @@ static void exynos_gem_dmabuf_kunmap(struct dma_buf *dma_buf,
|
||||||
/* TODO */
|
/* TODO */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int exynos_gem_dmabuf_mmap(struct dma_buf *dma_buf,
|
||||||
|
struct vm_area_struct *vma)
|
||||||
|
{
|
||||||
|
return -ENOTTY;
|
||||||
|
}
|
||||||
|
|
||||||
static struct dma_buf_ops exynos_dmabuf_ops = {
|
static struct dma_buf_ops exynos_dmabuf_ops = {
|
||||||
.map_dma_buf = exynos_gem_map_dma_buf,
|
.map_dma_buf = exynos_gem_map_dma_buf,
|
||||||
.unmap_dma_buf = exynos_gem_unmap_dma_buf,
|
.unmap_dma_buf = exynos_gem_unmap_dma_buf,
|
||||||
|
@ -170,6 +176,7 @@ static struct dma_buf_ops exynos_dmabuf_ops = {
|
||||||
.kmap_atomic = exynos_gem_dmabuf_kmap_atomic,
|
.kmap_atomic = exynos_gem_dmabuf_kmap_atomic,
|
||||||
.kunmap = exynos_gem_dmabuf_kunmap,
|
.kunmap = exynos_gem_dmabuf_kunmap,
|
||||||
.kunmap_atomic = exynos_gem_dmabuf_kunmap_atomic,
|
.kunmap_atomic = exynos_gem_dmabuf_kunmap_atomic,
|
||||||
|
.mmap = exynos_gem_dmabuf_mmap,
|
||||||
.release = exynos_dmabuf_release,
|
.release = exynos_dmabuf_release,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -160,7 +160,6 @@ static int exynos_drm_open(struct drm_device *dev, struct drm_file *file)
|
||||||
if (!file_priv)
|
if (!file_priv)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
drm_prime_init_file_private(&file->prime);
|
|
||||||
file->driver_priv = file_priv;
|
file->driver_priv = file_priv;
|
||||||
|
|
||||||
return exynos_drm_subdrv_open(dev, file);
|
return exynos_drm_subdrv_open(dev, file);
|
||||||
|
@ -184,7 +183,6 @@ static void exynos_drm_preclose(struct drm_device *dev,
|
||||||
e->base.destroy(&e->base);
|
e->base.destroy(&e->base);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
drm_prime_destroy_file_private(&file->prime);
|
|
||||||
spin_unlock_irqrestore(&dev->event_lock, flags);
|
spin_unlock_irqrestore(&dev->event_lock, flags);
|
||||||
|
|
||||||
exynos_drm_subdrv_close(dev, file);
|
exynos_drm_subdrv_close(dev, file);
|
||||||
|
@ -241,6 +239,9 @@ static const struct file_operations exynos_drm_driver_fops = {
|
||||||
.poll = drm_poll,
|
.poll = drm_poll,
|
||||||
.read = drm_read,
|
.read = drm_read,
|
||||||
.unlocked_ioctl = drm_ioctl,
|
.unlocked_ioctl = drm_ioctl,
|
||||||
|
#ifdef CONFIG_COMPAT
|
||||||
|
.compat_ioctl = drm_compat_ioctl,
|
||||||
|
#endif
|
||||||
.release = drm_release,
|
.release = drm_release,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue