mirror of https://github.com/xemu-project/xemu.git
hw/i2c: pmbus: fix error returns and guard against out of range accesses
Signed-off-by: Titus Rwantare <titusr@google.com> Reviewed-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Acked-by: Corey Minyard <cminyard@mvista.com> Message-Id: <20220307200605.4001451-3-titusr@google.com> Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
This commit is contained in:
parent
32480293db
commit
38870253f1
|
@ -149,7 +149,7 @@ static uint8_t pmbus_out_buf_pop(PMBusDevice *pmdev)
|
||||||
qemu_log_mask(LOG_GUEST_ERROR,
|
qemu_log_mask(LOG_GUEST_ERROR,
|
||||||
"%s: tried to read from empty buffer",
|
"%s: tried to read from empty buffer",
|
||||||
__func__);
|
__func__);
|
||||||
return 0xFF;
|
return PMBUS_ERR_BYTE;
|
||||||
}
|
}
|
||||||
uint8_t data = pmdev->out_buf[pmdev->out_buf_len - 1];
|
uint8_t data = pmdev->out_buf[pmdev->out_buf_len - 1];
|
||||||
pmdev->out_buf_len--;
|
pmdev->out_buf_len--;
|
||||||
|
@ -243,18 +243,47 @@ void pmbus_check_limits(PMBusDevice *pmdev)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* assert the status_cml error upon receipt of malformed command */
|
||||||
|
static void pmbus_cml_error(PMBusDevice *pmdev)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < pmdev->num_pages; i++) {
|
||||||
|
pmdev->pages[i].status_word |= PMBUS_STATUS_CML;
|
||||||
|
pmdev->pages[i].status_cml |= PB_CML_FAULT_INVALID_CMD;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static uint8_t pmbus_receive_byte(SMBusDevice *smd)
|
static uint8_t pmbus_receive_byte(SMBusDevice *smd)
|
||||||
{
|
{
|
||||||
PMBusDevice *pmdev = PMBUS_DEVICE(smd);
|
PMBusDevice *pmdev = PMBUS_DEVICE(smd);
|
||||||
PMBusDeviceClass *pmdc = PMBUS_DEVICE_GET_CLASS(pmdev);
|
PMBusDeviceClass *pmdc = PMBUS_DEVICE_GET_CLASS(pmdev);
|
||||||
uint8_t ret = 0xFF;
|
uint8_t ret = PMBUS_ERR_BYTE;
|
||||||
uint8_t index = pmdev->page;
|
uint8_t index;
|
||||||
|
|
||||||
if (pmdev->out_buf_len != 0) {
|
if (pmdev->out_buf_len != 0) {
|
||||||
ret = pmbus_out_buf_pop(pmdev);
|
ret = pmbus_out_buf_pop(pmdev);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Reading from all pages will return the value from page 0,
|
||||||
|
* this is unspecified behaviour in general.
|
||||||
|
*/
|
||||||
|
if (pmdev->page == PB_ALL_PAGES) {
|
||||||
|
index = 0;
|
||||||
|
qemu_log_mask(LOG_GUEST_ERROR,
|
||||||
|
"%s: tried to read from all pages\n",
|
||||||
|
__func__);
|
||||||
|
pmbus_cml_error(pmdev);
|
||||||
|
} else if (pmdev->page > pmdev->num_pages - 1) {
|
||||||
|
qemu_log_mask(LOG_GUEST_ERROR,
|
||||||
|
"%s: page %d is out of range\n",
|
||||||
|
__func__, pmdev->page);
|
||||||
|
pmbus_cml_error(pmdev);
|
||||||
|
return PMBUS_ERR_BYTE;
|
||||||
|
} else {
|
||||||
|
index = pmdev->page;
|
||||||
|
}
|
||||||
|
|
||||||
switch (pmdev->code) {
|
switch (pmdev->code) {
|
||||||
case PMBUS_PAGE:
|
case PMBUS_PAGE:
|
||||||
pmbus_send8(pmdev, pmdev->page);
|
pmbus_send8(pmdev, pmdev->page);
|
||||||
|
@ -1019,7 +1048,7 @@ static int pmbus_write_data(SMBusDevice *smd, uint8_t *buf, uint8_t len)
|
||||||
|
|
||||||
if (len == 0) {
|
if (len == 0) {
|
||||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: writing empty data\n", __func__);
|
qemu_log_mask(LOG_GUEST_ERROR, "%s: writing empty data\n", __func__);
|
||||||
return -1;
|
return PMBUS_ERR_BYTE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pmdev->pages) { /* allocate memory for pages on first use */
|
if (!pmdev->pages) { /* allocate memory for pages on first use */
|
||||||
|
@ -1038,6 +1067,7 @@ static int pmbus_write_data(SMBusDevice *smd, uint8_t *buf, uint8_t len)
|
||||||
pmdev->page = pmbus_receive8(pmdev);
|
pmdev->page = pmbus_receive8(pmdev);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* loop through all the pages when 0xFF is received */
|
/* loop through all the pages when 0xFF is received */
|
||||||
if (pmdev->page == PB_ALL_PAGES) {
|
if (pmdev->page == PB_ALL_PAGES) {
|
||||||
for (int i = 0; i < pmdev->num_pages; i++) {
|
for (int i = 0; i < pmdev->num_pages; i++) {
|
||||||
|
@ -1048,6 +1078,15 @@ static int pmbus_write_data(SMBusDevice *smd, uint8_t *buf, uint8_t len)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (pmdev->page > pmdev->num_pages - 1) {
|
||||||
|
qemu_log_mask(LOG_GUEST_ERROR,
|
||||||
|
"%s: page %u is out of range\n",
|
||||||
|
__func__, pmdev->page);
|
||||||
|
pmdev->page = 0; /* undefined behaviour - reset to page 0 */
|
||||||
|
pmbus_cml_error(pmdev);
|
||||||
|
return PMBUS_ERR_BYTE;
|
||||||
|
}
|
||||||
|
|
||||||
index = pmdev->page;
|
index = pmdev->page;
|
||||||
|
|
||||||
switch (pmdev->code) {
|
switch (pmdev->code) {
|
||||||
|
|
|
@ -228,6 +228,8 @@ enum pmbus_registers {
|
||||||
#define PB_MAX_PAGES 0x1F
|
#define PB_MAX_PAGES 0x1F
|
||||||
#define PB_ALL_PAGES 0xFF
|
#define PB_ALL_PAGES 0xFF
|
||||||
|
|
||||||
|
#define PMBUS_ERR_BYTE 0xFF
|
||||||
|
|
||||||
#define TYPE_PMBUS_DEVICE "pmbus-device"
|
#define TYPE_PMBUS_DEVICE "pmbus-device"
|
||||||
OBJECT_DECLARE_TYPE(PMBusDevice, PMBusDeviceClass,
|
OBJECT_DECLARE_TYPE(PMBusDevice, PMBusDeviceClass,
|
||||||
PMBUS_DEVICE)
|
PMBUS_DEVICE)
|
||||||
|
|
Loading…
Reference in New Issue