x86/platform/UV: Remove Obsolete GRU MMR address translation
Use no-op messages in place of cross-partition interrupts when nacking a put message in the GRU. This allows us to remove MMR's as a destination from the GRU driver. Tested-by: John Estabrook <estabrook@sgi.com> Tested-by: Gary Kroening <gfk@sgi.com> Tested-by: Nathan Zimmer <nzimmer@sgi.com> Signed-off-by: Dimitri Sivanich <sivanich@sgi.com> Signed-off-by: Mike Travis <travis@sgi.com> Cc: Andrew Banman <abanman@sgi.com> Cc: Andrew Morton <akpm@linux-foundation.org> Cc: Andy Lutomirski <luto@amacapital.net> Cc: Borislav Petkov <bp@alien8.de> Cc: Brian Gerst <brgerst@gmail.com> Cc: Denys Vlasenko <dvlasenk@redhat.com> Cc: H. Peter Anvin <hpa@zytor.com> Cc: Len Brown <len.brown@intel.com> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Russ Anderson <rja@sgi.com> Cc: Thomas Gleixner <tglx@linutronix.de> Link: http://lkml.kernel.org/r/20160429215406.012228480@asylum.americas.sgi.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
This commit is contained in:
parent
c85375cd19
commit
40bfb8eedf
@ -647,16 +647,6 @@ static inline unsigned long uv_read_global_mmr64(int pnode, unsigned long offset
|
|||||||
return readq(uv_global_mmr64_address(pnode, offset));
|
return readq(uv_global_mmr64_address(pnode, offset));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Global MMR space addresses when referenced by the GRU. (GRU does
|
|
||||||
* NOT use socket addressing).
|
|
||||||
*/
|
|
||||||
static inline unsigned long uv_global_gru_mmr_address(int pnode, unsigned long offset)
|
|
||||||
{
|
|
||||||
return UV_GLOBAL_GRU_MMR_BASE | offset |
|
|
||||||
((unsigned long)pnode << uv_hub_info->m_val);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void uv_write_global_mmr8(int pnode, unsigned long offset, unsigned char val)
|
static inline void uv_write_global_mmr8(int pnode, unsigned long offset, unsigned char val)
|
||||||
{
|
{
|
||||||
writeb(val, uv_global_mmr64_address(pnode, offset));
|
writeb(val, uv_global_mmr64_address(pnode, offset));
|
||||||
|
@ -718,8 +718,8 @@ cberr:
|
|||||||
static int send_message_put_nacked(void *cb, struct gru_message_queue_desc *mqd,
|
static int send_message_put_nacked(void *cb, struct gru_message_queue_desc *mqd,
|
||||||
void *mesg, int lines)
|
void *mesg, int lines)
|
||||||
{
|
{
|
||||||
unsigned long m, *val = mesg, gpa, save;
|
unsigned long m;
|
||||||
int ret;
|
int ret, loops = 200; /* experimentally determined */
|
||||||
|
|
||||||
m = mqd->mq_gpa + (gru_get_amo_value_head(cb) << 6);
|
m = mqd->mq_gpa + (gru_get_amo_value_head(cb) << 6);
|
||||||
if (lines == 2) {
|
if (lines == 2) {
|
||||||
@ -735,22 +735,28 @@ static int send_message_put_nacked(void *cb, struct gru_message_queue_desc *mqd,
|
|||||||
return MQE_OK;
|
return MQE_OK;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Send a cross-partition interrupt to the SSI that contains the target
|
* Send a noop message in order to deliver a cross-partition interrupt
|
||||||
* message queue. Normally, the interrupt is automatically delivered by
|
* to the SSI that contains the target message queue. Normally, the
|
||||||
* hardware but some error conditions require explicit delivery.
|
* interrupt is automatically delivered by hardware following mesq
|
||||||
* Use the GRU to deliver the interrupt. Otherwise partition failures
|
* operations, but some error conditions require explicit delivery.
|
||||||
|
* The noop message will trigger delivery. Otherwise partition failures
|
||||||
* could cause unrecovered errors.
|
* could cause unrecovered errors.
|
||||||
*/
|
*/
|
||||||
gpa = uv_global_gru_mmr_address(mqd->interrupt_pnode, UVH_IPI_INT);
|
do {
|
||||||
save = *val;
|
ret = send_noop_message(cb, mqd, mesg);
|
||||||
*val = uv_hub_ipi_value(mqd->interrupt_apicid, mqd->interrupt_vector,
|
} while ((ret == MQIE_AGAIN || ret == MQE_CONGESTION) && (loops-- > 0));
|
||||||
dest_Fixed);
|
|
||||||
gru_vstore_phys(cb, gpa, gru_get_tri(mesg), IAA_REGISTER, IMA);
|
if (ret == MQIE_AGAIN || ret == MQE_CONGESTION) {
|
||||||
ret = gru_wait(cb);
|
/*
|
||||||
*val = save;
|
* Don't indicate to the app to resend the message, as it's
|
||||||
if (ret != CBS_IDLE)
|
* already been successfully sent. We simply send an OK
|
||||||
return MQE_UNEXPECTED_CB_ERR;
|
* (rather than fail the send with MQE_UNEXPECTED_CB_ERR),
|
||||||
return MQE_OK;
|
* assuming that the other side is receiving enough
|
||||||
|
* interrupts to get this message processed anyway.
|
||||||
|
*/
|
||||||
|
ret = MQE_OK;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Loading…
Reference in New Issue
Block a user