hw/misc/sifive_u_otp: Add backend drive support
Add '-drive' support to OTP device. Allow users to assign a raw file as OTP image. test commands for 16k otp.img filled with zero: $ dd if=/dev/zero of=./otp.img bs=1k count=16 $ ./qemu-system-riscv64 -M sifive_u -m 256M -nographic -bios none \ -kernel ../opensbi/build/platform/sifive/fu540/firmware/fw_payload.elf \ -d guest_errors -drive if=none,format=raw,file=otp.img Signed-off-by: Green Wan <green.wan@sifive.com> Reviewed-by: Bin Meng <bin.meng@windriver.com> Tested-by: Bin Meng <bin.meng@windriver.com> Acked-by: Alistair Francis <alistair.francis@wdc.com> Message-id: 20201020033732.12921-3-green.wan@sifive.com Signed-off-by: Alistair Francis <alistair.francis@wdc.com>
This commit is contained in:
parent
a54d259157
commit
51b6c1bbc3
@ -19,11 +19,14 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "qemu/osdep.h"
|
#include "qemu/osdep.h"
|
||||||
|
#include "qapi/error.h"
|
||||||
#include "hw/qdev-properties.h"
|
#include "hw/qdev-properties.h"
|
||||||
#include "hw/sysbus.h"
|
#include "hw/sysbus.h"
|
||||||
#include "qemu/log.h"
|
#include "qemu/log.h"
|
||||||
#include "qemu/module.h"
|
#include "qemu/module.h"
|
||||||
#include "hw/misc/sifive_u_otp.h"
|
#include "hw/misc/sifive_u_otp.h"
|
||||||
|
#include "sysemu/blockdev.h"
|
||||||
|
#include "sysemu/block-backend.h"
|
||||||
|
|
||||||
#define WRITTEN_BIT_ON 0x1
|
#define WRITTEN_BIT_ON 0x1
|
||||||
|
|
||||||
@ -54,6 +57,16 @@ static uint64_t sifive_u_otp_read(void *opaque, hwaddr addr, unsigned int size)
|
|||||||
if ((s->pce & SIFIVE_U_OTP_PCE_EN) &&
|
if ((s->pce & SIFIVE_U_OTP_PCE_EN) &&
|
||||||
(s->pdstb & SIFIVE_U_OTP_PDSTB_EN) &&
|
(s->pdstb & SIFIVE_U_OTP_PDSTB_EN) &&
|
||||||
(s->ptrim & SIFIVE_U_OTP_PTRIM_EN)) {
|
(s->ptrim & SIFIVE_U_OTP_PTRIM_EN)) {
|
||||||
|
|
||||||
|
/* read from backend */
|
||||||
|
if (s->blk) {
|
||||||
|
int32_t buf;
|
||||||
|
|
||||||
|
blk_pread(s->blk, s->pa * SIFIVE_U_OTP_FUSE_WORD, &buf,
|
||||||
|
SIFIVE_U_OTP_FUSE_WORD);
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
return s->fuse[s->pa & SIFIVE_U_OTP_PA_MASK];
|
return s->fuse[s->pa & SIFIVE_U_OTP_PA_MASK];
|
||||||
} else {
|
} else {
|
||||||
return 0xff;
|
return 0xff;
|
||||||
@ -145,6 +158,12 @@ static void sifive_u_otp_write(void *opaque, hwaddr addr,
|
|||||||
/* write bit data */
|
/* write bit data */
|
||||||
SET_FUSEARRAY_BIT(s->fuse, s->pa, s->paio, s->pdin);
|
SET_FUSEARRAY_BIT(s->fuse, s->pa, s->paio, s->pdin);
|
||||||
|
|
||||||
|
/* write to backend */
|
||||||
|
if (s->blk) {
|
||||||
|
blk_pwrite(s->blk, s->pa * SIFIVE_U_OTP_FUSE_WORD,
|
||||||
|
&s->fuse[s->pa], SIFIVE_U_OTP_FUSE_WORD, 0);
|
||||||
|
}
|
||||||
|
|
||||||
/* update written bit */
|
/* update written bit */
|
||||||
SET_FUSEARRAY_BIT(s->fuse_wo, s->pa, s->paio, WRITTEN_BIT_ON);
|
SET_FUSEARRAY_BIT(s->fuse_wo, s->pa, s->paio, WRITTEN_BIT_ON);
|
||||||
}
|
}
|
||||||
@ -168,16 +187,48 @@ static const MemoryRegionOps sifive_u_otp_ops = {
|
|||||||
|
|
||||||
static Property sifive_u_otp_properties[] = {
|
static Property sifive_u_otp_properties[] = {
|
||||||
DEFINE_PROP_UINT32("serial", SiFiveUOTPState, serial, 0),
|
DEFINE_PROP_UINT32("serial", SiFiveUOTPState, serial, 0),
|
||||||
|
DEFINE_PROP_DRIVE("drive", SiFiveUOTPState, blk),
|
||||||
DEFINE_PROP_END_OF_LIST(),
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
};
|
};
|
||||||
|
|
||||||
static void sifive_u_otp_realize(DeviceState *dev, Error **errp)
|
static void sifive_u_otp_realize(DeviceState *dev, Error **errp)
|
||||||
{
|
{
|
||||||
SiFiveUOTPState *s = SIFIVE_U_OTP(dev);
|
SiFiveUOTPState *s = SIFIVE_U_OTP(dev);
|
||||||
|
DriveInfo *dinfo;
|
||||||
|
|
||||||
memory_region_init_io(&s->mmio, OBJECT(dev), &sifive_u_otp_ops, s,
|
memory_region_init_io(&s->mmio, OBJECT(dev), &sifive_u_otp_ops, s,
|
||||||
TYPE_SIFIVE_U_OTP, SIFIVE_U_OTP_REG_SIZE);
|
TYPE_SIFIVE_U_OTP, SIFIVE_U_OTP_REG_SIZE);
|
||||||
sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->mmio);
|
sysbus_init_mmio(SYS_BUS_DEVICE(dev), &s->mmio);
|
||||||
|
|
||||||
|
dinfo = drive_get_next(IF_NONE);
|
||||||
|
if (dinfo) {
|
||||||
|
int ret;
|
||||||
|
uint64_t perm;
|
||||||
|
int filesize;
|
||||||
|
BlockBackend *blk;
|
||||||
|
|
||||||
|
blk = blk_by_legacy_dinfo(dinfo);
|
||||||
|
filesize = SIFIVE_U_OTP_NUM_FUSES * SIFIVE_U_OTP_FUSE_WORD;
|
||||||
|
if (blk_getlength(blk) < filesize) {
|
||||||
|
error_setg(errp, "OTP drive size < 16K");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
qdev_prop_set_drive_err(dev, "drive", blk, errp);
|
||||||
|
|
||||||
|
if (s->blk) {
|
||||||
|
perm = BLK_PERM_CONSISTENT_READ |
|
||||||
|
(blk_is_read_only(s->blk) ? 0 : BLK_PERM_WRITE);
|
||||||
|
ret = blk_set_perm(s->blk, perm, BLK_PERM_ALL, errp);
|
||||||
|
if (ret < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (blk_pread(s->blk, 0, s->fuse, filesize) != filesize) {
|
||||||
|
error_setg(errp, "failed to read the initial flash content");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sifive_u_otp_reset(DeviceState *dev)
|
static void sifive_u_otp_reset(DeviceState *dev)
|
||||||
@ -191,6 +242,20 @@ static void sifive_u_otp_reset(DeviceState *dev)
|
|||||||
s->fuse[SIFIVE_U_OTP_SERIAL_ADDR] = s->serial;
|
s->fuse[SIFIVE_U_OTP_SERIAL_ADDR] = s->serial;
|
||||||
s->fuse[SIFIVE_U_OTP_SERIAL_ADDR + 1] = ~(s->serial);
|
s->fuse[SIFIVE_U_OTP_SERIAL_ADDR + 1] = ~(s->serial);
|
||||||
|
|
||||||
|
if (s->blk) {
|
||||||
|
/* Put serial number to backend as well*/
|
||||||
|
uint32_t serial_data;
|
||||||
|
int index = SIFIVE_U_OTP_SERIAL_ADDR;
|
||||||
|
|
||||||
|
serial_data = s->serial;
|
||||||
|
blk_pwrite(s->blk, index * SIFIVE_U_OTP_FUSE_WORD,
|
||||||
|
&serial_data, SIFIVE_U_OTP_FUSE_WORD, 0);
|
||||||
|
|
||||||
|
serial_data = ~(s->serial);
|
||||||
|
blk_pwrite(s->blk, (index + 1) * SIFIVE_U_OTP_FUSE_WORD,
|
||||||
|
&serial_data, SIFIVE_U_OTP_FUSE_WORD, 0);
|
||||||
|
}
|
||||||
|
|
||||||
/* Initialize write-once map */
|
/* Initialize write-once map */
|
||||||
memset(s->fuse_wo, 0x00, sizeof(s->fuse_wo));
|
memset(s->fuse_wo, 0x00, sizeof(s->fuse_wo));
|
||||||
}
|
}
|
||||||
|
@ -46,6 +46,7 @@
|
|||||||
|
|
||||||
#define SIFIVE_U_OTP_PA_MASK 0xfff
|
#define SIFIVE_U_OTP_PA_MASK 0xfff
|
||||||
#define SIFIVE_U_OTP_NUM_FUSES 0x1000
|
#define SIFIVE_U_OTP_NUM_FUSES 0x1000
|
||||||
|
#define SIFIVE_U_OTP_FUSE_WORD 4
|
||||||
#define SIFIVE_U_OTP_SERIAL_ADDR 0xfc
|
#define SIFIVE_U_OTP_SERIAL_ADDR 0xfc
|
||||||
|
|
||||||
#define SIFIVE_U_OTP_REG_SIZE 0x1000
|
#define SIFIVE_U_OTP_REG_SIZE 0x1000
|
||||||
@ -80,6 +81,7 @@ struct SiFiveUOTPState {
|
|||||||
uint32_t fuse_wo[SIFIVE_U_OTP_NUM_FUSES];
|
uint32_t fuse_wo[SIFIVE_U_OTP_NUM_FUSES];
|
||||||
/* config */
|
/* config */
|
||||||
uint32_t serial;
|
uint32_t serial;
|
||||||
|
BlockBackend *blk;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* HW_SIFIVE_U_OTP_H */
|
#endif /* HW_SIFIVE_U_OTP_H */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user