ssi: Support for multiple attached devices
Removed assertion that only one device is attached to the SSI bus. When multiple devices are attached, all slaves have their transfer function called for transfers. Each device is responsible for knowing whether or not its CS is active, and if not returning 0. The returned data is the logical or of all responses from the (mulitple) devices. Signed-off-by: Peter A. G. Crosthwaite <peter.crosthwaite@petalogix.com> Acked-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
parent
4bb26682f7
commit
b4a76e84f4
24
hw/ssi.c
24
hw/ssi.c
@ -2,6 +2,8 @@
|
|||||||
* QEMU Synchronous Serial Interface support
|
* QEMU Synchronous Serial Interface support
|
||||||
*
|
*
|
||||||
* Copyright (c) 2009 CodeSourcery.
|
* Copyright (c) 2009 CodeSourcery.
|
||||||
|
* Copyright (c) 2012 Peter A.G. Crosthwaite (peter.crosthwaite@petalogix.com)
|
||||||
|
* Copyright (c) 2012 PetaLogix Pty Ltd.
|
||||||
* Written by Paul Brook
|
* Written by Paul Brook
|
||||||
*
|
*
|
||||||
* This code is licensed under the GNU GPL v2.
|
* This code is licensed under the GNU GPL v2.
|
||||||
@ -29,14 +31,6 @@ static int ssi_slave_init(DeviceState *dev)
|
|||||||
{
|
{
|
||||||
SSISlave *s = SSI_SLAVE(dev);
|
SSISlave *s = SSI_SLAVE(dev);
|
||||||
SSISlaveClass *ssc = SSI_SLAVE_GET_CLASS(s);
|
SSISlaveClass *ssc = SSI_SLAVE_GET_CLASS(s);
|
||||||
SSIBus *bus;
|
|
||||||
BusChild *kid;
|
|
||||||
|
|
||||||
bus = FROM_QBUS(SSIBus, qdev_get_parent_bus(dev));
|
|
||||||
kid = QTAILQ_FIRST(&bus->qbus.children);
|
|
||||||
if (kid->child != dev || QTAILQ_NEXT(kid, sibling) != NULL) {
|
|
||||||
hw_error("Too many devices on SSI bus");
|
|
||||||
}
|
|
||||||
|
|
||||||
return ssc->init(s);
|
return ssc->init(s);
|
||||||
}
|
}
|
||||||
@ -74,16 +68,16 @@ SSIBus *ssi_create_bus(DeviceState *parent, const char *name)
|
|||||||
uint32_t ssi_transfer(SSIBus *bus, uint32_t val)
|
uint32_t ssi_transfer(SSIBus *bus, uint32_t val)
|
||||||
{
|
{
|
||||||
BusChild *kid;
|
BusChild *kid;
|
||||||
SSISlave *slave;
|
|
||||||
SSISlaveClass *ssc;
|
SSISlaveClass *ssc;
|
||||||
|
uint32_t r = 0;
|
||||||
|
|
||||||
kid = QTAILQ_FIRST(&bus->qbus.children);
|
QTAILQ_FOREACH(kid, &bus->qbus.children, sibling) {
|
||||||
if (!kid) {
|
SSISlave *slave = SSI_SLAVE(kid->child);
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
slave = SSI_SLAVE(kid->child);
|
|
||||||
ssc = SSI_SLAVE_GET_CLASS(slave);
|
ssc = SSI_SLAVE_GET_CLASS(slave);
|
||||||
return ssc->transfer(slave, val);
|
r |= ssc->transfer(slave, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ssi_slave_register_types(void)
|
static void ssi_slave_register_types(void)
|
||||||
|
Loading…
Reference in New Issue
Block a user