powerpc fixes for 4.15 #3
Two fixes for nasty kexec/kdump crashes in certain configurations. A couple of minor fixes for the new TIDR code. A fix for an oops in a CXL error handling path. Thanks to: Andrew Donnellan, Christophe Lombard, David Gibson, Mahesh Salgaonkar, Vaibhav Jain. -----BEGIN PGP SIGNATURE----- iQIwBAABCAAaBQJaITuLExxtcGVAZWxsZXJtYW4uaWQuYXUACgkQUevqPMjhpYBS 9A/+IYjSoDdTcJXqfR46xn2147RGBQIym2rqaBJd+WwZj8Br0Yap5hrPtr1zAilD 75aj0CRR0Y91nodnfishjujsJZckyQYOHi0/WQluLbpRlWEmeQt47gDjz70Wt1T8 BZUEVqPF2k6Mk5WJV6sSIHBtw2uKrl/lJZAUJbTobOWgsMdopO504MkFxvySWKMV AX7UEXrcxPLb/yVGk9Ih9iwXxm/ymvQrkljp4s3jWqkc7bWwN93CmimIQ+X6bop0 yqmAzCiUJsPsulmkBkmsY78llPg0roUrh98R4JIe0+cUiQROa5Kvt/u0zohN/rqS 6SkPT0ds2Fs1z5cHayyQWMN0j0A5sfwW2KRMLHCJjAwAxzoT2CdMZDv0+QLi0ETy RGtYvnew8eCqrfBpyBneEP1JySARJ85ML4rZvudewSHJoMzTkYDnSEKU8+wlqRIf KHdvHmErRMlF7OB6Om3Uxz6oIXan/Puj7HsdL8f7MazjFPqb/r+/AuTDzUov17Fs 7Y0qVawFyJyAJ8zkUAGB1kN2FN+eYnsFxUa7ubpeJY7VX+8pUOwT24rFc803eAu4 p/ad1CpBy+8xaq83WeaM6BpMqW80ao2BzzbQUhDcEQl4qovO/ZEZxQt0ySoQp+SY MqE8SnZMkL/30CasbKTAqmt+P44GCSYZVtOUwTmvLNMTjSg= =/LD/ -----END PGP SIGNATURE----- Merge tag 'powerpc-4.15-3' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux Pull powerpc fixes from Michael Ellerman: "Two fixes for nasty kexec/kdump crashes in certain configurations. A couple of minor fixes for the new TIDR code. A fix for an oops in a CXL error handling path. Thanks to: Andrew Donnellan, Christophe Lombard, David Gibson, Mahesh Salgaonkar, Vaibhav Jain" * tag 'powerpc-4.15-3' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux: powerpc: Do not assign thread.tidr if already assigned powerpc: Avoid signed to unsigned conversion in set_thread_tidr() powerpc/kexec: Fix kexec/kdump in P9 guest kernels powerpc/powernv: Fix kexec crashes caused by tlbie tracing cxl: Check if vphb exists before iterating over AFU devices
This commit is contained in:
commit
a0651c7fa2
@ -623,7 +623,9 @@ BEGIN_FTR_SECTION
|
|||||||
* NOTE, we rely on r0 being 0 from above.
|
* NOTE, we rely on r0 being 0 from above.
|
||||||
*/
|
*/
|
||||||
mtspr SPRN_IAMR,r0
|
mtspr SPRN_IAMR,r0
|
||||||
|
BEGIN_FTR_SECTION_NESTED(42)
|
||||||
mtspr SPRN_AMOR,r0
|
mtspr SPRN_AMOR,r0
|
||||||
|
END_FTR_SECTION_NESTED_IFSET(CPU_FTR_HVMODE, 42)
|
||||||
END_FTR_SECTION_IFSET(CPU_FTR_ARCH_300)
|
END_FTR_SECTION_IFSET(CPU_FTR_ARCH_300)
|
||||||
|
|
||||||
/* save regs for local vars on new stack.
|
/* save regs for local vars on new stack.
|
||||||
|
@ -1569,16 +1569,22 @@ void arch_release_task_struct(struct task_struct *t)
|
|||||||
*/
|
*/
|
||||||
int set_thread_tidr(struct task_struct *t)
|
int set_thread_tidr(struct task_struct *t)
|
||||||
{
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
if (!cpu_has_feature(CPU_FTR_ARCH_300))
|
if (!cpu_has_feature(CPU_FTR_ARCH_300))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (t != current)
|
if (t != current)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
t->thread.tidr = assign_thread_tidr();
|
if (t->thread.tidr)
|
||||||
if (t->thread.tidr < 0)
|
return 0;
|
||||||
return t->thread.tidr;
|
|
||||||
|
|
||||||
|
rc = assign_thread_tidr();
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
t->thread.tidr = rc;
|
||||||
mtspr(SPRN_TIDR, t->thread.tidr);
|
mtspr(SPRN_TIDR, t->thread.tidr);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -47,7 +47,8 @@
|
|||||||
|
|
||||||
DEFINE_RAW_SPINLOCK(native_tlbie_lock);
|
DEFINE_RAW_SPINLOCK(native_tlbie_lock);
|
||||||
|
|
||||||
static inline void __tlbie(unsigned long vpn, int psize, int apsize, int ssize)
|
static inline unsigned long ___tlbie(unsigned long vpn, int psize,
|
||||||
|
int apsize, int ssize)
|
||||||
{
|
{
|
||||||
unsigned long va;
|
unsigned long va;
|
||||||
unsigned int penc;
|
unsigned int penc;
|
||||||
@ -100,7 +101,15 @@ static inline void __tlbie(unsigned long vpn, int psize, int apsize, int ssize)
|
|||||||
: "memory");
|
: "memory");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
trace_tlbie(0, 0, va, 0, 0, 0, 0);
|
return va;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void __tlbie(unsigned long vpn, int psize, int apsize, int ssize)
|
||||||
|
{
|
||||||
|
unsigned long rb;
|
||||||
|
|
||||||
|
rb = ___tlbie(vpn, psize, apsize, ssize);
|
||||||
|
trace_tlbie(0, 0, rb, 0, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void __tlbiel(unsigned long vpn, int psize, int apsize, int ssize)
|
static inline void __tlbiel(unsigned long vpn, int psize, int apsize, int ssize)
|
||||||
@ -652,7 +661,7 @@ static void native_hpte_clear(void)
|
|||||||
if (hpte_v & HPTE_V_VALID) {
|
if (hpte_v & HPTE_V_VALID) {
|
||||||
hpte_decode(hptep, slot, &psize, &apsize, &ssize, &vpn);
|
hpte_decode(hptep, slot, &psize, &apsize, &ssize, &vpn);
|
||||||
hptep->v = 0;
|
hptep->v = 0;
|
||||||
__tlbie(vpn, psize, apsize, ssize);
|
___tlbie(vpn, psize, apsize, ssize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2083,6 +2083,9 @@ static pci_ers_result_t cxl_vphb_error_detected(struct cxl_afu *afu,
|
|||||||
/* There should only be one entry, but go through the list
|
/* There should only be one entry, but go through the list
|
||||||
* anyway
|
* anyway
|
||||||
*/
|
*/
|
||||||
|
if (afu->phb == NULL)
|
||||||
|
return result;
|
||||||
|
|
||||||
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
||||||
if (!afu_dev->driver)
|
if (!afu_dev->driver)
|
||||||
continue;
|
continue;
|
||||||
@ -2124,8 +2127,7 @@ static pci_ers_result_t cxl_pci_error_detected(struct pci_dev *pdev,
|
|||||||
* Tell the AFU drivers; but we don't care what they
|
* Tell the AFU drivers; but we don't care what they
|
||||||
* say, we're going away.
|
* say, we're going away.
|
||||||
*/
|
*/
|
||||||
if (afu->phb != NULL)
|
cxl_vphb_error_detected(afu, state);
|
||||||
cxl_vphb_error_detected(afu, state);
|
|
||||||
}
|
}
|
||||||
return PCI_ERS_RESULT_DISCONNECT;
|
return PCI_ERS_RESULT_DISCONNECT;
|
||||||
}
|
}
|
||||||
@ -2265,6 +2267,9 @@ static pci_ers_result_t cxl_pci_slot_reset(struct pci_dev *pdev)
|
|||||||
if (cxl_afu_select_best_mode(afu))
|
if (cxl_afu_select_best_mode(afu))
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
if (afu->phb == NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
||||||
/* Reset the device context.
|
/* Reset the device context.
|
||||||
* TODO: make this less disruptive
|
* TODO: make this less disruptive
|
||||||
@ -2327,6 +2332,9 @@ static void cxl_pci_resume(struct pci_dev *pdev)
|
|||||||
for (i = 0; i < adapter->slices; i++) {
|
for (i = 0; i < adapter->slices; i++) {
|
||||||
afu = adapter->afu[i];
|
afu = adapter->afu[i];
|
||||||
|
|
||||||
|
if (afu->phb == NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
list_for_each_entry(afu_dev, &afu->phb->bus->devices, bus_list) {
|
||||||
if (afu_dev->driver && afu_dev->driver->err_handler &&
|
if (afu_dev->driver && afu_dev->driver->err_handler &&
|
||||||
afu_dev->driver->err_handler->resume)
|
afu_dev->driver->err_handler->resume)
|
||||||
|
Loading…
Reference in New Issue
Block a user