Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

ARM: 8040/1: pj4: properly detect existence of iWMMXt coprocessor

commit fdb487f5c961b94486a78fa61fa28b8eff1954ab
("ARM: 8015/1: Add cpu_is_pj4 to distinguish PJ4 because it
has some differences with V7")
introduced a fix for checking PJ4 cpuid to not use PJ4 specific
coprocessor access on non-PJ4 platforms.

Unfortunately, this in turn broke Marvell Armada 370/XP, both
comprising Marvell PJ4B CPUs without iWMMXt extension. Instead
of only checking for cpuid, which may not be sufficient to
determine iWMMXt support, the presence of iWMMXt coprocessors
can be checked by enabling and reading the Coprocessor ID
register (wCID, register 0 of CP1).

Therefore this adds an explicit check for the presence and correct
wCID value, before enabling iWMMXt capabilities. As a bonus, also
print the iWMMXt version of a detected coprocessor.

This has been tested to properly detect iWMMXt presence/absence on:
- PJ4, CPUID 0x560f5815, wCID 0x56052001: Marvell Dove, iWMMXt v2
- PJ4B, CPUID 0x561f5811: Marvell Armada 370, no iWMMXt
- PJ4B, CPUID 0x562f5841, wCID 0x56052001: Marvell Armada 1500, iWMMXt v2
- PJ4B, CPUID 0x562f5842: Marvell Armada XP, no iWMMXt

Signed-off-by: Sebastian Hesselbarth <sebastian.hesselbarth@gmail.com>
Tested-by: Thomas Petazzoni <thomas.petazzoni@free-electrons.com>
Tested-by: Kevin Hilman <khilman@linaro.org>
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>

authored by

Sebastian Hesselbarth and committed by
Russell King
e89f443b 7d065659

+33 -1
+33 -1
arch/arm/kernel/pj4-cp0.c
··· 72 72 : "=r" (temp) : "r" (value)); 73 73 } 74 74 75 + static int __init pj4_get_iwmmxt_version(void) 76 + { 77 + u32 cp_access, wcid; 78 + 79 + cp_access = pj4_cp_access_read(); 80 + pj4_cp_access_write(cp_access | 0xf); 81 + 82 + /* check if coprocessor 0 and 1 are available */ 83 + if ((pj4_cp_access_read() & 0xf) != 0xf) { 84 + pj4_cp_access_write(cp_access); 85 + return -ENODEV; 86 + } 87 + 88 + /* read iWMMXt coprocessor id register p1, c0 */ 89 + __asm__ __volatile__ ("mrc p1, 0, %0, c0, c0, 0\n" : "=r" (wcid)); 90 + 91 + pj4_cp_access_write(cp_access); 92 + 93 + /* iWMMXt v1 */ 94 + if ((wcid & 0xffffff00) == 0x56051000) 95 + return 1; 96 + /* iWMMXt v2 */ 97 + if ((wcid & 0xffffff00) == 0x56052000) 98 + return 2; 99 + 100 + return -EINVAL; 101 + } 75 102 76 103 /* 77 104 * Disable CP0/CP1 on boot, and let call_fpe() and the iWMMXt lazy ··· 107 80 static int __init pj4_cp0_init(void) 108 81 { 109 82 u32 __maybe_unused cp_access; 83 + int vers; 110 84 111 85 if (!cpu_is_pj4()) 86 + return 0; 87 + 88 + vers = pj4_get_iwmmxt_version(); 89 + if (vers < 0) 112 90 return 0; 113 91 114 92 #ifndef CONFIG_IWMMXT ··· 122 90 cp_access = pj4_cp_access_read() & ~0xf; 123 91 pj4_cp_access_write(cp_access); 124 92 125 - printk(KERN_INFO "PJ4 iWMMXt coprocessor enabled.\n"); 93 + pr_info("PJ4 iWMMXt v%d coprocessor enabled.\n", vers); 126 94 elf_hwcap |= HWCAP_IWMMXT; 127 95 thread_register_notifier(&iwmmxt_notifier_block); 128 96 #endif