mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-11 15:49:56 +00:00
drivers/fsi: Add error handling for slave
This change implements error handling in the FSI core, by cleaining up and retrying failed operations, using the SISC, TERM and BREAK facilities. Signed-off-by: Jeremy Kerr <jk@ozlabs.org> Signed-off-by: Christopher Bostic <cbostic@linux.vnet.ibm.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
66433b05a3
commit
1fa847d74a
@ -46,7 +46,9 @@ static const int engine_page_size = 0x400;
|
|||||||
/*
|
/*
|
||||||
* FSI slave engine control register offsets
|
* FSI slave engine control register offsets
|
||||||
*/
|
*/
|
||||||
#define FSI_SMODE 0x0 /* R/W: Mode register */
|
#define FSI_SMODE 0x0 /* R/W: Mode register */
|
||||||
|
#define FSI_SISC 0x8 /* R/W: Interrupt condition */
|
||||||
|
#define FSI_SSTAT 0x14 /* R : Slave status */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* SMODE fields
|
* SMODE fields
|
||||||
@ -77,10 +79,14 @@ struct fsi_slave {
|
|||||||
#define to_fsi_master(d) container_of(d, struct fsi_master, dev)
|
#define to_fsi_master(d) container_of(d, struct fsi_master, dev)
|
||||||
#define to_fsi_slave(d) container_of(d, struct fsi_slave, dev)
|
#define to_fsi_slave(d) container_of(d, struct fsi_slave, dev)
|
||||||
|
|
||||||
|
static const int slave_retries = 2;
|
||||||
|
static int discard_errors;
|
||||||
|
|
||||||
static int fsi_master_read(struct fsi_master *master, int link,
|
static int fsi_master_read(struct fsi_master *master, int link,
|
||||||
uint8_t slave_id, uint32_t addr, void *val, size_t size);
|
uint8_t slave_id, uint32_t addr, void *val, size_t size);
|
||||||
static int fsi_master_write(struct fsi_master *master, int link,
|
static int fsi_master_write(struct fsi_master *master, int link,
|
||||||
uint8_t slave_id, uint32_t addr, const void *val, size_t size);
|
uint8_t slave_id, uint32_t addr, const void *val, size_t size);
|
||||||
|
static int fsi_master_break(struct fsi_master *master, int link);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* fsi_device_read() / fsi_device_write() / fsi_device_peek()
|
* fsi_device_read() / fsi_device_write() / fsi_device_peek()
|
||||||
@ -173,18 +179,107 @@ static int fsi_slave_calc_addr(struct fsi_slave *slave, uint32_t *addrp,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int fsi_slave_report_and_clear_errors(struct fsi_slave *slave)
|
||||||
|
{
|
||||||
|
struct fsi_master *master = slave->master;
|
||||||
|
uint32_t irq, stat;
|
||||||
|
int rc, link;
|
||||||
|
uint8_t id;
|
||||||
|
|
||||||
|
link = slave->link;
|
||||||
|
id = slave->id;
|
||||||
|
|
||||||
|
rc = fsi_master_read(master, link, id, FSI_SLAVE_BASE + FSI_SISC,
|
||||||
|
&irq, sizeof(irq));
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
rc = fsi_master_read(master, link, id, FSI_SLAVE_BASE + FSI_SSTAT,
|
||||||
|
&stat, sizeof(stat));
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
dev_info(&slave->dev, "status: 0x%08x, sisc: 0x%08x\n",
|
||||||
|
be32_to_cpu(stat), be32_to_cpu(irq));
|
||||||
|
|
||||||
|
/* clear interrupts */
|
||||||
|
return fsi_master_write(master, link, id, FSI_SLAVE_BASE + FSI_SISC,
|
||||||
|
&irq, sizeof(irq));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int fsi_slave_set_smode(struct fsi_master *master, int link, int id);
|
||||||
|
|
||||||
|
int fsi_slave_handle_error(struct fsi_slave *slave, bool write, uint32_t addr,
|
||||||
|
size_t size)
|
||||||
|
{
|
||||||
|
struct fsi_master *master = slave->master;
|
||||||
|
int rc, link;
|
||||||
|
uint32_t reg;
|
||||||
|
uint8_t id;
|
||||||
|
|
||||||
|
if (discard_errors)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
link = slave->link;
|
||||||
|
id = slave->id;
|
||||||
|
|
||||||
|
dev_dbg(&slave->dev, "handling error on %s to 0x%08x[%zd]",
|
||||||
|
write ? "write" : "read", addr, size);
|
||||||
|
|
||||||
|
/* try a simple clear of error conditions, which may fail if we've lost
|
||||||
|
* communication with the slave
|
||||||
|
*/
|
||||||
|
rc = fsi_slave_report_and_clear_errors(slave);
|
||||||
|
if (!rc)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* send a TERM and retry */
|
||||||
|
if (master->term) {
|
||||||
|
rc = master->term(master, link, id);
|
||||||
|
if (!rc) {
|
||||||
|
rc = fsi_master_read(master, link, id, 0,
|
||||||
|
®, sizeof(reg));
|
||||||
|
if (!rc)
|
||||||
|
rc = fsi_slave_report_and_clear_errors(slave);
|
||||||
|
if (!rc)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* getting serious, reset the slave via BREAK */
|
||||||
|
rc = fsi_master_break(master, link);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
rc = fsi_slave_set_smode(master, link, id);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
return fsi_slave_report_and_clear_errors(slave);
|
||||||
|
}
|
||||||
|
|
||||||
int fsi_slave_read(struct fsi_slave *slave, uint32_t addr,
|
int fsi_slave_read(struct fsi_slave *slave, uint32_t addr,
|
||||||
void *val, size_t size)
|
void *val, size_t size)
|
||||||
{
|
{
|
||||||
uint8_t id = slave->id;
|
uint8_t id = slave->id;
|
||||||
int rc;
|
int rc, err_rc, i;
|
||||||
|
|
||||||
rc = fsi_slave_calc_addr(slave, &addr, &id);
|
rc = fsi_slave_calc_addr(slave, &addr, &id);
|
||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
return fsi_master_read(slave->master, slave->link, id,
|
for (i = 0; i < slave_retries; i++) {
|
||||||
addr, val, size);
|
rc = fsi_master_read(slave->master, slave->link,
|
||||||
|
id, addr, val, size);
|
||||||
|
if (!rc)
|
||||||
|
break;
|
||||||
|
|
||||||
|
err_rc = fsi_slave_handle_error(slave, false, addr, size);
|
||||||
|
if (err_rc)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(fsi_slave_read);
|
EXPORT_SYMBOL_GPL(fsi_slave_read);
|
||||||
|
|
||||||
@ -192,14 +287,24 @@ int fsi_slave_write(struct fsi_slave *slave, uint32_t addr,
|
|||||||
const void *val, size_t size)
|
const void *val, size_t size)
|
||||||
{
|
{
|
||||||
uint8_t id = slave->id;
|
uint8_t id = slave->id;
|
||||||
int rc;
|
int rc, err_rc, i;
|
||||||
|
|
||||||
rc = fsi_slave_calc_addr(slave, &addr, &id);
|
rc = fsi_slave_calc_addr(slave, &addr, &id);
|
||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
return fsi_master_write(slave->master, slave->link, id,
|
for (i = 0; i < slave_retries; i++) {
|
||||||
addr, val, size);
|
rc = fsi_master_write(slave->master, slave->link,
|
||||||
|
id, addr, val, size);
|
||||||
|
if (!rc)
|
||||||
|
break;
|
||||||
|
|
||||||
|
err_rc = fsi_slave_handle_error(slave, true, addr, size);
|
||||||
|
if (err_rc)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(fsi_slave_write);
|
EXPORT_SYMBOL_GPL(fsi_slave_write);
|
||||||
|
|
||||||
@ -770,3 +875,5 @@ static void fsi_exit(void)
|
|||||||
|
|
||||||
module_init(fsi_init);
|
module_init(fsi_init);
|
||||||
module_exit(fsi_exit);
|
module_exit(fsi_exit);
|
||||||
|
module_param(discard_errors, int, 0664);
|
||||||
|
MODULE_PARM_DESC(discard_errors, "Don't invoke error handling on bus accesses");
|
||||||
|
Loading…
x
Reference in New Issue
Block a user