Compare commits
156 Commits
for-upstre
...
pull-input
Author | SHA1 | Date | |
---|---|---|---|
|
50c6617fcb | ||
|
e37f202450 | ||
|
29429c7244 | ||
|
70d3a7a7b8 | ||
|
136e67e9b5 | ||
|
041c96666d | ||
|
e0d6e6a5e7 | ||
|
607d98b81e | ||
|
35979d71c4 | ||
|
2dd081ae76 | ||
|
dfafd09088 | ||
|
043b7f8d12 | ||
|
9e729b57ac | ||
|
64e0e2de0c | ||
|
f149e3e847 | ||
|
c0f4af1719 | ||
|
f59492b984 | ||
|
6ac285b824 | ||
|
93c9aea9b4 | ||
|
65731d1c3e | ||
|
8f95ce2e4c | ||
|
0eacea7060 | ||
|
46747d1508 | ||
|
b60a7726cc | ||
|
70556264a8 | ||
|
e8c81b4d8a | ||
|
ed9114356b | ||
|
0ebcc56453 | ||
|
151623353f | ||
|
1c1f949844 | ||
|
e80084d352 | ||
|
a631892f9d | ||
|
235256a2bd | ||
|
e5048d15ce | ||
|
97699eff3a | ||
|
63ce8e150c | ||
|
dbe2a7a62a | ||
|
59a2c4ce2b | ||
|
597db727cc | ||
|
04404c21cc | ||
|
fe509ee237 | ||
|
cb55111b4e | ||
|
146db9f919 | ||
|
cee2dedb85 | ||
|
1f9296b51a | ||
|
e5554e2015 | ||
|
c5d1e2cce3 | ||
|
ed173cb704 | ||
|
cb021cfee7 | ||
|
eebf29401a | ||
|
10905cb26a | ||
|
2b8419cb49 | ||
|
81ab11a7a5 | ||
|
541be9274e | ||
|
a697d240ff | ||
|
5a6e8ba64f | ||
|
da1c4ec88a | ||
|
15124e1420 | ||
|
bc0d104c6a | ||
|
3af8f177fa | ||
|
931f53e184 | ||
|
89ae5831a5 | ||
|
40893768df | ||
|
4d249a2074 | ||
|
8a55acb1e0 | ||
|
88ed34ff5e | ||
|
f9bbba9569 | ||
|
44e7ebb8bb | ||
|
f4026f269a | ||
|
0bc375eb51 | ||
|
14101d028d | ||
|
1dde0f48d5 | ||
|
60e17d2822 | ||
|
2321442920 | ||
|
6a0fcbdf2d | ||
|
42f53fea9f | ||
|
458dd76656 | ||
|
e9854c3945 | ||
|
29cd33d3c7 | ||
|
fa4faba448 | ||
|
dfdb483454 | ||
|
fbb96c4b7f | ||
|
87afe467e2 | ||
|
e8925712e6 | ||
|
d8bb915972 | ||
|
f47ede195b | ||
|
dde7c241e3 | ||
|
5a1f7f44cf | ||
|
ab409bb3fe | ||
|
02bb9bbf1d | ||
|
37f3616aa8 | ||
|
9585db68c7 | ||
|
774f0abeae | ||
|
00f3fd63e1 | ||
|
374e0cd4db | ||
|
1039e1290f | ||
|
cffe7b3249 | ||
|
1ba50f4ea0 | ||
|
c9d17ad0dd | ||
|
9aebf3b892 | ||
|
a4127c428e | ||
|
f9d7b4b3b4 | ||
|
5abbf0ee4d | ||
|
247147fbc1 | ||
|
d224469d87 | ||
|
e5d7bbeb10 | ||
|
3ae59580a0 | ||
|
a0f1eab157 | ||
|
d4362d642e | ||
|
1862ed02df | ||
|
4f2280b219 | ||
|
c0f896810b | ||
|
ca976d1803 | ||
|
55d7bfe229 | ||
|
01f7cecf00 | ||
|
769188d3bb | ||
|
cad684c538 | ||
|
e8601dd5d0 | ||
|
a327c9215d | ||
|
450749141e | ||
|
ec56214f6f | ||
|
638ca939d8 | ||
|
3533c3d2bf | ||
|
7334d6507a | ||
|
594a53607e | ||
|
bd2ba2752d | ||
|
df9bb6660d | ||
|
77e35b4bbe | ||
|
5450eeaaad | ||
|
27107d416e | ||
|
276b7ac8e9 | ||
|
0b8b863fb2 | ||
|
38fff2c9f4 | ||
|
63cdca364c | ||
|
6b7afb7f0b | ||
|
b89dc7e33f | ||
|
f5dc597878 | ||
|
5a882e40d5 | ||
|
f3f8c45972 | ||
|
2e6a0dd1ac | ||
|
2aa76dc18c | ||
|
d73ad35990 | ||
|
7d553f27fc | ||
|
dc1f598845 | ||
|
f0bc7fe3b7 | ||
|
75bd0c7253 | ||
|
6758008e2c | ||
|
31376776d0 | ||
|
a30cf8760f | ||
|
4df7961faa | ||
|
5bde14078d | ||
|
7dbb4c49bf | ||
|
9a48bcd1b8 | ||
|
317b0a6d8b | ||
|
de9d61e83d | ||
|
be894f51b6 |
55
.travis.yml
55
.travis.yml
@@ -12,7 +12,7 @@ notifications:
|
||||
on_failure: always
|
||||
env:
|
||||
global:
|
||||
- TEST_CMD="make check"
|
||||
- TEST_CMD=""
|
||||
- EXTRA_CONFIG=""
|
||||
# Development packages, EXTRA_PKGS saved for additional builds
|
||||
- CORE_PKGS="libusb-1.0-0-dev libiscsi-dev librados-dev libncurses5-dev"
|
||||
@@ -20,31 +20,51 @@ env:
|
||||
- GUI_PKGS="libgtk-3-dev libvte-2.90-dev libsdl1.2-dev libpng12-dev libpixman-1-dev"
|
||||
- EXTRA_PKGS=""
|
||||
matrix:
|
||||
# Group major targets together with their linux-user counterparts
|
||||
- TARGETS=alpha-softmmu,alpha-linux-user
|
||||
- TARGETS=arm-softmmu,arm-linux-user
|
||||
- TARGETS=aarch64-softmmu,aarch64-linux-user
|
||||
- TARGETS=cris-softmmu
|
||||
- TARGETS=i386-softmmu,x86_64-softmmu
|
||||
- TARGETS=lm32-softmmu
|
||||
- TARGETS=m68k-softmmu
|
||||
- TARGETS=microblaze-softmmu,microblazeel-softmmu
|
||||
- TARGETS=arm-softmmu,arm-linux-user,armeb-linux-user,aarch64-softmmu,aarch64-linux-user
|
||||
- TARGETS=cris-softmmu,cris-linux-user
|
||||
- TARGETS=i386-softmmu,i386-linux-user,x86_64-softmmu,x86_64-linux-user
|
||||
- TARGETS=m68k-softmmu,m68k-linux-user
|
||||
- TARGETS=microblaze-softmmu,microblazeel-softmmu,microblaze-linux-user,microblazeel-linux-user
|
||||
- TARGETS=mips-softmmu,mips64-softmmu,mips64el-softmmu,mipsel-softmmu
|
||||
- TARGETS=moxie-softmmu
|
||||
- TARGETS=or32-softmmu,
|
||||
- TARGETS=ppc-softmmu,ppc64-softmmu,ppcemb-softmmu
|
||||
- TARGETS=s390x-softmmu
|
||||
- TARGETS=sh4-softmmu,sh4eb-softmmu
|
||||
- TARGETS=sparc-softmmu,sparc64-softmmu
|
||||
- TARGETS=unicore32-softmmu
|
||||
- TARGETS=xtensa-softmmu,xtensaeb-softmmu
|
||||
- TARGETS=mips-linux-user,mips64-linux-user,mips64el-linux-user,mipsel-linux-user,mipsn32-linux-user,mipsn32el-linux-user
|
||||
- TARGETS=or32-softmmu,or32-linux-user
|
||||
- TARGETS=ppc-softmmu,ppc64-softmmu,ppcemb-softmmu,ppc-linux-user,ppc64-linux-user,ppc64abi32-linux-user,ppc64le-linux-user
|
||||
- TARGETS=s390x-softmmu,s390x-linux-user
|
||||
- TARGETS=sh4-softmmu,sh4eb-softmmu,sh4-linux-user sh4eb-linux-user
|
||||
- TARGETS=sparc-softmmu,sparc64-softmmu,sparc-linux-user,sparc32plus-linux-user,sparc64-linux-user
|
||||
- TARGETS=unicore32-softmmu,unicore32-linux-user
|
||||
# Group remaining softmmu only targets into one build
|
||||
- TARGETS=lm32-softmmu,moxie-softmmu,tricore-softmmu,xtensa-softmmu,xtensaeb-softmmu
|
||||
git:
|
||||
# we want to do this ourselves
|
||||
submodules: false
|
||||
before_install:
|
||||
- wget -O - http://people.linaro.org/~alex.bennee/qemu-submodule-git-seed.tar.xz | tar -xvJ
|
||||
- git submodule update --init --recursive
|
||||
- sudo apt-get update -qq
|
||||
- sudo apt-get install -qq ${CORE_PKGS} ${NET_PKGS} ${GUI_PKGS} ${EXTRA_PKGS}
|
||||
script: "./configure --target-list=${TARGETS} ${EXTRA_CONFIG} && make && ${TEST_CMD}"
|
||||
before_script:
|
||||
- ./configure --target-list=${TARGETS} --enable-debug-tcg ${EXTRA_CONFIG}
|
||||
script:
|
||||
- make -j2 && ${TEST_CMD}
|
||||
matrix:
|
||||
# We manually include a number of additional build for non-standard bits
|
||||
include:
|
||||
# Make check target (we only do this once)
|
||||
- env:
|
||||
- TARGETS=alpha-softmmu,arm-softmmu,aarch64-softmmu,cris-softmmu,
|
||||
i386-softmmu,x86_64-softmmu,m68k-softmmu,microblaze-softmmu,
|
||||
microblazeel-softmmu,mips-softmmu,mips64-softmmu,
|
||||
mips64el-softmmu,mipsel-softmmu,or32-softmmu,ppc-softmmu,
|
||||
ppc64-softmmu,ppcemb-softmmu,s390x-softmmu,sh4-softmmu,
|
||||
sh4eb-softmmu,sparc-softmmu,sparc64-softmmu,
|
||||
unicore32-softmmu,unicore32-linux-user,
|
||||
lm32-softmmu,moxie-softmmu,tricore-softmmu,xtensa-softmmu,
|
||||
xtensaeb-softmmu
|
||||
TEST_CMD="make check"
|
||||
compiler: gcc
|
||||
# Debug related options
|
||||
- env: TARGETS=i386-softmmu,x86_64-softmmu
|
||||
EXTRA_CONFIG="--enable-debug"
|
||||
@@ -73,7 +93,6 @@ matrix:
|
||||
compiler: gcc
|
||||
- env: TARGETS=i386-softmmu,x86_64-softmmu
|
||||
EXTRA_CONFIG="--enable-trace-backends=ftrace"
|
||||
TEST_CMD=""
|
||||
compiler: gcc
|
||||
- env: TARGETS=i386-softmmu,x86_64-softmmu
|
||||
EXTRA_PKGS="liblttng-ust-dev liburcu-dev"
|
||||
|
1
Makefile
1
Makefile
@@ -418,6 +418,7 @@ endif
|
||||
set -e; for x in $(KEYMAPS); do \
|
||||
$(INSTALL_DATA) $(SRC_PATH)/pc-bios/keymaps/$$x "$(DESTDIR)$(qemu_datadir)/keymaps"; \
|
||||
done
|
||||
$(INSTALL_DATA) $(SRC_PATH)/trace-events "$(DESTDIR)$(qemu_datadir)/trace-events"
|
||||
for d in $(TARGET_DIRS); do \
|
||||
$(MAKE) $(SUBDIR_MAKEFLAGS) TARGET_DIR=$$d/ -C $$d $@ || exit 1 ; \
|
||||
done
|
||||
|
21
block.c
21
block.c
@@ -29,6 +29,7 @@
|
||||
#include "qemu/module.h"
|
||||
#include "qapi/qmp/qjson.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
#include "sysemu/blockdev.h" /* FIXME layering violation */
|
||||
#include "qemu/notify.h"
|
||||
#include "block/coroutine.h"
|
||||
#include "block/qapi.h"
|
||||
@@ -334,19 +335,30 @@ void bdrv_register(BlockDriver *bdrv)
|
||||
QLIST_INSERT_HEAD(&bdrv_drivers, bdrv, list);
|
||||
}
|
||||
|
||||
static bool bdrv_is_valid_name(const char *name)
|
||||
{
|
||||
return qemu_opts_id_wellformed(name);
|
||||
}
|
||||
|
||||
/* create a new block device (by default it is empty) */
|
||||
BlockDriverState *bdrv_new(const char *device_name, Error **errp)
|
||||
{
|
||||
BlockDriverState *bs;
|
||||
int i;
|
||||
|
||||
if (*device_name && !bdrv_is_valid_name(device_name)) {
|
||||
error_setg(errp, "Invalid device name");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (bdrv_find(device_name)) {
|
||||
error_setg(errp, "Device with id '%s' already exists",
|
||||
device_name);
|
||||
return NULL;
|
||||
}
|
||||
if (bdrv_find_node(device_name)) {
|
||||
error_setg(errp, "Device with node-name '%s' already exists",
|
||||
error_setg(errp,
|
||||
"Device name '%s' conflicts with an existing node name",
|
||||
device_name);
|
||||
return NULL;
|
||||
}
|
||||
@@ -861,9 +873,9 @@ static void bdrv_assign_node_name(BlockDriverState *bs,
|
||||
return;
|
||||
}
|
||||
|
||||
/* empty string node name is invalid */
|
||||
if (node_name[0] == '\0') {
|
||||
error_setg(errp, "Empty node name");
|
||||
/* Check for empty string or invalid characters */
|
||||
if (!bdrv_is_valid_name(node_name)) {
|
||||
error_setg(errp, "Invalid node name");
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -2110,6 +2122,7 @@ static void bdrv_delete(BlockDriverState *bs)
|
||||
/* remove from list, if necessary */
|
||||
bdrv_make_anon(bs);
|
||||
|
||||
drive_info_del(drive_get_by_blockdev(bs));
|
||||
g_free(bs);
|
||||
}
|
||||
|
||||
|
@@ -214,6 +214,7 @@ static int get_event_by_name(const char *name, BlkDebugEvent *event)
|
||||
struct add_rule_data {
|
||||
BDRVBlkdebugState *s;
|
||||
int action;
|
||||
Error **errp;
|
||||
};
|
||||
|
||||
static int add_rule(QemuOpts *opts, void *opaque)
|
||||
@@ -226,7 +227,11 @@ static int add_rule(QemuOpts *opts, void *opaque)
|
||||
|
||||
/* Find the right event for the rule */
|
||||
event_name = qemu_opt_get(opts, "event");
|
||||
if (!event_name || get_event_by_name(event_name, &event) < 0) {
|
||||
if (!event_name) {
|
||||
error_setg(d->errp, "Missing event name for rule");
|
||||
return -1;
|
||||
} else if (get_event_by_name(event_name, &event) < 0) {
|
||||
error_setg(d->errp, "Invalid event name \"%s\"", event_name);
|
||||
return -1;
|
||||
}
|
||||
|
||||
@@ -312,10 +317,21 @@ static int read_config(BDRVBlkdebugState *s, const char *filename,
|
||||
|
||||
d.s = s;
|
||||
d.action = ACTION_INJECT_ERROR;
|
||||
qemu_opts_foreach(&inject_error_opts, add_rule, &d, 0);
|
||||
d.errp = &local_err;
|
||||
qemu_opts_foreach(&inject_error_opts, add_rule, &d, 1);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
ret = -EINVAL;
|
||||
goto fail;
|
||||
}
|
||||
|
||||
d.action = ACTION_SET_STATE;
|
||||
qemu_opts_foreach(&set_state_opts, add_rule, &d, 0);
|
||||
qemu_opts_foreach(&set_state_opts, add_rule, &d, 1);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
ret = -EINVAL;
|
||||
goto fail;
|
||||
}
|
||||
|
||||
ret = 0;
|
||||
fail:
|
||||
|
@@ -34,7 +34,6 @@
|
||||
#include "qemu/bitops.h"
|
||||
#include "qemu/bitmap.h"
|
||||
#include "block/block_int.h"
|
||||
#include "trace.h"
|
||||
#include "block/scsi.h"
|
||||
#include "qemu/iov.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
@@ -317,13 +316,6 @@ static bool is_request_lun_aligned(int64_t sector_num, int nb_sectors,
|
||||
return 1;
|
||||
}
|
||||
|
||||
static unsigned long *iscsi_allocationmap_init(IscsiLun *iscsilun)
|
||||
{
|
||||
return bitmap_try_new(DIV_ROUND_UP(sector_lun2qemu(iscsilun->num_blocks,
|
||||
iscsilun),
|
||||
iscsilun->cluster_sectors));
|
||||
}
|
||||
|
||||
static void iscsi_allocationmap_set(IscsiLun *iscsilun, int64_t sector_num,
|
||||
int nb_sectors)
|
||||
{
|
||||
@@ -1410,10 +1402,9 @@ static int iscsi_open(BlockDriverState *bs, QDict *options, int flags,
|
||||
iscsilun->cluster_sectors = (iscsilun->bl.opt_unmap_gran *
|
||||
iscsilun->block_size) >> BDRV_SECTOR_BITS;
|
||||
if (iscsilun->lbprz && !(bs->open_flags & BDRV_O_NOCACHE)) {
|
||||
iscsilun->allocationmap = iscsi_allocationmap_init(iscsilun);
|
||||
if (iscsilun->allocationmap == NULL) {
|
||||
ret = -ENOMEM;
|
||||
}
|
||||
iscsilun->allocationmap =
|
||||
bitmap_new(DIV_ROUND_UP(bs->total_sectors,
|
||||
iscsilun->cluster_sectors));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1506,7 +1497,10 @@ static int iscsi_truncate(BlockDriverState *bs, int64_t offset)
|
||||
|
||||
if (iscsilun->allocationmap != NULL) {
|
||||
g_free(iscsilun->allocationmap);
|
||||
iscsilun->allocationmap = iscsi_allocationmap_init(iscsilun);
|
||||
iscsilun->allocationmap =
|
||||
bitmap_new(DIV_ROUND_UP(sector_lun2qemu(iscsilun->num_blocks,
|
||||
iscsilun),
|
||||
iscsilun->cluster_sectors));
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@@ -1416,16 +1416,21 @@ static int raw_create(const char *filename, QemuOpts *opts, Error **errp)
|
||||
goto out_close;
|
||||
}
|
||||
|
||||
if (prealloc == PREALLOC_MODE_FALLOC) {
|
||||
switch (prealloc) {
|
||||
#ifdef CONFIG_POSIX_FALLOCATE
|
||||
case PREALLOC_MODE_FALLOC:
|
||||
/* posix_fallocate() doesn't set errno. */
|
||||
result = -posix_fallocate(fd, 0, total_size);
|
||||
if (result != 0) {
|
||||
error_setg_errno(errp, -result,
|
||||
"Could not preallocate data for the new file");
|
||||
}
|
||||
} else if (prealloc == PREALLOC_MODE_FULL) {
|
||||
buf = g_malloc0(65536);
|
||||
break;
|
||||
#endif
|
||||
case PREALLOC_MODE_FULL:
|
||||
{
|
||||
int64_t num = 0, left = total_size;
|
||||
buf = g_malloc0(65536);
|
||||
|
||||
while (left > 0) {
|
||||
num = MIN(left, 65536);
|
||||
@@ -1440,10 +1445,15 @@ static int raw_create(const char *filename, QemuOpts *opts, Error **errp)
|
||||
}
|
||||
fsync(fd);
|
||||
g_free(buf);
|
||||
} else if (prealloc != PREALLOC_MODE_OFF) {
|
||||
break;
|
||||
}
|
||||
case PREALLOC_MODE_OFF:
|
||||
break;
|
||||
default:
|
||||
result = -EINVAL;
|
||||
error_setg(errp, "Unsupported preallocation mode: %s",
|
||||
PreallocMode_lookup[prealloc]);
|
||||
break;
|
||||
}
|
||||
|
||||
out_close:
|
||||
|
44
block/vpc.c
44
block/vpc.c
@@ -207,7 +207,7 @@ static int vpc_open(BlockDriverState *bs, QDict *options, int flags,
|
||||
"incorrect.\n", bs->filename);
|
||||
|
||||
/* Write 'checksum' back to footer, or else will leave it with zero. */
|
||||
footer->checksum = be32_to_cpu(checksum);
|
||||
footer->checksum = cpu_to_be32(checksum);
|
||||
|
||||
// The visible size of a image in Virtual PC depends on the geometry
|
||||
// rather than on the size stored in the footer (the size in the footer
|
||||
@@ -472,7 +472,7 @@ static int64_t alloc_block(BlockDriverState* bs, int64_t sector_num)
|
||||
|
||||
// Write BAT entry to disk
|
||||
bat_offset = s->bat_offset + (4 * index);
|
||||
bat_value = be32_to_cpu(s->pagetable[index]);
|
||||
bat_value = cpu_to_be32(s->pagetable[index]);
|
||||
ret = bdrv_pwrite_sync(bs->file, bat_offset, &bat_value, 4);
|
||||
if (ret < 0)
|
||||
goto fail;
|
||||
@@ -699,13 +699,13 @@ static int create_dynamic_disk(BlockDriverState *bs, uint8_t *buf,
|
||||
* Note: The spec is actually wrong here for data_offset, it says
|
||||
* 0xFFFFFFFF, but MS tools expect all 64 bits to be set.
|
||||
*/
|
||||
dyndisk_header->data_offset = be64_to_cpu(0xFFFFFFFFFFFFFFFFULL);
|
||||
dyndisk_header->table_offset = be64_to_cpu(3 * 512);
|
||||
dyndisk_header->version = be32_to_cpu(0x00010000);
|
||||
dyndisk_header->block_size = be32_to_cpu(block_size);
|
||||
dyndisk_header->max_table_entries = be32_to_cpu(num_bat_entries);
|
||||
dyndisk_header->data_offset = cpu_to_be64(0xFFFFFFFFFFFFFFFFULL);
|
||||
dyndisk_header->table_offset = cpu_to_be64(3 * 512);
|
||||
dyndisk_header->version = cpu_to_be32(0x00010000);
|
||||
dyndisk_header->block_size = cpu_to_be32(block_size);
|
||||
dyndisk_header->max_table_entries = cpu_to_be32(num_bat_entries);
|
||||
|
||||
dyndisk_header->checksum = be32_to_cpu(vpc_checksum(buf, 1024));
|
||||
dyndisk_header->checksum = cpu_to_be32(vpc_checksum(buf, 1024));
|
||||
|
||||
// Write the header
|
||||
offset = 512;
|
||||
@@ -810,36 +810,36 @@ static int vpc_create(const char *filename, QemuOpts *opts, Error **errp)
|
||||
memcpy(footer->creator_app, "qemu", 4);
|
||||
memcpy(footer->creator_os, "Wi2k", 4);
|
||||
|
||||
footer->features = be32_to_cpu(0x02);
|
||||
footer->version = be32_to_cpu(0x00010000);
|
||||
footer->features = cpu_to_be32(0x02);
|
||||
footer->version = cpu_to_be32(0x00010000);
|
||||
if (disk_type == VHD_DYNAMIC) {
|
||||
footer->data_offset = be64_to_cpu(HEADER_SIZE);
|
||||
footer->data_offset = cpu_to_be64(HEADER_SIZE);
|
||||
} else {
|
||||
footer->data_offset = be64_to_cpu(0xFFFFFFFFFFFFFFFFULL);
|
||||
footer->data_offset = cpu_to_be64(0xFFFFFFFFFFFFFFFFULL);
|
||||
}
|
||||
footer->timestamp = be32_to_cpu(time(NULL) - VHD_TIMESTAMP_BASE);
|
||||
footer->timestamp = cpu_to_be32(time(NULL) - VHD_TIMESTAMP_BASE);
|
||||
|
||||
/* Version of Virtual PC 2007 */
|
||||
footer->major = be16_to_cpu(0x0005);
|
||||
footer->minor = be16_to_cpu(0x0003);
|
||||
footer->major = cpu_to_be16(0x0005);
|
||||
footer->minor = cpu_to_be16(0x0003);
|
||||
if (disk_type == VHD_DYNAMIC) {
|
||||
footer->orig_size = be64_to_cpu(total_sectors * 512);
|
||||
footer->size = be64_to_cpu(total_sectors * 512);
|
||||
footer->orig_size = cpu_to_be64(total_sectors * 512);
|
||||
footer->size = cpu_to_be64(total_sectors * 512);
|
||||
} else {
|
||||
footer->orig_size = be64_to_cpu(total_size);
|
||||
footer->size = be64_to_cpu(total_size);
|
||||
footer->orig_size = cpu_to_be64(total_size);
|
||||
footer->size = cpu_to_be64(total_size);
|
||||
}
|
||||
footer->cyls = be16_to_cpu(cyls);
|
||||
footer->cyls = cpu_to_be16(cyls);
|
||||
footer->heads = heads;
|
||||
footer->secs_per_cyl = secs_per_cyl;
|
||||
|
||||
footer->type = be32_to_cpu(disk_type);
|
||||
footer->type = cpu_to_be32(disk_type);
|
||||
|
||||
#if defined(CONFIG_UUID)
|
||||
uuid_generate(footer->uuid);
|
||||
#endif
|
||||
|
||||
footer->checksum = be32_to_cpu(vpc_checksum(buf, HEADER_SIZE));
|
||||
footer->checksum = cpu_to_be32(vpc_checksum(buf, HEADER_SIZE));
|
||||
|
||||
if (disk_type == VHD_DYNAMIC) {
|
||||
ret = create_dynamic_disk(bs, buf, total_sectors);
|
||||
|
103
blockdev.c
103
blockdev.c
@@ -216,11 +216,17 @@ static void bdrv_format_print(void *opaque, const char *name)
|
||||
|
||||
void drive_del(DriveInfo *dinfo)
|
||||
{
|
||||
bdrv_unref(dinfo->bdrv);
|
||||
}
|
||||
|
||||
void drive_info_del(DriveInfo *dinfo)
|
||||
{
|
||||
if (!dinfo) {
|
||||
return;
|
||||
}
|
||||
if (dinfo->opts) {
|
||||
qemu_opts_del(dinfo->opts);
|
||||
}
|
||||
|
||||
bdrv_unref(dinfo->bdrv);
|
||||
g_free(dinfo->id);
|
||||
QTAILQ_REMOVE(&drives, dinfo, next);
|
||||
g_free(dinfo->serial);
|
||||
@@ -301,6 +307,7 @@ static DriveInfo *blockdev_init(const char *file, QDict *bs_opts,
|
||||
int ro = 0;
|
||||
int bdrv_flags = 0;
|
||||
int on_read_error, on_write_error;
|
||||
BlockDriverState *bs;
|
||||
DriveInfo *dinfo;
|
||||
ThrottleConfig cfg;
|
||||
int snapshot = 0;
|
||||
@@ -456,26 +463,27 @@ static DriveInfo *blockdev_init(const char *file, QDict *bs_opts,
|
||||
}
|
||||
|
||||
/* init */
|
||||
dinfo = g_malloc0(sizeof(*dinfo));
|
||||
dinfo->id = g_strdup(qemu_opts_id(opts));
|
||||
dinfo->bdrv = bdrv_new(dinfo->id, &error);
|
||||
if (error) {
|
||||
error_propagate(errp, error);
|
||||
goto bdrv_new_err;
|
||||
bs = bdrv_new(qemu_opts_id(opts), errp);
|
||||
if (!bs) {
|
||||
goto early_err;
|
||||
}
|
||||
dinfo->bdrv->open_flags = snapshot ? BDRV_O_SNAPSHOT : 0;
|
||||
dinfo->bdrv->read_only = ro;
|
||||
dinfo->bdrv->detect_zeroes = detect_zeroes;
|
||||
QTAILQ_INSERT_TAIL(&drives, dinfo, next);
|
||||
bs->open_flags = snapshot ? BDRV_O_SNAPSHOT : 0;
|
||||
bs->read_only = ro;
|
||||
bs->detect_zeroes = detect_zeroes;
|
||||
|
||||
bdrv_set_on_error(dinfo->bdrv, on_read_error, on_write_error);
|
||||
bdrv_set_on_error(bs, on_read_error, on_write_error);
|
||||
|
||||
/* disk I/O throttling */
|
||||
if (throttle_enabled(&cfg)) {
|
||||
bdrv_io_limits_enable(dinfo->bdrv);
|
||||
bdrv_set_io_limits(dinfo->bdrv, &cfg);
|
||||
bdrv_io_limits_enable(bs);
|
||||
bdrv_set_io_limits(bs, &cfg);
|
||||
}
|
||||
|
||||
dinfo = g_malloc0(sizeof(*dinfo));
|
||||
dinfo->id = g_strdup(qemu_opts_id(opts));
|
||||
dinfo->bdrv = bs;
|
||||
QTAILQ_INSERT_TAIL(&drives, dinfo, next);
|
||||
|
||||
if (!file || !*file) {
|
||||
if (has_driver_specific_opts) {
|
||||
file = NULL;
|
||||
@@ -502,7 +510,8 @@ static DriveInfo *blockdev_init(const char *file, QDict *bs_opts,
|
||||
bdrv_flags |= ro ? 0 : BDRV_O_RDWR;
|
||||
|
||||
QINCREF(bs_opts);
|
||||
ret = bdrv_open(&dinfo->bdrv, file, NULL, bs_opts, bdrv_flags, drv, &error);
|
||||
ret = bdrv_open(&bs, file, NULL, bs_opts, bdrv_flags, drv, &error);
|
||||
assert(bs == dinfo->bdrv);
|
||||
|
||||
if (ret < 0) {
|
||||
error_setg(errp, "could not open disk image %s: %s",
|
||||
@@ -511,8 +520,9 @@ static DriveInfo *blockdev_init(const char *file, QDict *bs_opts,
|
||||
goto err;
|
||||
}
|
||||
|
||||
if (bdrv_key_required(dinfo->bdrv))
|
||||
if (bdrv_key_required(bs)) {
|
||||
autostart = 0;
|
||||
}
|
||||
|
||||
QDECREF(bs_opts);
|
||||
qemu_opts_del(opts);
|
||||
@@ -520,11 +530,7 @@ static DriveInfo *blockdev_init(const char *file, QDict *bs_opts,
|
||||
return dinfo;
|
||||
|
||||
err:
|
||||
bdrv_unref(dinfo->bdrv);
|
||||
QTAILQ_REMOVE(&drives, dinfo, next);
|
||||
bdrv_new_err:
|
||||
g_free(dinfo->id);
|
||||
g_free(dinfo);
|
||||
bdrv_unref(bs);
|
||||
early_err:
|
||||
qemu_opts_del(opts);
|
||||
err_no_opts:
|
||||
@@ -532,12 +538,18 @@ err_no_opts:
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void qemu_opt_rename(QemuOpts *opts, const char *from, const char *to)
|
||||
static void qemu_opt_rename(QemuOpts *opts, const char *from, const char *to,
|
||||
Error **errp)
|
||||
{
|
||||
const char *value;
|
||||
|
||||
value = qemu_opt_get(opts, from);
|
||||
if (value) {
|
||||
if (qemu_opt_find(opts, to)) {
|
||||
error_setg(errp, "'%s' and its alias '%s' can't be used at the "
|
||||
"same time", to, from);
|
||||
return;
|
||||
}
|
||||
qemu_opt_set(opts, to, value);
|
||||
qemu_opt_unset(opts, from);
|
||||
}
|
||||
@@ -641,28 +653,43 @@ DriveInfo *drive_new(QemuOpts *all_opts, BlockInterfaceType block_default_type)
|
||||
const char *serial;
|
||||
const char *filename;
|
||||
Error *local_err = NULL;
|
||||
int i;
|
||||
|
||||
/* Change legacy command line options into QMP ones */
|
||||
qemu_opt_rename(all_opts, "iops", "throttling.iops-total");
|
||||
qemu_opt_rename(all_opts, "iops_rd", "throttling.iops-read");
|
||||
qemu_opt_rename(all_opts, "iops_wr", "throttling.iops-write");
|
||||
static const struct {
|
||||
const char *from;
|
||||
const char *to;
|
||||
} opt_renames[] = {
|
||||
{ "iops", "throttling.iops-total" },
|
||||
{ "iops_rd", "throttling.iops-read" },
|
||||
{ "iops_wr", "throttling.iops-write" },
|
||||
|
||||
qemu_opt_rename(all_opts, "bps", "throttling.bps-total");
|
||||
qemu_opt_rename(all_opts, "bps_rd", "throttling.bps-read");
|
||||
qemu_opt_rename(all_opts, "bps_wr", "throttling.bps-write");
|
||||
{ "bps", "throttling.bps-total" },
|
||||
{ "bps_rd", "throttling.bps-read" },
|
||||
{ "bps_wr", "throttling.bps-write" },
|
||||
|
||||
qemu_opt_rename(all_opts, "iops_max", "throttling.iops-total-max");
|
||||
qemu_opt_rename(all_opts, "iops_rd_max", "throttling.iops-read-max");
|
||||
qemu_opt_rename(all_opts, "iops_wr_max", "throttling.iops-write-max");
|
||||
{ "iops_max", "throttling.iops-total-max" },
|
||||
{ "iops_rd_max", "throttling.iops-read-max" },
|
||||
{ "iops_wr_max", "throttling.iops-write-max" },
|
||||
|
||||
qemu_opt_rename(all_opts, "bps_max", "throttling.bps-total-max");
|
||||
qemu_opt_rename(all_opts, "bps_rd_max", "throttling.bps-read-max");
|
||||
qemu_opt_rename(all_opts, "bps_wr_max", "throttling.bps-write-max");
|
||||
{ "bps_max", "throttling.bps-total-max" },
|
||||
{ "bps_rd_max", "throttling.bps-read-max" },
|
||||
{ "bps_wr_max", "throttling.bps-write-max" },
|
||||
|
||||
qemu_opt_rename(all_opts,
|
||||
"iops_size", "throttling.iops-size");
|
||||
{ "iops_size", "throttling.iops-size" },
|
||||
|
||||
qemu_opt_rename(all_opts, "readonly", "read-only");
|
||||
{ "readonly", "read-only" },
|
||||
};
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(opt_renames); i++) {
|
||||
qemu_opt_rename(all_opts, opt_renames[i].from, opt_renames[i].to,
|
||||
&local_err);
|
||||
if (local_err) {
|
||||
error_report("%s", error_get_pretty(local_err));
|
||||
error_free(local_err);
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
value = qemu_opt_get(all_opts, "cache");
|
||||
if (value) {
|
||||
|
23
configure
vendored
23
configure
vendored
@@ -3308,6 +3308,21 @@ if compile_prog "" "" ; then
|
||||
fallocate_punch_hole=yes
|
||||
fi
|
||||
|
||||
# check for posix_fallocate
|
||||
posix_fallocate=no
|
||||
cat > $TMPC << EOF
|
||||
#include <fcntl.h>
|
||||
|
||||
int main(void)
|
||||
{
|
||||
posix_fallocate(0, 0, 0);
|
||||
return 0;
|
||||
}
|
||||
EOF
|
||||
if compile_prog "" "" ; then
|
||||
posix_fallocate=yes
|
||||
fi
|
||||
|
||||
# check for sync_file_range
|
||||
sync_file_range=no
|
||||
cat > $TMPC << EOF
|
||||
@@ -3956,12 +3971,11 @@ else
|
||||
fi
|
||||
|
||||
########################################
|
||||
# check if we have valgrind/valgrind.h and valgrind/memcheck.h
|
||||
# check if we have valgrind/valgrind.h
|
||||
|
||||
valgrind_h=no
|
||||
cat > $TMPC << EOF
|
||||
#include <valgrind/valgrind.h>
|
||||
#include <valgrind/memcheck.h>
|
||||
int main(void) {
|
||||
return 0;
|
||||
}
|
||||
@@ -4523,6 +4537,9 @@ fi
|
||||
if test "$fallocate_punch_hole" = "yes" ; then
|
||||
echo "CONFIG_FALLOCATE_PUNCH_HOLE=y" >> $config_host_mak
|
||||
fi
|
||||
if test "$posix_fallocate" = "yes" ; then
|
||||
echo "CONFIG_POSIX_FALLOCATE=y" >> $config_host_mak
|
||||
fi
|
||||
if test "$sync_file_range" = "yes" ; then
|
||||
echo "CONFIG_SYNC_FILE_RANGE=y" >> $config_host_mak
|
||||
fi
|
||||
@@ -5011,7 +5028,7 @@ case "$target_name" in
|
||||
aarch64)
|
||||
TARGET_BASE_ARCH=arm
|
||||
bflt="yes"
|
||||
gdb_xml_files="aarch64-core.xml aarch64-fpu.xml"
|
||||
gdb_xml_files="aarch64-core.xml aarch64-fpu.xml arm-core.xml arm-vfp.xml arm-vfp3.xml arm-neon.xml"
|
||||
;;
|
||||
cris)
|
||||
;;
|
||||
|
329
cpu-exec.c
329
cpu-exec.c
@@ -317,10 +317,7 @@ volatile sig_atomic_t exit_request;
|
||||
int cpu_exec(CPUArchState *env)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(env);
|
||||
#if !(defined(CONFIG_USER_ONLY) && \
|
||||
(defined(TARGET_M68K) || defined(TARGET_PPC) || defined(TARGET_S390X)))
|
||||
CPUClass *cc = CPU_GET_CLASS(cpu);
|
||||
#endif
|
||||
#ifdef TARGET_I386
|
||||
X86CPU *x86_cpu = X86_CPU(cpu);
|
||||
#endif
|
||||
@@ -355,36 +352,7 @@ int cpu_exec(CPUArchState *env)
|
||||
cpu->exit_request = 1;
|
||||
}
|
||||
|
||||
#if defined(TARGET_I386)
|
||||
/* put eflags in CPU temporary format */
|
||||
CC_SRC = env->eflags & (CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C);
|
||||
env->df = 1 - (2 * ((env->eflags >> 10) & 1));
|
||||
CC_OP = CC_OP_EFLAGS;
|
||||
env->eflags &= ~(DF_MASK | CC_O | CC_S | CC_Z | CC_A | CC_P | CC_C);
|
||||
#elif defined(TARGET_SPARC)
|
||||
#elif defined(TARGET_M68K)
|
||||
env->cc_op = CC_OP_FLAGS;
|
||||
env->cc_dest = env->sr & 0xf;
|
||||
env->cc_x = (env->sr >> 4) & 1;
|
||||
#elif defined(TARGET_ALPHA)
|
||||
#elif defined(TARGET_ARM)
|
||||
#elif defined(TARGET_UNICORE32)
|
||||
#elif defined(TARGET_PPC)
|
||||
env->reserve_addr = -1;
|
||||
#elif defined(TARGET_LM32)
|
||||
#elif defined(TARGET_MICROBLAZE)
|
||||
#elif defined(TARGET_MIPS)
|
||||
#elif defined(TARGET_MOXIE)
|
||||
#elif defined(TARGET_OPENRISC)
|
||||
#elif defined(TARGET_SH4)
|
||||
#elif defined(TARGET_CRIS)
|
||||
#elif defined(TARGET_S390X)
|
||||
#elif defined(TARGET_XTENSA)
|
||||
#elif defined(TARGET_TRICORE)
|
||||
/* XXXXX */
|
||||
#else
|
||||
#error unsupported target CPU
|
||||
#endif
|
||||
cc->cpu_exec_enter(cpu);
|
||||
cpu->exception_index = -1;
|
||||
|
||||
/* Calculate difference between guest clock and host clock.
|
||||
@@ -436,17 +404,12 @@ int cpu_exec(CPUArchState *env)
|
||||
cpu->exception_index = EXCP_DEBUG;
|
||||
cpu_loop_exit(cpu);
|
||||
}
|
||||
#if defined(TARGET_ARM) || defined(TARGET_SPARC) || defined(TARGET_MIPS) || \
|
||||
defined(TARGET_PPC) || defined(TARGET_ALPHA) || defined(TARGET_CRIS) || \
|
||||
defined(TARGET_MICROBLAZE) || defined(TARGET_LM32) || \
|
||||
defined(TARGET_UNICORE32) || defined(TARGET_TRICORE)
|
||||
if (interrupt_request & CPU_INTERRUPT_HALT) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_HALT;
|
||||
cpu->halted = 1;
|
||||
cpu->exception_index = EXCP_HLT;
|
||||
cpu_loop_exit(cpu);
|
||||
}
|
||||
#endif
|
||||
#if defined(TARGET_I386)
|
||||
if (interrupt_request & CPU_INTERRUPT_INIT) {
|
||||
cpu_svm_check_intercept_param(env, SVM_EXIT_INIT, 0);
|
||||
@@ -459,257 +422,15 @@ int cpu_exec(CPUArchState *env)
|
||||
cpu_reset(cpu);
|
||||
}
|
||||
#endif
|
||||
#if defined(TARGET_I386)
|
||||
#if !defined(CONFIG_USER_ONLY)
|
||||
if (interrupt_request & CPU_INTERRUPT_POLL) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_POLL;
|
||||
apic_poll_irq(x86_cpu->apic_state);
|
||||
}
|
||||
#endif
|
||||
if (interrupt_request & CPU_INTERRUPT_SIPI) {
|
||||
do_cpu_sipi(x86_cpu);
|
||||
} else if (env->hflags2 & HF2_GIF_MASK) {
|
||||
if ((interrupt_request & CPU_INTERRUPT_SMI) &&
|
||||
!(env->hflags & HF_SMM_MASK)) {
|
||||
cpu_svm_check_intercept_param(env, SVM_EXIT_SMI,
|
||||
0);
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_SMI;
|
||||
do_smm_enter(x86_cpu);
|
||||
next_tb = 0;
|
||||
} else if ((interrupt_request & CPU_INTERRUPT_NMI) &&
|
||||
!(env->hflags2 & HF2_NMI_MASK)) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_NMI;
|
||||
env->hflags2 |= HF2_NMI_MASK;
|
||||
do_interrupt_x86_hardirq(env, EXCP02_NMI, 1);
|
||||
next_tb = 0;
|
||||
} else if (interrupt_request & CPU_INTERRUPT_MCE) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_MCE;
|
||||
do_interrupt_x86_hardirq(env, EXCP12_MCHK, 0);
|
||||
next_tb = 0;
|
||||
} else if ((interrupt_request & CPU_INTERRUPT_HARD) &&
|
||||
(((env->hflags2 & HF2_VINTR_MASK) &&
|
||||
(env->hflags2 & HF2_HIF_MASK)) ||
|
||||
(!(env->hflags2 & HF2_VINTR_MASK) &&
|
||||
(env->eflags & IF_MASK &&
|
||||
!(env->hflags & HF_INHIBIT_IRQ_MASK))))) {
|
||||
int intno;
|
||||
cpu_svm_check_intercept_param(env, SVM_EXIT_INTR,
|
||||
0);
|
||||
cpu->interrupt_request &= ~(CPU_INTERRUPT_HARD |
|
||||
CPU_INTERRUPT_VIRQ);
|
||||
intno = cpu_get_pic_interrupt(env);
|
||||
qemu_log_mask(CPU_LOG_TB_IN_ASM, "Servicing hardware INT=0x%02x\n", intno);
|
||||
do_interrupt_x86_hardirq(env, intno, 1);
|
||||
/* ensure that no TB jump will be modified as
|
||||
the program flow was changed */
|
||||
next_tb = 0;
|
||||
#if !defined(CONFIG_USER_ONLY)
|
||||
} else if ((interrupt_request & CPU_INTERRUPT_VIRQ) &&
|
||||
(env->eflags & IF_MASK) &&
|
||||
!(env->hflags & HF_INHIBIT_IRQ_MASK)) {
|
||||
int intno;
|
||||
/* FIXME: this should respect TPR */
|
||||
cpu_svm_check_intercept_param(env, SVM_EXIT_VINTR,
|
||||
0);
|
||||
intno = ldl_phys(cpu->as,
|
||||
env->vm_vmcb
|
||||
+ offsetof(struct vmcb,
|
||||
control.int_vector));
|
||||
qemu_log_mask(CPU_LOG_TB_IN_ASM, "Servicing virtual hardware INT=0x%02x\n", intno);
|
||||
do_interrupt_x86_hardirq(env, intno, 1);
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_VIRQ;
|
||||
next_tb = 0;
|
||||
#endif
|
||||
}
|
||||
}
|
||||
#elif defined(TARGET_PPC)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD) {
|
||||
ppc_hw_interrupt(env);
|
||||
if (env->pending_interrupts == 0) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_HARD;
|
||||
}
|
||||
/* The target hook has 3 exit conditions:
|
||||
False when the interrupt isn't processed,
|
||||
True when it is, and we should restart on a new TB,
|
||||
and via longjmp via cpu_loop_exit. */
|
||||
if (cc->cpu_exec_interrupt(cpu, interrupt_request)) {
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_LM32)
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD)
|
||||
&& (env->ie & IE_IE)) {
|
||||
cpu->exception_index = EXCP_IRQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_MICROBLAZE)
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD)
|
||||
&& (env->sregs[SR_MSR] & MSR_IE)
|
||||
&& !(env->sregs[SR_MSR] & (MSR_EIP | MSR_BIP))
|
||||
&& !(env->iflags & (D_FLAG | IMM_FLAG))) {
|
||||
cpu->exception_index = EXCP_IRQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_MIPS)
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD) &&
|
||||
cpu_mips_hw_interrupts_pending(env)) {
|
||||
/* Raise it */
|
||||
cpu->exception_index = EXCP_EXT_INTERRUPT;
|
||||
env->error_code = 0;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_TRICORE)
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD)) {
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
|
||||
#elif defined(TARGET_OPENRISC)
|
||||
{
|
||||
int idx = -1;
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD)
|
||||
&& (env->sr & SR_IEE)) {
|
||||
idx = EXCP_INT;
|
||||
}
|
||||
if ((interrupt_request & CPU_INTERRUPT_TIMER)
|
||||
&& (env->sr & SR_TEE)) {
|
||||
idx = EXCP_TICK;
|
||||
}
|
||||
if (idx >= 0) {
|
||||
cpu->exception_index = idx;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
}
|
||||
#elif defined(TARGET_SPARC)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD) {
|
||||
if (cpu_interrupts_enabled(env) &&
|
||||
env->interrupt_index > 0) {
|
||||
int pil = env->interrupt_index & 0xf;
|
||||
int type = env->interrupt_index & 0xf0;
|
||||
|
||||
if (((type == TT_EXTINT) &&
|
||||
cpu_pil_allowed(env, pil)) ||
|
||||
type != TT_EXTINT) {
|
||||
cpu->exception_index = env->interrupt_index;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
#elif defined(TARGET_ARM)
|
||||
if (interrupt_request & CPU_INTERRUPT_FIQ
|
||||
&& !(env->daif & PSTATE_F)) {
|
||||
cpu->exception_index = EXCP_FIQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
/* ARMv7-M interrupt return works by loading a magic value
|
||||
into the PC. On real hardware the load causes the
|
||||
return to occur. The qemu implementation performs the
|
||||
jump normally, then does the exception return when the
|
||||
CPU tries to execute code at the magic address.
|
||||
This will cause the magic PC value to be pushed to
|
||||
the stack if an interrupt occurred at the wrong time.
|
||||
We avoid this by disabling interrupts when
|
||||
pc contains a magic address. */
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD
|
||||
&& !(env->daif & PSTATE_I)
|
||||
&& (!IS_M(env) || env->regs[15] < 0xfffffff0)) {
|
||||
cpu->exception_index = EXCP_IRQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_UNICORE32)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD
|
||||
&& !(env->uncached_asr & ASR_I)) {
|
||||
cpu->exception_index = UC32_EXCP_INTR;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_SH4)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD) {
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_ALPHA)
|
||||
{
|
||||
int idx = -1;
|
||||
/* ??? This hard-codes the OSF/1 interrupt levels. */
|
||||
switch (env->pal_mode ? 7 : env->ps & PS_INT_MASK) {
|
||||
case 0 ... 3:
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD) {
|
||||
idx = EXCP_DEV_INTERRUPT;
|
||||
}
|
||||
/* FALLTHRU */
|
||||
case 4:
|
||||
if (interrupt_request & CPU_INTERRUPT_TIMER) {
|
||||
idx = EXCP_CLK_INTERRUPT;
|
||||
}
|
||||
/* FALLTHRU */
|
||||
case 5:
|
||||
if (interrupt_request & CPU_INTERRUPT_SMP) {
|
||||
idx = EXCP_SMP_INTERRUPT;
|
||||
}
|
||||
/* FALLTHRU */
|
||||
case 6:
|
||||
if (interrupt_request & CPU_INTERRUPT_MCHK) {
|
||||
idx = EXCP_MCHK;
|
||||
}
|
||||
}
|
||||
if (idx >= 0) {
|
||||
cpu->exception_index = idx;
|
||||
env->error_code = 0;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
}
|
||||
#elif defined(TARGET_CRIS)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD
|
||||
&& (env->pregs[PR_CCS] & I_FLAG)
|
||||
&& !env->locked_irq) {
|
||||
cpu->exception_index = EXCP_IRQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
if (interrupt_request & CPU_INTERRUPT_NMI) {
|
||||
unsigned int m_flag_archval;
|
||||
if (env->pregs[PR_VR] < 32) {
|
||||
m_flag_archval = M_FLAG_V10;
|
||||
} else {
|
||||
m_flag_archval = M_FLAG_V32;
|
||||
}
|
||||
if ((env->pregs[PR_CCS] & m_flag_archval)) {
|
||||
cpu->exception_index = EXCP_NMI;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
}
|
||||
#elif defined(TARGET_M68K)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD
|
||||
&& ((env->sr & SR_I) >> SR_I_SHIFT)
|
||||
< env->pending_level) {
|
||||
/* Real hardware gets the interrupt vector via an
|
||||
IACK cycle at this point. Current emulated
|
||||
hardware doesn't rely on this, so we
|
||||
provide/save the vector when the interrupt is
|
||||
first signalled. */
|
||||
cpu->exception_index = env->pending_vector;
|
||||
do_interrupt_m68k_hardirq(env);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_S390X) && !defined(CONFIG_USER_ONLY)
|
||||
if ((interrupt_request & CPU_INTERRUPT_HARD) &&
|
||||
(env->psw.mask & PSW_MASK_EXT)) {
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#elif defined(TARGET_XTENSA)
|
||||
if (interrupt_request & CPU_INTERRUPT_HARD) {
|
||||
cpu->exception_index = EXC_IRQ;
|
||||
cc->do_interrupt(cpu);
|
||||
next_tb = 0;
|
||||
}
|
||||
#endif
|
||||
/* Don't use the cached interrupt_request value,
|
||||
do_interrupt may have updated the EXITTB flag. */
|
||||
/* Don't use the cached interrupt_request value,
|
||||
do_interrupt may have updated the EXITTB flag. */
|
||||
if (cpu->interrupt_request & CPU_INTERRUPT_EXITTB) {
|
||||
cpu->interrupt_request &= ~CPU_INTERRUPT_EXITTB;
|
||||
/* ensure that no TB jump will be modified as
|
||||
@@ -815,10 +536,7 @@ int cpu_exec(CPUArchState *env)
|
||||
* local variables as longjmp is marked 'noreturn'. */
|
||||
cpu = current_cpu;
|
||||
env = cpu->env_ptr;
|
||||
#if !(defined(CONFIG_USER_ONLY) && \
|
||||
(defined(TARGET_M68K) || defined(TARGET_PPC) || defined(TARGET_S390X)))
|
||||
cc = CPU_GET_CLASS(cpu);
|
||||
#endif
|
||||
#ifdef TARGET_I386
|
||||
x86_cpu = X86_CPU(cpu);
|
||||
#endif
|
||||
@@ -829,36 +547,7 @@ int cpu_exec(CPUArchState *env)
|
||||
}
|
||||
} /* for(;;) */
|
||||
|
||||
|
||||
#if defined(TARGET_I386)
|
||||
/* restore flags in standard format */
|
||||
env->eflags = env->eflags | cpu_cc_compute_all(env, CC_OP)
|
||||
| (env->df & DF_MASK);
|
||||
#elif defined(TARGET_ARM)
|
||||
/* XXX: Save/restore host fpu exception state?. */
|
||||
#elif defined(TARGET_UNICORE32)
|
||||
#elif defined(TARGET_SPARC)
|
||||
#elif defined(TARGET_PPC)
|
||||
#elif defined(TARGET_LM32)
|
||||
#elif defined(TARGET_M68K)
|
||||
cpu_m68k_flush_flags(env, env->cc_op);
|
||||
env->cc_op = CC_OP_FLAGS;
|
||||
env->sr = (env->sr & 0xffe0)
|
||||
| env->cc_dest | (env->cc_x << 4);
|
||||
#elif defined(TARGET_MICROBLAZE)
|
||||
#elif defined(TARGET_MIPS)
|
||||
#elif defined(TARGET_TRICORE)
|
||||
#elif defined(TARGET_MOXIE)
|
||||
#elif defined(TARGET_OPENRISC)
|
||||
#elif defined(TARGET_SH4)
|
||||
#elif defined(TARGET_ALPHA)
|
||||
#elif defined(TARGET_CRIS)
|
||||
#elif defined(TARGET_S390X)
|
||||
#elif defined(TARGET_XTENSA)
|
||||
/* XXXXX */
|
||||
#else
|
||||
#error unsupported target CPU
|
||||
#endif
|
||||
cc->cpu_exec_exit(cpu);
|
||||
|
||||
/* fail safe : never use current_cpu outside cpu_exec() */
|
||||
current_cpu = NULL;
|
||||
|
9
cpus.c
9
cpus.c
@@ -593,6 +593,15 @@ void cpu_synchronize_all_post_init(void)
|
||||
}
|
||||
}
|
||||
|
||||
void cpu_clean_all_dirty(void)
|
||||
{
|
||||
CPUState *cpu;
|
||||
|
||||
CPU_FOREACH(cpu) {
|
||||
cpu_clean_state(cpu);
|
||||
}
|
||||
}
|
||||
|
||||
static int do_vm_stop(RunState state)
|
||||
{
|
||||
int ret = 0;
|
||||
|
161
docs/blkdebug.txt
Normal file
161
docs/blkdebug.txt
Normal file
@@ -0,0 +1,161 @@
|
||||
Block I/O error injection using blkdebug
|
||||
----------------------------------------
|
||||
Copyright (C) 2014 Red Hat Inc
|
||||
|
||||
This work is licensed under the terms of the GNU GPL, version 2 or later. See
|
||||
the COPYING file in the top-level directory.
|
||||
|
||||
The blkdebug block driver is a rule-based error injection engine. It can be
|
||||
used to exercise error code paths in block drivers including ENOSPC (out of
|
||||
space) and EIO.
|
||||
|
||||
This document gives an overview of the features available in blkdebug.
|
||||
|
||||
Background
|
||||
----------
|
||||
Block drivers have many error code paths that handle I/O errors. Image formats
|
||||
are especially complex since metadata I/O errors during cluster allocation or
|
||||
while updating tables happen halfway through request processing and require
|
||||
discipline to keep image files consistent.
|
||||
|
||||
Error injection allows test cases to trigger I/O errors at specific points.
|
||||
This way, all error paths can be tested to make sure they are correct.
|
||||
|
||||
Rules
|
||||
-----
|
||||
The blkdebug block driver takes a list of "rules" that tell the error injection
|
||||
engine when to fail an I/O request.
|
||||
|
||||
Each I/O request is evaluated against the rules. If a rule matches the request
|
||||
then its "action" is executed.
|
||||
|
||||
Rules can be placed in a configuration file; the configuration file
|
||||
follows the same .ini-like format used by QEMU's -readconfig option, and
|
||||
each section of the file represents a rule.
|
||||
|
||||
The following configuration file defines a single rule:
|
||||
|
||||
$ cat blkdebug.conf
|
||||
[inject-error]
|
||||
event = "read_aio"
|
||||
errno = "28"
|
||||
|
||||
This rule fails all aio read requests with ENOSPC (28). Note that the errno
|
||||
value depends on the host. On Linux, see
|
||||
/usr/include/asm-generic/errno-base.h for errno values.
|
||||
|
||||
Invoke QEMU as follows:
|
||||
|
||||
$ qemu-system-x86_64
|
||||
-drive if=none,cache=none,file=blkdebug:blkdebug.conf:test.img,id=drive0 \
|
||||
-device virtio-blk-pci,drive=drive0,id=virtio-blk-pci0
|
||||
|
||||
Rules support the following attributes:
|
||||
|
||||
event - which type of operation to match (e.g. read_aio, write_aio,
|
||||
flush_to_os, flush_to_disk). See the "Events" section for
|
||||
information on events.
|
||||
|
||||
state - (optional) the engine must be in this state number in order for this
|
||||
rule to match. See the "State transitions" section for information
|
||||
on states.
|
||||
|
||||
errno - the numeric errno value to return when a request matches this rule.
|
||||
The errno values depend on the host since the numeric values are not
|
||||
standarized in the POSIX specification.
|
||||
|
||||
sector - (optional) a sector number that the request must overlap in order to
|
||||
match this rule
|
||||
|
||||
once - (optional, default "off") only execute this action on the first
|
||||
matching request
|
||||
|
||||
immediately - (optional, default "off") return a NULL BlockDriverAIOCB
|
||||
pointer and fail without an errno instead. This exercises the
|
||||
code path where BlockDriverAIOCB fails and the caller's
|
||||
BlockDriverCompletionFunc is not invoked.
|
||||
|
||||
Events
|
||||
------
|
||||
Block drivers provide information about the type of I/O request they are about
|
||||
to make so rules can match specific types of requests. For example, the qcow2
|
||||
block driver tells blkdebug when it accesses the L1 table so rules can match
|
||||
only L1 table accesses and not other metadata or guest data requests.
|
||||
|
||||
The core events are:
|
||||
|
||||
read_aio - guest data read
|
||||
|
||||
write_aio - guest data write
|
||||
|
||||
flush_to_os - write out unwritten block driver state (e.g. cached metadata)
|
||||
|
||||
flush_to_disk - flush the host block device's disk cache
|
||||
|
||||
See block/blkdebug.c:event_names[] for the full list of events. You may need
|
||||
to grep block driver source code to understand the meaning of specific events.
|
||||
|
||||
State transitions
|
||||
-----------------
|
||||
There are cases where more power is needed to match a particular I/O request in
|
||||
a longer sequence of requests. For example:
|
||||
|
||||
write_aio
|
||||
flush_to_disk
|
||||
write_aio
|
||||
|
||||
How do we match the 2nd write_aio but not the first? This is where state
|
||||
transitions come in.
|
||||
|
||||
The error injection engine has an integer called the "state" that always starts
|
||||
initialized to 1. The state integer is internal to blkdebug and cannot be
|
||||
observed from outside but rules can interact with it for powerful matching
|
||||
behavior.
|
||||
|
||||
Rules can be conditional on the current state and they can transition to a new
|
||||
state.
|
||||
|
||||
When a rule's "state" attribute is non-zero then the current state must equal
|
||||
the attribute in order for the rule to match.
|
||||
|
||||
For example, to match the 2nd write_aio:
|
||||
|
||||
[set-state]
|
||||
event = "write_aio"
|
||||
state = "1"
|
||||
new_state = "2"
|
||||
|
||||
[inject-error]
|
||||
event = "write_aio"
|
||||
state = "2"
|
||||
errno = "5"
|
||||
|
||||
The first write_aio request matches the set-state rule and transitions from
|
||||
state 1 to state 2. Once state 2 has been entered, the set-state rule no
|
||||
longer matches since it requires state 1. But the inject-error rule now
|
||||
matches the next write_aio request and injects EIO (5).
|
||||
|
||||
State transition rules support the following attributes:
|
||||
|
||||
event - which type of operation to match (e.g. read_aio, write_aio,
|
||||
flush_to_os, flush_to_disk). See the "Events" section for
|
||||
information on events.
|
||||
|
||||
state - (optional) the engine must be in this state number in order for this
|
||||
rule to match
|
||||
|
||||
new_state - transition to this state number
|
||||
|
||||
Suspend and resume
|
||||
------------------
|
||||
Exercising code paths in block drivers may require specific ordering amongst
|
||||
concurrent requests. The "breakpoint" feature allows requests to be halted on
|
||||
a blkdebug event and resumed later. This makes it possible to achieve
|
||||
deterministic ordering when multiple requests are in flight.
|
||||
|
||||
Breakpoints on blkdebug events are associated with a user-defined "tag" string.
|
||||
This tag serves as an identifier by which the request can be resumed at a later
|
||||
point.
|
||||
|
||||
See the qemu-io(1) break, resume, remove_break, and wait_break commands for
|
||||
details.
|
@@ -1,10 +1,5 @@
|
||||
= How to use the QAPI code generator =
|
||||
|
||||
* Note: as of this writing, QMP does not use QAPI. Eventually QMP
|
||||
commands will be converted to use QAPI internally. The following
|
||||
information describes QMP/QAPI as it will exist after the
|
||||
conversion.
|
||||
|
||||
QAPI is a native C API within QEMU which provides management-level
|
||||
functionality to internal/external users. For external
|
||||
users/processes, this interface is made available by a JSON-based
|
||||
@@ -19,7 +14,7 @@ marshaling/dispatch code for the guest agent server running in the
|
||||
guest.
|
||||
|
||||
This document will describe how the schemas, scripts, and resulting
|
||||
code is used.
|
||||
code are used.
|
||||
|
||||
|
||||
== QMP/Guest agent schema ==
|
||||
@@ -234,6 +229,7 @@ Resulting in this JSON object:
|
||||
"data": { "b": "test string" },
|
||||
"timestamp": { "seconds": 1267020223, "microseconds": 435656 } }
|
||||
|
||||
|
||||
== Code generation ==
|
||||
|
||||
Schemas are fed into 3 scripts to generate all the code/files that, paired
|
||||
@@ -256,6 +252,8 @@ command which takes that type as a parameter and returns the same type:
|
||||
'data': {'arg1': 'UserDefOne'},
|
||||
'returns': 'UserDefOne' }
|
||||
|
||||
{ 'event': 'MY_EVENT' }
|
||||
|
||||
=== scripts/qapi-types.py ===
|
||||
|
||||
Used to generate the C types defined by a schema. The following files are
|
||||
@@ -277,7 +275,7 @@ Example:
|
||||
$ cat qapi-generated/example-qapi-types.c
|
||||
[Uninteresting stuff omitted...]
|
||||
|
||||
void qapi_free_UserDefOneList(UserDefOneList * obj)
|
||||
void qapi_free_UserDefOneList(UserDefOneList *obj)
|
||||
{
|
||||
QapiDeallocVisitor *md;
|
||||
Visitor *v;
|
||||
@@ -292,7 +290,7 @@ Example:
|
||||
qapi_dealloc_visitor_cleanup(md);
|
||||
}
|
||||
|
||||
void qapi_free_UserDefOne(UserDefOne * obj)
|
||||
void qapi_free_UserDefOne(UserDefOne *obj)
|
||||
{
|
||||
QapiDeallocVisitor *md;
|
||||
Visitor *v;
|
||||
@@ -331,11 +329,11 @@ Example:
|
||||
struct UserDefOne
|
||||
{
|
||||
int64_t integer;
|
||||
char * string;
|
||||
char *string;
|
||||
};
|
||||
|
||||
void qapi_free_UserDefOneList(UserDefOneList * obj);
|
||||
void qapi_free_UserDefOne(UserDefOne * obj);
|
||||
void qapi_free_UserDefOneList(UserDefOneList *obj);
|
||||
void qapi_free_UserDefOne(UserDefOne *obj);
|
||||
|
||||
#endif
|
||||
|
||||
@@ -364,7 +362,7 @@ Example:
|
||||
$ cat qapi-generated/example-qapi-visit.c
|
||||
[Uninteresting stuff omitted...]
|
||||
|
||||
static void visit_type_UserDefOne_fields(Visitor *m, UserDefOne ** obj, Error **errp)
|
||||
static void visit_type_UserDefOne_fields(Visitor *m, UserDefOne **obj, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
visit_type_int(m, &(*obj)->integer, "integer", &err);
|
||||
@@ -380,7 +378,7 @@ Example:
|
||||
error_propagate(errp, err);
|
||||
}
|
||||
|
||||
void visit_type_UserDefOne(Visitor *m, UserDefOne ** obj, const char *name, Error **errp)
|
||||
void visit_type_UserDefOne(Visitor *m, UserDefOne **obj, const char *name, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
|
||||
@@ -394,7 +392,7 @@ Example:
|
||||
error_propagate(errp, err);
|
||||
}
|
||||
|
||||
void visit_type_UserDefOneList(Visitor *m, UserDefOneList ** obj, const char *name, Error **errp)
|
||||
void visit_type_UserDefOneList(Visitor *m, UserDefOneList **obj, const char *name, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
GenericList *i, **prev;
|
||||
@@ -427,8 +425,8 @@ Example:
|
||||
|
||||
[Visitors for builtin types omitted...]
|
||||
|
||||
void visit_type_UserDefOne(Visitor *m, UserDefOne ** obj, const char *name, Error **errp);
|
||||
void visit_type_UserDefOneList(Visitor *m, UserDefOneList ** obj, const char *name, Error **errp);
|
||||
void visit_type_UserDefOne(Visitor *m, UserDefOne **obj, const char *name, Error **errp);
|
||||
void visit_type_UserDefOneList(Visitor *m, UserDefOneList **obj, const char *name, Error **errp);
|
||||
|
||||
#endif
|
||||
|
||||
@@ -451,10 +449,12 @@ $(prefix)qmp-commands.h: Function prototypes for the QMP commands
|
||||
|
||||
Example:
|
||||
|
||||
$ python scripts/qapi-commands.py --output-dir="qapi-generated"
|
||||
--prefix="example-" --input-file=example-schema.json
|
||||
$ cat qapi-generated/example-qmp-marshal.c
|
||||
[Uninteresting stuff omitted...]
|
||||
|
||||
static void qmp_marshal_output_my_command(UserDefOne * ret_in, QObject **ret_out, Error **errp)
|
||||
static void qmp_marshal_output_my_command(UserDefOne *ret_in, QObject **ret_out, Error **errp)
|
||||
{
|
||||
Error *local_err = NULL;
|
||||
QmpOutputVisitor *mo = qmp_output_visitor_new();
|
||||
@@ -480,11 +480,11 @@ Example:
|
||||
static void qmp_marshal_input_my_command(QDict *args, QObject **ret, Error **errp)
|
||||
{
|
||||
Error *local_err = NULL;
|
||||
UserDefOne * retval = NULL;
|
||||
UserDefOne *retval = NULL;
|
||||
QmpInputVisitor *mi = qmp_input_visitor_new_strict(QOBJECT(args));
|
||||
QapiDeallocVisitor *md;
|
||||
Visitor *v;
|
||||
UserDefOne * arg1 = NULL;
|
||||
UserDefOne *arg1 = NULL;
|
||||
|
||||
v = qmp_input_get_visitor(mi);
|
||||
visit_type_UserDefOne(v, &arg1, "arg1", &local_err);
|
||||
@@ -525,6 +525,66 @@ Example:
|
||||
#include "qapi/qmp/qdict.h"
|
||||
#include "qapi/error.h"
|
||||
|
||||
UserDefOne * qmp_my_command(UserDefOne * arg1, Error **errp);
|
||||
UserDefOne *qmp_my_command(UserDefOne *arg1, Error **errp);
|
||||
|
||||
#endif
|
||||
|
||||
=== scripts/qapi-event.py ===
|
||||
|
||||
Used to generate the event-related C code defined by a schema. The
|
||||
following files are created:
|
||||
|
||||
$(prefix)qapi-event.h - Function prototypes for each event type, plus an
|
||||
enumeration of all event names
|
||||
$(prefix)qapi-event.c - Implementation of functions to send an event
|
||||
|
||||
Example:
|
||||
|
||||
$ python scripts/qapi-event.py --output-dir="qapi-generated"
|
||||
--prefix="example-" --input-file=example-schema.json
|
||||
$ cat qapi-generated/example-qapi-event.c
|
||||
[Uninteresting stuff omitted...]
|
||||
|
||||
void qapi_event_send_my_event(Error **errp)
|
||||
{
|
||||
QDict *qmp;
|
||||
Error *local_err = NULL;
|
||||
QMPEventFuncEmit emit;
|
||||
emit = qmp_event_get_func_emit();
|
||||
if (!emit) {
|
||||
return;
|
||||
}
|
||||
|
||||
qmp = qmp_event_build_dict("MY_EVENT");
|
||||
|
||||
emit(EXAMPLE_QAPI_EVENT_MY_EVENT, qmp, &local_err);
|
||||
|
||||
error_propagate(errp, local_err);
|
||||
QDECREF(qmp);
|
||||
}
|
||||
|
||||
const char *EXAMPLE_QAPIEvent_lookup[] = {
|
||||
"MY_EVENT",
|
||||
NULL,
|
||||
};
|
||||
$ cat qapi-generated/example-qapi-event.h
|
||||
[Uninteresting stuff omitted...]
|
||||
|
||||
#ifndef EXAMPLE_QAPI_EVENT_H
|
||||
#define EXAMPLE_QAPI_EVENT_H
|
||||
|
||||
#include "qapi/error.h"
|
||||
#include "qapi/qmp/qdict.h"
|
||||
#include "example-qapi-types.h"
|
||||
|
||||
|
||||
void qapi_event_send_my_event(Error **errp);
|
||||
|
||||
extern const char *EXAMPLE_QAPIEvent_lookup[];
|
||||
typedef enum EXAMPLE_QAPIEvent
|
||||
{
|
||||
EXAMPLE_QAPI_EVENT_MY_EVENT = 0,
|
||||
EXAMPLE_QAPI_EVENT_MAX = 1,
|
||||
} EXAMPLE_QAPIEvent;
|
||||
|
||||
#endif
|
||||
|
@@ -23,7 +23,7 @@ for debugging, profiling, and observing execution.
|
||||
|
||||
4. Pretty-print the binary trace file:
|
||||
|
||||
./scripts/simpletrace.py trace-events trace-*
|
||||
./scripts/simpletrace.py trace-events trace-* # Override * with QEMU <pid>
|
||||
|
||||
== Trace events ==
|
||||
|
||||
|
@@ -1778,6 +1778,8 @@ show qdev device model list
|
||||
show roms
|
||||
@item info tpm
|
||||
show the TPM device
|
||||
@item info memory-devices
|
||||
show the memory devices
|
||||
@end table
|
||||
ETEXI
|
||||
|
||||
|
38
hmp.c
38
hmp.c
@@ -1720,3 +1720,41 @@ void hmp_info_memdev(Monitor *mon, const QDict *qdict)
|
||||
|
||||
qapi_free_MemdevList(memdev_list);
|
||||
}
|
||||
|
||||
void hmp_info_memory_devices(Monitor *mon, const QDict *qdict)
|
||||
{
|
||||
Error *err = NULL;
|
||||
MemoryDeviceInfoList *info_list = qmp_query_memory_devices(&err);
|
||||
MemoryDeviceInfoList *info;
|
||||
MemoryDeviceInfo *value;
|
||||
PCDIMMDeviceInfo *di;
|
||||
|
||||
for (info = info_list; info; info = info->next) {
|
||||
value = info->value;
|
||||
|
||||
if (value) {
|
||||
switch (value->kind) {
|
||||
case MEMORY_DEVICE_INFO_KIND_DIMM:
|
||||
di = value->dimm;
|
||||
|
||||
monitor_printf(mon, "Memory device [%s]: \"%s\"\n",
|
||||
MemoryDeviceInfoKind_lookup[value->kind],
|
||||
di->id ? di->id : "");
|
||||
monitor_printf(mon, " addr: 0x%" PRIx64 "\n", di->addr);
|
||||
monitor_printf(mon, " slot: %" PRId64 "\n", di->slot);
|
||||
monitor_printf(mon, " node: %" PRId64 "\n", di->node);
|
||||
monitor_printf(mon, " size: %" PRIu64 "\n", di->size);
|
||||
monitor_printf(mon, " memdev: %s\n", di->memdev);
|
||||
monitor_printf(mon, " hotplugged: %s\n",
|
||||
di->hotplugged ? "true" : "false");
|
||||
monitor_printf(mon, " hotpluggable: %s\n",
|
||||
di->hotpluggable ? "true" : "false");
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
qapi_free_MemoryDeviceInfoList(info_list);
|
||||
}
|
||||
|
1
hmp.h
1
hmp.h
@@ -94,6 +94,7 @@ void hmp_cpu_add(Monitor *mon, const QDict *qdict);
|
||||
void hmp_object_add(Monitor *mon, const QDict *qdict);
|
||||
void hmp_object_del(Monitor *mon, const QDict *qdict);
|
||||
void hmp_info_memdev(Monitor *mon, const QDict *qdict);
|
||||
void hmp_info_memory_devices(Monitor *mon, const QDict *qdict);
|
||||
void object_add_completion(ReadLineState *rs, int nb_args, const char *str);
|
||||
void object_del_completion(ReadLineState *rs, int nb_args, const char *str);
|
||||
void device_add_completion(ReadLineState *rs, int nb_args, const char *str);
|
||||
|
@@ -815,6 +815,9 @@ static void serial_reset(void *opaque)
|
||||
s->thr_ipending = 0;
|
||||
s->last_break_enable = 0;
|
||||
qemu_irq_lower(s->irq);
|
||||
|
||||
serial_update_msl(s);
|
||||
s->msr &= ~UART_MSR_ANY_DELTA;
|
||||
}
|
||||
|
||||
void serial_realize_core(SerialState *s, Error **errp)
|
||||
@@ -833,6 +836,7 @@ void serial_realize_core(SerialState *s, Error **errp)
|
||||
serial_event, s);
|
||||
fifo8_create(&s->recv_fifo, UART_FIFO_LENGTH);
|
||||
fifo8_create(&s->xmit_fifo, UART_FIFO_LENGTH);
|
||||
serial_reset(s);
|
||||
}
|
||||
|
||||
void serial_exit_core(SerialState *s)
|
||||
@@ -944,7 +948,5 @@ SerialState *serial_mm_init(MemoryRegion *address_space,
|
||||
memory_region_init_io(&s->io, NULL, &serial_mm_ops[end], s,
|
||||
"serial", 8 << it_shift);
|
||||
memory_region_add_subregion(address_space, base, &s->io);
|
||||
|
||||
serial_update_msl(s);
|
||||
return s;
|
||||
}
|
||||
|
@@ -134,14 +134,6 @@ static const int blizzard_iformat_bpp[0x10] = {
|
||||
0, 0, 0, 0, 0, 0,
|
||||
};
|
||||
|
||||
static inline void blizzard_rgb2yuv(int r, int g, int b,
|
||||
int *y, int *u, int *v)
|
||||
{
|
||||
*y = 0x10 + ((0x838 * r + 0x1022 * g + 0x322 * b) >> 13);
|
||||
*u = 0x80 + ((0xe0e * b - 0x04c1 * r - 0x94e * g) >> 13);
|
||||
*v = 0x80 + ((0xe0e * r - 0x0bc7 * g - 0x247 * b) >> 13);
|
||||
}
|
||||
|
||||
static void blizzard_window(BlizzardState *s)
|
||||
{
|
||||
DisplaySurface *surface = qemu_console_surface(s->con);
|
||||
|
@@ -279,14 +279,6 @@ static inline void pxa2xx_dma_ber_set(PXA2xxLCDState *s, int ch)
|
||||
s->liidr = s->dma_ch[ch].id;
|
||||
}
|
||||
|
||||
/* Set Read Status interrupt high and poke associated registers */
|
||||
static inline void pxa2xx_dma_rdst_set(PXA2xxLCDState *s)
|
||||
{
|
||||
s->status[0] |= LCSR0_RDST;
|
||||
if (s->irqlevel && !(s->control[0] & LCCR0_RDSTM))
|
||||
s->status[0] |= LCSR0_SINT;
|
||||
}
|
||||
|
||||
/* Load new Frame Descriptors from DMA */
|
||||
static void pxa2xx_descriptor_load(PXA2xxLCDState *s)
|
||||
{
|
||||
|
@@ -132,6 +132,8 @@ static void qxl_reset_memslots(PCIQXLDevice *d);
|
||||
static void qxl_reset_surfaces(PCIQXLDevice *d);
|
||||
static void qxl_ring_set_dirty(PCIQXLDevice *qxl);
|
||||
|
||||
static void qxl_hw_update(void *opaque);
|
||||
|
||||
void qxl_set_guest_bug(PCIQXLDevice *qxl, const char *msg, ...)
|
||||
{
|
||||
trace_qxl_set_guest_bug(qxl->id);
|
||||
@@ -1076,6 +1078,10 @@ static const QXLInterface qxl_interface = {
|
||||
.client_monitors_config = interface_client_monitors_config,
|
||||
};
|
||||
|
||||
static const GraphicHwOps qxl_ops = {
|
||||
.gfx_update = qxl_hw_update,
|
||||
};
|
||||
|
||||
static void qxl_enter_vga_mode(PCIQXLDevice *d)
|
||||
{
|
||||
if (d->mode == QXL_MODE_VGA) {
|
||||
@@ -1085,6 +1091,7 @@ static void qxl_enter_vga_mode(PCIQXLDevice *d)
|
||||
#if SPICE_SERVER_VERSION >= 0x000c03 /* release 0.12.3 */
|
||||
spice_qxl_driver_unload(&d->ssd.qxl);
|
||||
#endif
|
||||
graphic_console_set_hwops(d->ssd.dcl.con, d->vga.hw_ops, &d->vga);
|
||||
qemu_spice_create_host_primary(&d->ssd);
|
||||
d->mode = QXL_MODE_VGA;
|
||||
vga_dirty_log_start(&d->vga);
|
||||
@@ -1097,6 +1104,7 @@ static void qxl_exit_vga_mode(PCIQXLDevice *d)
|
||||
return;
|
||||
}
|
||||
trace_qxl_exit_vga_mode(d->id);
|
||||
graphic_console_set_hwops(d->ssd.dcl.con, &qxl_ops, d);
|
||||
vga_dirty_log_stop(&d->vga);
|
||||
qxl_destroy_primary(d, QXL_SYNC);
|
||||
}
|
||||
@@ -1756,41 +1764,8 @@ static void qxl_send_events(PCIQXLDevice *d, uint32_t events)
|
||||
static void qxl_hw_update(void *opaque)
|
||||
{
|
||||
PCIQXLDevice *qxl = opaque;
|
||||
VGACommonState *vga = &qxl->vga;
|
||||
|
||||
switch (qxl->mode) {
|
||||
case QXL_MODE_VGA:
|
||||
vga->hw_ops->gfx_update(vga);
|
||||
break;
|
||||
case QXL_MODE_COMPAT:
|
||||
case QXL_MODE_NATIVE:
|
||||
qxl_render_update(qxl);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void qxl_hw_invalidate(void *opaque)
|
||||
{
|
||||
PCIQXLDevice *qxl = opaque;
|
||||
VGACommonState *vga = &qxl->vga;
|
||||
|
||||
if (qxl->mode == QXL_MODE_VGA) {
|
||||
vga->hw_ops->invalidate(vga);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
static void qxl_hw_text_update(void *opaque, console_ch_t *chardata)
|
||||
{
|
||||
PCIQXLDevice *qxl = opaque;
|
||||
VGACommonState *vga = &qxl->vga;
|
||||
|
||||
if (qxl->mode == QXL_MODE_VGA) {
|
||||
vga->hw_ops->text_update(vga, chardata);
|
||||
return;
|
||||
}
|
||||
qxl_render_update(qxl);
|
||||
}
|
||||
|
||||
static void qxl_dirty_surfaces(PCIQXLDevice *qxl)
|
||||
@@ -2049,12 +2024,6 @@ static int qxl_init_common(PCIQXLDevice *qxl)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const GraphicHwOps qxl_ops = {
|
||||
.invalidate = qxl_hw_invalidate,
|
||||
.gfx_update = qxl_hw_update,
|
||||
.text_update = qxl_hw_text_update,
|
||||
};
|
||||
|
||||
static int qxl_init_primary(PCIDevice *dev)
|
||||
{
|
||||
PCIQXLDevice *qxl = DO_UPCAST(PCIQXLDevice, pci, dev);
|
||||
|
679
hw/display/tcx.c
679
hw/display/tcx.c
@@ -33,10 +33,20 @@
|
||||
|
||||
#define MAXX 1024
|
||||
#define MAXY 768
|
||||
#define TCX_DAC_NREGS 16
|
||||
#define TCX_THC_NREGS_8 0x081c
|
||||
#define TCX_THC_NREGS_24 0x1000
|
||||
#define TCX_DAC_NREGS 16
|
||||
#define TCX_THC_NREGS 0x1000
|
||||
#define TCX_DHC_NREGS 0x4000
|
||||
#define TCX_TEC_NREGS 0x1000
|
||||
#define TCX_ALT_NREGS 0x8000
|
||||
#define TCX_STIP_NREGS 0x800000
|
||||
#define TCX_BLIT_NREGS 0x800000
|
||||
#define TCX_RSTIP_NREGS 0x800000
|
||||
#define TCX_RBLIT_NREGS 0x800000
|
||||
|
||||
#define TCX_THC_MISC 0x818
|
||||
#define TCX_THC_CURSXY 0x8fc
|
||||
#define TCX_THC_CURSMASK 0x900
|
||||
#define TCX_THC_CURSBITS 0x980
|
||||
|
||||
#define TYPE_TCX "SUNW,tcx"
|
||||
#define TCX(obj) OBJECT_CHECK(TCXState, (obj), TYPE_TCX)
|
||||
@@ -45,6 +55,7 @@ typedef struct TCXState {
|
||||
SysBusDevice parent_obj;
|
||||
|
||||
QemuConsole *con;
|
||||
qemu_irq irq;
|
||||
uint8_t *vram;
|
||||
uint32_t *vram24, *cplane;
|
||||
hwaddr prom_addr;
|
||||
@@ -52,17 +63,30 @@ typedef struct TCXState {
|
||||
MemoryRegion vram_mem;
|
||||
MemoryRegion vram_8bit;
|
||||
MemoryRegion vram_24bit;
|
||||
MemoryRegion stip;
|
||||
MemoryRegion blit;
|
||||
MemoryRegion vram_cplane;
|
||||
MemoryRegion dac;
|
||||
MemoryRegion rstip;
|
||||
MemoryRegion rblit;
|
||||
MemoryRegion tec;
|
||||
MemoryRegion dac;
|
||||
MemoryRegion thc;
|
||||
MemoryRegion dhc;
|
||||
MemoryRegion alt;
|
||||
MemoryRegion thc24;
|
||||
MemoryRegion thc8;
|
||||
|
||||
ram_addr_t vram24_offset, cplane_offset;
|
||||
uint32_t tmpblit;
|
||||
uint32_t vram_size;
|
||||
uint32_t palette[256];
|
||||
uint8_t r[256], g[256], b[256];
|
||||
uint32_t palette[260];
|
||||
uint8_t r[260], g[260], b[260];
|
||||
uint16_t width, height, depth;
|
||||
uint8_t dac_index, dac_state;
|
||||
uint32_t thcmisc;
|
||||
uint32_t cursmask[32];
|
||||
uint32_t cursbits[32];
|
||||
uint16_t cursx;
|
||||
uint16_t cursy;
|
||||
} TCXState;
|
||||
|
||||
static void tcx_set_dirty(TCXState *s)
|
||||
@@ -70,10 +94,36 @@ static void tcx_set_dirty(TCXState *s)
|
||||
memory_region_set_dirty(&s->vram_mem, 0, MAXX * MAXY);
|
||||
}
|
||||
|
||||
static void tcx24_set_dirty(TCXState *s)
|
||||
static inline int tcx24_check_dirty(TCXState *s, ram_addr_t page,
|
||||
ram_addr_t page24, ram_addr_t cpage)
|
||||
{
|
||||
memory_region_set_dirty(&s->vram_mem, s->vram24_offset, MAXX * MAXY * 4);
|
||||
memory_region_set_dirty(&s->vram_mem, s->cplane_offset, MAXX * MAXY * 4);
|
||||
int ret;
|
||||
|
||||
ret = memory_region_get_dirty(&s->vram_mem, page, TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
ret |= memory_region_get_dirty(&s->vram_mem, page24, TARGET_PAGE_SIZE * 4,
|
||||
DIRTY_MEMORY_VGA);
|
||||
ret |= memory_region_get_dirty(&s->vram_mem, cpage, TARGET_PAGE_SIZE * 4,
|
||||
DIRTY_MEMORY_VGA);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static inline void tcx24_reset_dirty(TCXState *ts, ram_addr_t page_min,
|
||||
ram_addr_t page_max, ram_addr_t page24,
|
||||
ram_addr_t cpage)
|
||||
{
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
page_min,
|
||||
(page_max - page_min) + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
page24 + page_min * 4,
|
||||
(page_max - page_min) * 4 + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
cpage + page_min * 4,
|
||||
(page_max - page_min) * 4 + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
}
|
||||
|
||||
static void update_palette_entries(TCXState *s, int start, int end)
|
||||
@@ -102,11 +152,7 @@ static void update_palette_entries(TCXState *s, int start, int end)
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (s->depth == 24) {
|
||||
tcx24_set_dirty(s);
|
||||
} else {
|
||||
tcx_set_dirty(s);
|
||||
}
|
||||
tcx_set_dirty(s);
|
||||
}
|
||||
|
||||
static void tcx_draw_line32(TCXState *s1, uint8_t *d,
|
||||
@@ -116,7 +162,7 @@ static void tcx_draw_line32(TCXState *s1, uint8_t *d,
|
||||
uint8_t val;
|
||||
uint32_t *p = (uint32_t *)d;
|
||||
|
||||
for(x = 0; x < width; x++) {
|
||||
for (x = 0; x < width; x++) {
|
||||
val = *s++;
|
||||
*p++ = s1->palette[val];
|
||||
}
|
||||
@@ -129,7 +175,7 @@ static void tcx_draw_line16(TCXState *s1, uint8_t *d,
|
||||
uint8_t val;
|
||||
uint16_t *p = (uint16_t *)d;
|
||||
|
||||
for(x = 0; x < width; x++) {
|
||||
for (x = 0; x < width; x++) {
|
||||
val = *s++;
|
||||
*p++ = s1->palette[val];
|
||||
}
|
||||
@@ -147,6 +193,83 @@ static void tcx_draw_line8(TCXState *s1, uint8_t *d,
|
||||
}
|
||||
}
|
||||
|
||||
static void tcx_draw_cursor32(TCXState *s1, uint8_t *d,
|
||||
int y, int width)
|
||||
{
|
||||
int x, len;
|
||||
uint32_t mask, bits;
|
||||
uint32_t *p = (uint32_t *)d;
|
||||
|
||||
y = y - s1->cursy;
|
||||
mask = s1->cursmask[y];
|
||||
bits = s1->cursbits[y];
|
||||
len = MIN(width - s1->cursx, 32);
|
||||
p = &p[s1->cursx];
|
||||
for (x = 0; x < len; x++) {
|
||||
if (mask & 0x80000000) {
|
||||
if (bits & 0x80000000) {
|
||||
*p = s1->palette[259];
|
||||
} else {
|
||||
*p = s1->palette[258];
|
||||
}
|
||||
}
|
||||
p++;
|
||||
mask <<= 1;
|
||||
bits <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
static void tcx_draw_cursor16(TCXState *s1, uint8_t *d,
|
||||
int y, int width)
|
||||
{
|
||||
int x, len;
|
||||
uint32_t mask, bits;
|
||||
uint16_t *p = (uint16_t *)d;
|
||||
|
||||
y = y - s1->cursy;
|
||||
mask = s1->cursmask[y];
|
||||
bits = s1->cursbits[y];
|
||||
len = MIN(width - s1->cursx, 32);
|
||||
p = &p[s1->cursx];
|
||||
for (x = 0; x < len; x++) {
|
||||
if (mask & 0x80000000) {
|
||||
if (bits & 0x80000000) {
|
||||
*p = s1->palette[259];
|
||||
} else {
|
||||
*p = s1->palette[258];
|
||||
}
|
||||
}
|
||||
p++;
|
||||
mask <<= 1;
|
||||
bits <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
static void tcx_draw_cursor8(TCXState *s1, uint8_t *d,
|
||||
int y, int width)
|
||||
{
|
||||
int x, len;
|
||||
uint32_t mask, bits;
|
||||
|
||||
y = y - s1->cursy;
|
||||
mask = s1->cursmask[y];
|
||||
bits = s1->cursbits[y];
|
||||
len = MIN(width - s1->cursx, 32);
|
||||
d = &d[s1->cursx];
|
||||
for (x = 0; x < len; x++) {
|
||||
if (mask & 0x80000000) {
|
||||
if (bits & 0x80000000) {
|
||||
*d = s1->palette[259];
|
||||
} else {
|
||||
*d = s1->palette[258];
|
||||
}
|
||||
}
|
||||
d++;
|
||||
mask <<= 1;
|
||||
bits <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
XXX Could be much more optimal:
|
||||
* detect if line/page/whole screen is in 24 bit mode
|
||||
@@ -162,11 +285,10 @@ static inline void tcx24_draw_line32(TCXState *s1, uint8_t *d,
|
||||
uint8_t val, *p8;
|
||||
uint32_t *p = (uint32_t *)d;
|
||||
uint32_t dval;
|
||||
|
||||
bgr = is_surface_bgr(surface);
|
||||
for(x = 0; x < width; x++, s++, s24++) {
|
||||
if ((be32_to_cpu(*cplane++) & 0xff000000) == 0x03000000) {
|
||||
// 24-bit direct, BGR order
|
||||
if (be32_to_cpu(*cplane) & 0x03000000) {
|
||||
/* 24-bit direct, BGR order */
|
||||
p8 = (uint8_t *)s24;
|
||||
p8++;
|
||||
b = *p8++;
|
||||
@@ -177,47 +299,18 @@ static inline void tcx24_draw_line32(TCXState *s1, uint8_t *d,
|
||||
else
|
||||
dval = rgb_to_pixel32(r, g, b);
|
||||
} else {
|
||||
/* 8-bit pseudocolor */
|
||||
val = *s;
|
||||
dval = s1->palette[val];
|
||||
}
|
||||
*p++ = dval;
|
||||
cplane++;
|
||||
}
|
||||
}
|
||||
|
||||
static inline int check_dirty(TCXState *s, ram_addr_t page, ram_addr_t page24,
|
||||
ram_addr_t cpage)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = memory_region_get_dirty(&s->vram_mem, page, TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
ret |= memory_region_get_dirty(&s->vram_mem, page24, TARGET_PAGE_SIZE * 4,
|
||||
DIRTY_MEMORY_VGA);
|
||||
ret |= memory_region_get_dirty(&s->vram_mem, cpage, TARGET_PAGE_SIZE * 4,
|
||||
DIRTY_MEMORY_VGA);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static inline void reset_dirty(TCXState *ts, ram_addr_t page_min,
|
||||
ram_addr_t page_max, ram_addr_t page24,
|
||||
ram_addr_t cpage)
|
||||
{
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
page_min,
|
||||
(page_max - page_min) + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
page24 + page_min * 4,
|
||||
(page_max - page_min) * 4 + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
memory_region_reset_dirty(&ts->vram_mem,
|
||||
cpage + page_min * 4,
|
||||
(page_max - page_min) * 4 + TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA);
|
||||
}
|
||||
|
||||
/* Fixed line length 1024 allows us to do nice tricks not possible on
|
||||
VGA... */
|
||||
|
||||
static void tcx_update_display(void *opaque)
|
||||
{
|
||||
TCXState *ts = opaque;
|
||||
@@ -226,6 +319,7 @@ static void tcx_update_display(void *opaque)
|
||||
int y, y_start, dd, ds;
|
||||
uint8_t *d, *s;
|
||||
void (*f)(TCXState *s1, uint8_t *dst, const uint8_t *src, int width);
|
||||
void (*fc)(TCXState *s1, uint8_t *dst, int y, int width);
|
||||
|
||||
if (surface_bits_per_pixel(surface) == 0) {
|
||||
return;
|
||||
@@ -243,20 +337,23 @@ static void tcx_update_display(void *opaque)
|
||||
switch (surface_bits_per_pixel(surface)) {
|
||||
case 32:
|
||||
f = tcx_draw_line32;
|
||||
fc = tcx_draw_cursor32;
|
||||
break;
|
||||
case 15:
|
||||
case 16:
|
||||
f = tcx_draw_line16;
|
||||
fc = tcx_draw_cursor16;
|
||||
break;
|
||||
default:
|
||||
case 8:
|
||||
f = tcx_draw_line8;
|
||||
fc = tcx_draw_cursor8;
|
||||
break;
|
||||
case 0:
|
||||
return;
|
||||
}
|
||||
|
||||
for(y = 0; y < ts->height; y += 4, page += TARGET_PAGE_SIZE) {
|
||||
for (y = 0; y < ts->height; page += TARGET_PAGE_SIZE) {
|
||||
if (memory_region_get_dirty(&ts->vram_mem, page, TARGET_PAGE_SIZE,
|
||||
DIRTY_MEMORY_VGA)) {
|
||||
if (y_start < 0)
|
||||
@@ -265,18 +362,38 @@ static void tcx_update_display(void *opaque)
|
||||
page_min = page;
|
||||
if (page > page_max)
|
||||
page_max = page;
|
||||
|
||||
f(ts, d, s, ts->width);
|
||||
if (y >= ts->cursy && y < ts->cursy + 32 && ts->cursx < ts->width) {
|
||||
fc(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
y++;
|
||||
|
||||
f(ts, d, s, ts->width);
|
||||
if (y >= ts->cursy && y < ts->cursy + 32 && ts->cursx < ts->width) {
|
||||
fc(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
y++;
|
||||
|
||||
f(ts, d, s, ts->width);
|
||||
if (y >= ts->cursy && y < ts->cursy + 32 && ts->cursx < ts->width) {
|
||||
fc(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
y++;
|
||||
|
||||
f(ts, d, s, ts->width);
|
||||
if (y >= ts->cursy && y < ts->cursy + 32 && ts->cursx < ts->width) {
|
||||
fc(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
y++;
|
||||
} else {
|
||||
if (y_start >= 0) {
|
||||
/* flush to display */
|
||||
@@ -286,6 +403,7 @@ static void tcx_update_display(void *opaque)
|
||||
}
|
||||
d += dd * 4;
|
||||
s += ds * 4;
|
||||
y += 4;
|
||||
}
|
||||
}
|
||||
if (y_start >= 0) {
|
||||
@@ -328,9 +446,9 @@ static void tcx24_update_display(void *opaque)
|
||||
dd = surface_stride(surface);
|
||||
ds = 1024;
|
||||
|
||||
for(y = 0; y < ts->height; y += 4, page += TARGET_PAGE_SIZE,
|
||||
for (y = 0; y < ts->height; page += TARGET_PAGE_SIZE,
|
||||
page24 += TARGET_PAGE_SIZE, cpage += TARGET_PAGE_SIZE) {
|
||||
if (check_dirty(ts, page, page24, cpage)) {
|
||||
if (tcx24_check_dirty(ts, page, page24, cpage)) {
|
||||
if (y_start < 0)
|
||||
y_start = y;
|
||||
if (page < page_min)
|
||||
@@ -338,25 +456,41 @@ static void tcx24_update_display(void *opaque)
|
||||
if (page > page_max)
|
||||
page_max = page;
|
||||
tcx24_draw_line32(ts, d, s, ts->width, cptr, s24);
|
||||
if (y >= ts->cursy && y < ts->cursy+32 && ts->cursx < ts->width) {
|
||||
tcx_draw_cursor32(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
cptr += ds;
|
||||
s24 += ds;
|
||||
y++;
|
||||
tcx24_draw_line32(ts, d, s, ts->width, cptr, s24);
|
||||
if (y >= ts->cursy && y < ts->cursy+32 && ts->cursx < ts->width) {
|
||||
tcx_draw_cursor32(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
cptr += ds;
|
||||
s24 += ds;
|
||||
y++;
|
||||
tcx24_draw_line32(ts, d, s, ts->width, cptr, s24);
|
||||
if (y >= ts->cursy && y < ts->cursy+32 && ts->cursx < ts->width) {
|
||||
tcx_draw_cursor32(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
cptr += ds;
|
||||
s24 += ds;
|
||||
y++;
|
||||
tcx24_draw_line32(ts, d, s, ts->width, cptr, s24);
|
||||
if (y >= ts->cursy && y < ts->cursy+32 && ts->cursx < ts->width) {
|
||||
tcx_draw_cursor32(ts, d, y, ts->width);
|
||||
}
|
||||
d += dd;
|
||||
s += ds;
|
||||
cptr += ds;
|
||||
s24 += ds;
|
||||
y++;
|
||||
} else {
|
||||
if (y_start >= 0) {
|
||||
/* flush to display */
|
||||
@@ -368,6 +502,7 @@ static void tcx24_update_display(void *opaque)
|
||||
s += ds * 4;
|
||||
cptr += ds * 4;
|
||||
s24 += ds * 4;
|
||||
y += 4;
|
||||
}
|
||||
}
|
||||
if (y_start >= 0) {
|
||||
@@ -377,7 +512,7 @@ static void tcx24_update_display(void *opaque)
|
||||
}
|
||||
/* reset modified pages */
|
||||
if (page_max >= page_min) {
|
||||
reset_dirty(ts, page_min, page_max, page24, cpage);
|
||||
tcx24_reset_dirty(ts, page_min, page_max, page24, cpage);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -394,7 +529,6 @@ static void tcx24_invalidate_display(void *opaque)
|
||||
TCXState *s = opaque;
|
||||
|
||||
tcx_set_dirty(s);
|
||||
tcx24_set_dirty(s);
|
||||
qemu_console_resize(s->con, s->width, s->height);
|
||||
}
|
||||
|
||||
@@ -403,12 +537,7 @@ static int vmstate_tcx_post_load(void *opaque, int version_id)
|
||||
TCXState *s = opaque;
|
||||
|
||||
update_palette_entries(s, 0, 256);
|
||||
if (s->depth == 24) {
|
||||
tcx24_set_dirty(s);
|
||||
} else {
|
||||
tcx_set_dirty(s);
|
||||
}
|
||||
|
||||
tcx_set_dirty(s);
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -435,56 +564,87 @@ static void tcx_reset(DeviceState *d)
|
||||
TCXState *s = TCX(d);
|
||||
|
||||
/* Initialize palette */
|
||||
memset(s->r, 0, 256);
|
||||
memset(s->g, 0, 256);
|
||||
memset(s->b, 0, 256);
|
||||
memset(s->r, 0, 260);
|
||||
memset(s->g, 0, 260);
|
||||
memset(s->b, 0, 260);
|
||||
s->r[255] = s->g[255] = s->b[255] = 255;
|
||||
update_palette_entries(s, 0, 256);
|
||||
s->r[256] = s->g[256] = s->b[256] = 255;
|
||||
s->r[258] = s->g[258] = s->b[258] = 255;
|
||||
update_palette_entries(s, 0, 260);
|
||||
memset(s->vram, 0, MAXX*MAXY);
|
||||
memory_region_reset_dirty(&s->vram_mem, 0, MAXX * MAXY * (1 + 4 + 4),
|
||||
DIRTY_MEMORY_VGA);
|
||||
s->dac_index = 0;
|
||||
s->dac_state = 0;
|
||||
s->cursx = 0xf000; /* Put cursor off screen */
|
||||
s->cursy = 0xf000;
|
||||
}
|
||||
|
||||
static uint64_t tcx_dac_readl(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
return 0;
|
||||
TCXState *s = opaque;
|
||||
uint32_t val = 0;
|
||||
|
||||
switch (s->dac_state) {
|
||||
case 0:
|
||||
val = s->r[s->dac_index] << 24;
|
||||
s->dac_state++;
|
||||
break;
|
||||
case 1:
|
||||
val = s->g[s->dac_index] << 24;
|
||||
s->dac_state++;
|
||||
break;
|
||||
case 2:
|
||||
val = s->b[s->dac_index] << 24;
|
||||
s->dac_index = (s->dac_index + 1) & 0xff; /* Index autoincrement */
|
||||
default:
|
||||
s->dac_state = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
static void tcx_dac_writel(void *opaque, hwaddr addr, uint64_t val,
|
||||
unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
unsigned index;
|
||||
|
||||
switch (addr) {
|
||||
case 0:
|
||||
case 0: /* Address */
|
||||
s->dac_index = val >> 24;
|
||||
s->dac_state = 0;
|
||||
break;
|
||||
case 4:
|
||||
case 4: /* Pixel colours */
|
||||
case 12: /* Overlay (cursor) colours */
|
||||
if (addr & 8) {
|
||||
index = (s->dac_index & 3) + 256;
|
||||
} else {
|
||||
index = s->dac_index;
|
||||
}
|
||||
switch (s->dac_state) {
|
||||
case 0:
|
||||
s->r[s->dac_index] = val >> 24;
|
||||
update_palette_entries(s, s->dac_index, s->dac_index + 1);
|
||||
s->r[index] = val >> 24;
|
||||
update_palette_entries(s, index, index + 1);
|
||||
s->dac_state++;
|
||||
break;
|
||||
case 1:
|
||||
s->g[s->dac_index] = val >> 24;
|
||||
update_palette_entries(s, s->dac_index, s->dac_index + 1);
|
||||
s->g[index] = val >> 24;
|
||||
update_palette_entries(s, index, index + 1);
|
||||
s->dac_state++;
|
||||
break;
|
||||
case 2:
|
||||
s->b[s->dac_index] = val >> 24;
|
||||
update_palette_entries(s, s->dac_index, s->dac_index + 1);
|
||||
s->dac_index = (s->dac_index + 1) & 255; // Index autoincrement
|
||||
s->b[index] = val >> 24;
|
||||
update_palette_entries(s, index, index + 1);
|
||||
s->dac_index = (s->dac_index + 1) & 0xff; /* Index autoincrement */
|
||||
default:
|
||||
s->dac_state = 0;
|
||||
break;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
default: /* Control registers */
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -499,20 +659,266 @@ static const MemoryRegionOps tcx_dac_ops = {
|
||||
},
|
||||
};
|
||||
|
||||
static uint64_t dummy_readl(void *opaque, hwaddr addr,
|
||||
static uint64_t tcx_stip_readl(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void tcx_stip_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
int i;
|
||||
uint32_t col;
|
||||
|
||||
if (!(addr & 4)) {
|
||||
s->tmpblit = val;
|
||||
} else {
|
||||
addr = (addr >> 3) & 0xfffff;
|
||||
col = cpu_to_be32(s->tmpblit);
|
||||
if (s->depth == 24) {
|
||||
for (i = 0; i < 32; i++) {
|
||||
if (val & 0x80000000) {
|
||||
s->vram[addr + i] = s->tmpblit;
|
||||
s->vram24[addr + i] = col;
|
||||
}
|
||||
val <<= 1;
|
||||
}
|
||||
} else {
|
||||
for (i = 0; i < 32; i++) {
|
||||
if (val & 0x80000000) {
|
||||
s->vram[addr + i] = s->tmpblit;
|
||||
}
|
||||
val <<= 1;
|
||||
}
|
||||
}
|
||||
memory_region_set_dirty(&s->vram_mem, addr, 32);
|
||||
}
|
||||
}
|
||||
|
||||
static void tcx_rstip_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
int i;
|
||||
uint32_t col;
|
||||
|
||||
if (!(addr & 4)) {
|
||||
s->tmpblit = val;
|
||||
} else {
|
||||
addr = (addr >> 3) & 0xfffff;
|
||||
col = cpu_to_be32(s->tmpblit);
|
||||
if (s->depth == 24) {
|
||||
for (i = 0; i < 32; i++) {
|
||||
if (val & 0x80000000) {
|
||||
s->vram[addr + i] = s->tmpblit;
|
||||
s->vram24[addr + i] = col;
|
||||
s->cplane[addr + i] = col;
|
||||
}
|
||||
val <<= 1;
|
||||
}
|
||||
} else {
|
||||
for (i = 0; i < 32; i++) {
|
||||
if (val & 0x80000000) {
|
||||
s->vram[addr + i] = s->tmpblit;
|
||||
}
|
||||
val <<= 1;
|
||||
}
|
||||
}
|
||||
memory_region_set_dirty(&s->vram_mem, addr, 32);
|
||||
}
|
||||
}
|
||||
|
||||
static const MemoryRegionOps tcx_stip_ops = {
|
||||
.read = tcx_stip_readl,
|
||||
.write = tcx_stip_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
.max_access_size = 4,
|
||||
},
|
||||
};
|
||||
|
||||
static const MemoryRegionOps tcx_rstip_ops = {
|
||||
.read = tcx_stip_readl,
|
||||
.write = tcx_rstip_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
.max_access_size = 4,
|
||||
},
|
||||
};
|
||||
|
||||
static uint64_t tcx_blit_readl(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void tcx_blit_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
uint32_t adsr, len;
|
||||
int i;
|
||||
|
||||
if (!(addr & 4)) {
|
||||
s->tmpblit = val;
|
||||
} else {
|
||||
addr = (addr >> 3) & 0xfffff;
|
||||
adsr = val & 0xffffff;
|
||||
len = ((val >> 24) & 0x1f) + 1;
|
||||
if (adsr == 0xffffff) {
|
||||
memset(&s->vram[addr], s->tmpblit, len);
|
||||
if (s->depth == 24) {
|
||||
val = s->tmpblit & 0xffffff;
|
||||
val = cpu_to_be32(val);
|
||||
for (i = 0; i < len; i++) {
|
||||
s->vram24[addr + i] = val;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
memcpy(&s->vram[addr], &s->vram[adsr], len);
|
||||
if (s->depth == 24) {
|
||||
memcpy(&s->vram24[addr], &s->vram24[adsr], len * 4);
|
||||
}
|
||||
}
|
||||
memory_region_set_dirty(&s->vram_mem, addr, len);
|
||||
}
|
||||
}
|
||||
|
||||
static void tcx_rblit_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
uint32_t adsr, len;
|
||||
int i;
|
||||
|
||||
if (!(addr & 4)) {
|
||||
s->tmpblit = val;
|
||||
} else {
|
||||
addr = (addr >> 3) & 0xfffff;
|
||||
adsr = val & 0xffffff;
|
||||
len = ((val >> 24) & 0x1f) + 1;
|
||||
if (adsr == 0xffffff) {
|
||||
memset(&s->vram[addr], s->tmpblit, len);
|
||||
if (s->depth == 24) {
|
||||
val = s->tmpblit & 0xffffff;
|
||||
val = cpu_to_be32(val);
|
||||
for (i = 0; i < len; i++) {
|
||||
s->vram24[addr + i] = val;
|
||||
s->cplane[addr + i] = val;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
memcpy(&s->vram[addr], &s->vram[adsr], len);
|
||||
if (s->depth == 24) {
|
||||
memcpy(&s->vram24[addr], &s->vram24[adsr], len * 4);
|
||||
memcpy(&s->cplane[addr], &s->cplane[adsr], len * 4);
|
||||
}
|
||||
}
|
||||
memory_region_set_dirty(&s->vram_mem, addr, len);
|
||||
}
|
||||
}
|
||||
|
||||
static const MemoryRegionOps tcx_blit_ops = {
|
||||
.read = tcx_blit_readl,
|
||||
.write = tcx_blit_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
.max_access_size = 4,
|
||||
},
|
||||
};
|
||||
|
||||
static const MemoryRegionOps tcx_rblit_ops = {
|
||||
.read = tcx_blit_readl,
|
||||
.write = tcx_rblit_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
.max_access_size = 4,
|
||||
},
|
||||
};
|
||||
|
||||
static void tcx_invalidate_cursor_position(TCXState *s)
|
||||
{
|
||||
int ymin, ymax, start, end;
|
||||
|
||||
/* invalidate only near the cursor */
|
||||
ymin = s->cursy;
|
||||
if (ymin >= s->height) {
|
||||
return;
|
||||
}
|
||||
ymax = MIN(s->height, ymin + 32);
|
||||
start = ymin * 1024;
|
||||
end = ymax * 1024;
|
||||
|
||||
memory_region_set_dirty(&s->vram_mem, start, end-start);
|
||||
}
|
||||
|
||||
static uint64_t tcx_thc_readl(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
uint64_t val;
|
||||
|
||||
if (addr == TCX_THC_MISC) {
|
||||
val = s->thcmisc | 0x02000000;
|
||||
} else {
|
||||
val = 0;
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
static void tcx_thc_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
TCXState *s = opaque;
|
||||
|
||||
if (addr == TCX_THC_CURSXY) {
|
||||
tcx_invalidate_cursor_position(s);
|
||||
s->cursx = val >> 16;
|
||||
s->cursy = val;
|
||||
tcx_invalidate_cursor_position(s);
|
||||
} else if (addr >= TCX_THC_CURSMASK && addr < TCX_THC_CURSMASK + 128) {
|
||||
s->cursmask[(addr - TCX_THC_CURSMASK) >> 2] = val;
|
||||
tcx_invalidate_cursor_position(s);
|
||||
} else if (addr >= TCX_THC_CURSBITS && addr < TCX_THC_CURSBITS + 128) {
|
||||
s->cursbits[(addr - TCX_THC_CURSBITS) >> 2] = val;
|
||||
tcx_invalidate_cursor_position(s);
|
||||
} else if (addr == TCX_THC_MISC) {
|
||||
s->thcmisc = val;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
static const MemoryRegionOps tcx_thc_ops = {
|
||||
.read = tcx_thc_readl,
|
||||
.write = tcx_thc_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
.max_access_size = 4,
|
||||
},
|
||||
};
|
||||
|
||||
static uint64_t tcx_dummy_readl(void *opaque, hwaddr addr,
|
||||
unsigned size)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void dummy_writel(void *opaque, hwaddr addr,
|
||||
static void tcx_dummy_writel(void *opaque, hwaddr addr,
|
||||
uint64_t val, unsigned size)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
static const MemoryRegionOps dummy_ops = {
|
||||
.read = dummy_readl,
|
||||
.write = dummy_writel,
|
||||
static const MemoryRegionOps tcx_dummy_ops = {
|
||||
.read = tcx_dummy_readl,
|
||||
.write = tcx_dummy_writel,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.valid = {
|
||||
.min_access_size = 4,
|
||||
@@ -540,20 +946,50 @@ static void tcx_initfn(Object *obj)
|
||||
memory_region_set_readonly(&s->rom, true);
|
||||
sysbus_init_mmio(sbd, &s->rom);
|
||||
|
||||
/* DAC */
|
||||
/* 2/STIP : Stippler */
|
||||
memory_region_init_io(&s->stip, OBJECT(s), &tcx_stip_ops, s, "tcx.stip",
|
||||
TCX_STIP_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->stip);
|
||||
|
||||
/* 3/BLIT : Blitter */
|
||||
memory_region_init_io(&s->blit, OBJECT(s), &tcx_blit_ops, s, "tcx.blit",
|
||||
TCX_BLIT_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->blit);
|
||||
|
||||
/* 5/RSTIP : Raw Stippler */
|
||||
memory_region_init_io(&s->rstip, OBJECT(s), &tcx_rstip_ops, s, "tcx.rstip",
|
||||
TCX_RSTIP_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->rstip);
|
||||
|
||||
/* 6/RBLIT : Raw Blitter */
|
||||
memory_region_init_io(&s->rblit, OBJECT(s), &tcx_rblit_ops, s, "tcx.rblit",
|
||||
TCX_RBLIT_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->rblit);
|
||||
|
||||
/* 7/TEC : ??? */
|
||||
memory_region_init_io(&s->tec, OBJECT(s), &tcx_dummy_ops, s,
|
||||
"tcx.tec", TCX_TEC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->tec);
|
||||
|
||||
/* 8/CMAP : DAC */
|
||||
memory_region_init_io(&s->dac, OBJECT(s), &tcx_dac_ops, s,
|
||||
"tcx.dac", TCX_DAC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->dac);
|
||||
|
||||
/* TEC (dummy) */
|
||||
memory_region_init_io(&s->tec, OBJECT(s), &dummy_ops, s,
|
||||
"tcx.tec", TCX_TEC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->tec);
|
||||
/* 9/THC : Cursor */
|
||||
memory_region_init_io(&s->thc, OBJECT(s), &tcx_thc_ops, s, "tcx.thc",
|
||||
TCX_THC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->thc);
|
||||
|
||||
/* THC: NetBSD writes here even with 8-bit display: dummy */
|
||||
memory_region_init_io(&s->thc24, OBJECT(s), &dummy_ops, s, "tcx.thc24",
|
||||
TCX_THC_NREGS_24);
|
||||
sysbus_init_mmio(sbd, &s->thc24);
|
||||
/* 11/DHC : ??? */
|
||||
memory_region_init_io(&s->dhc, OBJECT(s), &tcx_dummy_ops, s, "tcx.dhc",
|
||||
TCX_DHC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->dhc);
|
||||
|
||||
/* 12/ALT : ??? */
|
||||
memory_region_init_io(&s->alt, OBJECT(s), &tcx_dummy_ops, s, "tcx.alt",
|
||||
TCX_ALT_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->alt);
|
||||
|
||||
return;
|
||||
}
|
||||
@@ -572,7 +1008,7 @@ static void tcx_realizefn(DeviceState *dev, Error **errp)
|
||||
vmstate_register_ram_global(&s->vram_mem);
|
||||
vram_base = memory_region_get_ram_ptr(&s->vram_mem);
|
||||
|
||||
/* FCode ROM */
|
||||
/* 10/ROM : FCode ROM */
|
||||
vmstate_register_ram_global(&s->rom);
|
||||
fcode_filename = qemu_find_file(QEMU_FILE_TYPE_BIOS, TCX_ROM_FILE);
|
||||
if (fcode_filename) {
|
||||
@@ -583,7 +1019,7 @@ static void tcx_realizefn(DeviceState *dev, Error **errp)
|
||||
}
|
||||
}
|
||||
|
||||
/* 8-bit plane */
|
||||
/* 0/DFB8 : 8-bit plane */
|
||||
s->vram = vram_base;
|
||||
size = s->vram_size;
|
||||
memory_region_init_alias(&s->vram_8bit, OBJECT(s), "tcx.vram.8bit",
|
||||
@@ -592,35 +1028,40 @@ static void tcx_realizefn(DeviceState *dev, Error **errp)
|
||||
vram_offset += size;
|
||||
vram_base += size;
|
||||
|
||||
if (s->depth == 24) {
|
||||
/* 24-bit plane */
|
||||
size = s->vram_size * 4;
|
||||
s->vram24 = (uint32_t *)vram_base;
|
||||
s->vram24_offset = vram_offset;
|
||||
memory_region_init_alias(&s->vram_24bit, OBJECT(s), "tcx.vram.24bit",
|
||||
&s->vram_mem, vram_offset, size);
|
||||
sysbus_init_mmio(sbd, &s->vram_24bit);
|
||||
vram_offset += size;
|
||||
vram_base += size;
|
||||
/* 1/DFB24 : 24bit plane */
|
||||
size = s->vram_size * 4;
|
||||
s->vram24 = (uint32_t *)vram_base;
|
||||
s->vram24_offset = vram_offset;
|
||||
memory_region_init_alias(&s->vram_24bit, OBJECT(s), "tcx.vram.24bit",
|
||||
&s->vram_mem, vram_offset, size);
|
||||
sysbus_init_mmio(sbd, &s->vram_24bit);
|
||||
vram_offset += size;
|
||||
vram_base += size;
|
||||
|
||||
/* Control plane */
|
||||
size = s->vram_size * 4;
|
||||
s->cplane = (uint32_t *)vram_base;
|
||||
s->cplane_offset = vram_offset;
|
||||
memory_region_init_alias(&s->vram_cplane, OBJECT(s), "tcx.vram.cplane",
|
||||
&s->vram_mem, vram_offset, size);
|
||||
sysbus_init_mmio(sbd, &s->vram_cplane);
|
||||
/* 4/RDFB32 : Raw Framebuffer */
|
||||
size = s->vram_size * 4;
|
||||
s->cplane = (uint32_t *)vram_base;
|
||||
s->cplane_offset = vram_offset;
|
||||
memory_region_init_alias(&s->vram_cplane, OBJECT(s), "tcx.vram.cplane",
|
||||
&s->vram_mem, vram_offset, size);
|
||||
sysbus_init_mmio(sbd, &s->vram_cplane);
|
||||
|
||||
s->con = graphic_console_init(DEVICE(dev), 0, &tcx24_ops, s);
|
||||
} else {
|
||||
/* THC 8 bit (dummy) */
|
||||
memory_region_init_io(&s->thc8, OBJECT(s), &dummy_ops, s, "tcx.thc8",
|
||||
TCX_THC_NREGS_8);
|
||||
sysbus_init_mmio(sbd, &s->thc8);
|
||||
|
||||
s->con = graphic_console_init(DEVICE(dev), 0, &tcx_ops, s);
|
||||
/* 9/THC24bits : NetBSD writes here even with 8-bit display: dummy */
|
||||
if (s->depth == 8) {
|
||||
memory_region_init_io(&s->thc24, OBJECT(s), &tcx_dummy_ops, s,
|
||||
"tcx.thc24", TCX_THC_NREGS);
|
||||
sysbus_init_mmio(sbd, &s->thc24);
|
||||
}
|
||||
|
||||
sysbus_init_irq(sbd, &s->irq);
|
||||
|
||||
if (s->depth == 8) {
|
||||
s->con = graphic_console_init(DEVICE(dev), 0, &tcx_ops, s);
|
||||
} else {
|
||||
s->con = graphic_console_init(DEVICE(dev), 0, &tcx24_ops, s);
|
||||
}
|
||||
s->thcmisc = 0;
|
||||
|
||||
qemu_console_resize(s->con, s->width, s->height);
|
||||
}
|
||||
|
||||
|
@@ -24,6 +24,7 @@
|
||||
#include "hw/hw.h"
|
||||
#include "hw/isa/isa.h"
|
||||
#include "qemu/main-loop.h"
|
||||
#include "trace.h"
|
||||
|
||||
/* #define DEBUG_DMA */
|
||||
|
||||
@@ -473,8 +474,7 @@ static void dma_reset(void *opaque)
|
||||
|
||||
static int dma_phony_handler (void *opaque, int nchan, int dma_pos, int dma_len)
|
||||
{
|
||||
dolog ("unregistered DMA channel used nchan=%d dma_pos=%d dma_len=%d\n",
|
||||
nchan, dma_pos, dma_len);
|
||||
trace_i8257_unregistered_dma(nchan, dma_pos, dma_len);
|
||||
return dma_pos;
|
||||
}
|
||||
|
||||
|
@@ -14,8 +14,10 @@
|
||||
*/
|
||||
|
||||
#include "qemu-common.h"
|
||||
#include "qemu/host-utils.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
#include "sysemu/kvm.h"
|
||||
#include "sysemu/cpus.h"
|
||||
#include "hw/sysbus.h"
|
||||
#include "hw/kvm/clock.h"
|
||||
|
||||
@@ -34,6 +36,48 @@ typedef struct KVMClockState {
|
||||
bool clock_valid;
|
||||
} KVMClockState;
|
||||
|
||||
struct pvclock_vcpu_time_info {
|
||||
uint32_t version;
|
||||
uint32_t pad0;
|
||||
uint64_t tsc_timestamp;
|
||||
uint64_t system_time;
|
||||
uint32_t tsc_to_system_mul;
|
||||
int8_t tsc_shift;
|
||||
uint8_t flags;
|
||||
uint8_t pad[2];
|
||||
} __attribute__((__packed__)); /* 32 bytes */
|
||||
|
||||
static uint64_t kvmclock_current_nsec(KVMClockState *s)
|
||||
{
|
||||
CPUState *cpu = first_cpu;
|
||||
CPUX86State *env = cpu->env_ptr;
|
||||
hwaddr kvmclock_struct_pa = env->system_time_msr & ~1ULL;
|
||||
uint64_t migration_tsc = env->tsc;
|
||||
struct pvclock_vcpu_time_info time;
|
||||
uint64_t delta;
|
||||
uint64_t nsec_lo;
|
||||
uint64_t nsec_hi;
|
||||
uint64_t nsec;
|
||||
|
||||
if (!(env->system_time_msr & 1ULL)) {
|
||||
/* KVM clock not active */
|
||||
return 0;
|
||||
}
|
||||
|
||||
cpu_physical_memory_read(kvmclock_struct_pa, &time, sizeof(time));
|
||||
|
||||
assert(time.tsc_timestamp <= migration_tsc);
|
||||
delta = migration_tsc - time.tsc_timestamp;
|
||||
if (time.tsc_shift < 0) {
|
||||
delta >>= -time.tsc_shift;
|
||||
} else {
|
||||
delta <<= time.tsc_shift;
|
||||
}
|
||||
|
||||
mulu64(&nsec_lo, &nsec_hi, delta, time.tsc_to_system_mul);
|
||||
nsec = (nsec_lo >> 32) | (nsec_hi << 32);
|
||||
return nsec + time.system_time;
|
||||
}
|
||||
|
||||
static void kvmclock_vm_state_change(void *opaque, int running,
|
||||
RunState state)
|
||||
@@ -45,9 +89,15 @@ static void kvmclock_vm_state_change(void *opaque, int running,
|
||||
|
||||
if (running) {
|
||||
struct kvm_clock_data data;
|
||||
uint64_t time_at_migration = kvmclock_current_nsec(s);
|
||||
|
||||
s->clock_valid = false;
|
||||
|
||||
/* We can't rely on the migrated clock value, just discard it */
|
||||
if (time_at_migration) {
|
||||
s->clock = time_at_migration;
|
||||
}
|
||||
|
||||
data.clock = s->clock;
|
||||
data.flags = 0;
|
||||
ret = kvm_vm_ioctl(kvm_state, KVM_SET_CLOCK, &data);
|
||||
@@ -75,6 +125,9 @@ static void kvmclock_vm_state_change(void *opaque, int running,
|
||||
if (s->clock_valid) {
|
||||
return;
|
||||
}
|
||||
|
||||
cpu_synchronize_all_states();
|
||||
cpu_clean_all_dirty();
|
||||
ret = kvm_vm_ioctl(kvm_state, KVM_GET_CLOCK, &data);
|
||||
if (ret < 0) {
|
||||
fprintf(stderr, "KVM_GET_CLOCK failed: %s\n", strerror(ret));
|
||||
|
@@ -239,6 +239,7 @@ static void kvm_pit_vm_state_change(void *opaque, int running,
|
||||
|
||||
if (running) {
|
||||
kvm_pit_update_clock_offset(s);
|
||||
kvm_pit_put(PIT_COMMON(s));
|
||||
s->vm_stopped = false;
|
||||
} else {
|
||||
kvm_pit_update_clock_offset(s);
|
||||
@@ -314,8 +315,6 @@ static void kvm_pit_class_init(ObjectClass *klass, void *data)
|
||||
dc->realize = kvm_pit_realizefn;
|
||||
k->set_channel_gate = kvm_pit_set_gate;
|
||||
k->get_channel_info = kvm_pit_get_channel_info;
|
||||
k->pre_save = kvm_pit_get;
|
||||
k->post_load = kvm_pit_put;
|
||||
dc->reset = kvm_pit_reset;
|
||||
dc->props = kvm_pit_properties;
|
||||
}
|
||||
|
@@ -732,7 +732,11 @@ static void do_vapic_enable(void *data)
|
||||
VAPICROMState *s = data;
|
||||
X86CPU *cpu = X86_CPU(first_cpu);
|
||||
|
||||
vapic_enable(s, cpu);
|
||||
static const uint8_t enabled = 1;
|
||||
cpu_physical_memory_write(s->vapic_paddr + offsetof(VAPICState, enabled),
|
||||
&enabled, sizeof(enabled));
|
||||
apic_enable_vapic(cpu->apic_state, s->vapic_paddr);
|
||||
s->state = VAPIC_ACTIVE;
|
||||
}
|
||||
|
||||
static void kvmvapic_vm_state_change(void *opaque, int running,
|
||||
@@ -777,7 +781,10 @@ static int vapic_post_load(void *opaque, int version_id)
|
||||
}
|
||||
}
|
||||
|
||||
s->vmsentry = qemu_add_vm_change_state_handler(kvmvapic_vm_state_change, s);
|
||||
if (!s->vmsentry) {
|
||||
s->vmsentry =
|
||||
qemu_add_vm_change_state_handler(kvmvapic_vm_state_change, s);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@@ -215,36 +215,6 @@ typedef struct {
|
||||
int fsref;
|
||||
} TSC210xRateInfo;
|
||||
|
||||
/* { rate, dsor, fsref } */
|
||||
static const TSC210xRateInfo tsc2101_rates[] = {
|
||||
/* Fsref / 6.0 */
|
||||
{ 7350, 7, 1 },
|
||||
{ 8000, 7, 0 },
|
||||
/* Fsref / 5.5 */
|
||||
{ 8018, 6, 1 },
|
||||
{ 8727, 6, 0 },
|
||||
/* Fsref / 5.0 */
|
||||
{ 8820, 5, 1 },
|
||||
{ 9600, 5, 0 },
|
||||
/* Fsref / 4.0 */
|
||||
{ 11025, 4, 1 },
|
||||
{ 12000, 4, 0 },
|
||||
/* Fsref / 3.0 */
|
||||
{ 14700, 3, 1 },
|
||||
{ 16000, 3, 0 },
|
||||
/* Fsref / 2.0 */
|
||||
{ 22050, 2, 1 },
|
||||
{ 24000, 2, 0 },
|
||||
/* Fsref / 1.5 */
|
||||
{ 29400, 1, 1 },
|
||||
{ 32000, 1, 0 },
|
||||
/* Fsref */
|
||||
{ 44100, 0, 1 },
|
||||
{ 48000, 0, 0 },
|
||||
|
||||
{ 0, 0, 0 },
|
||||
};
|
||||
|
||||
/* { rate, dsor, fsref } */
|
||||
static const TSC210xRateInfo tsc2102_rates[] = {
|
||||
/* Fsref / 6.0 */
|
||||
|
@@ -97,15 +97,6 @@ static inline int imx_avic_prio(IMXAVICState *s, int irq)
|
||||
return 0xf & (s->prio[word] >> part);
|
||||
}
|
||||
|
||||
static inline void imx_avic_set_prio(IMXAVICState *s, int irq, int prio)
|
||||
{
|
||||
uint32_t word = irq / PRIO_PER_WORD;
|
||||
uint32_t part = 4 * (irq % PRIO_PER_WORD);
|
||||
uint32_t mask = ~(0xf << part);
|
||||
s->prio[word] &= mask;
|
||||
s->prio[word] |= prio << part;
|
||||
}
|
||||
|
||||
/* Update interrupts. */
|
||||
static void imx_avic_update(IMXAVICState *s)
|
||||
{
|
||||
|
@@ -1098,10 +1098,10 @@ static void vfio_bar_write(void *opaque, hwaddr addr,
|
||||
buf.byte = data;
|
||||
break;
|
||||
case 2:
|
||||
buf.word = data;
|
||||
buf.word = cpu_to_le16(data);
|
||||
break;
|
||||
case 4:
|
||||
buf.dword = data;
|
||||
buf.dword = cpu_to_le32(data);
|
||||
break;
|
||||
default:
|
||||
hw_error("vfio: unsupported write size, %d bytes", size);
|
||||
@@ -1158,10 +1158,10 @@ static uint64_t vfio_bar_read(void *opaque,
|
||||
data = buf.byte;
|
||||
break;
|
||||
case 2:
|
||||
data = buf.word;
|
||||
data = le16_to_cpu(buf.word);
|
||||
break;
|
||||
case 4:
|
||||
data = buf.dword;
|
||||
data = le32_to_cpu(buf.dword);
|
||||
break;
|
||||
default:
|
||||
hw_error("vfio: unsupported read size, %d bytes", size);
|
||||
@@ -1188,7 +1188,7 @@ static uint64_t vfio_bar_read(void *opaque,
|
||||
static const MemoryRegionOps vfio_bar_ops = {
|
||||
.read = vfio_bar_read,
|
||||
.write = vfio_bar_write,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||||
};
|
||||
|
||||
static void vfio_pci_load_rom(VFIODevice *vdev)
|
||||
@@ -1255,7 +1255,7 @@ static uint64_t vfio_rom_read(void *opaque, hwaddr addr, unsigned size)
|
||||
uint16_t word;
|
||||
uint32_t dword;
|
||||
uint64_t qword;
|
||||
} buf;
|
||||
} val;
|
||||
uint64_t data = 0;
|
||||
|
||||
/* Load the ROM lazily when the guest tries to read it */
|
||||
@@ -1263,21 +1263,21 @@ static uint64_t vfio_rom_read(void *opaque, hwaddr addr, unsigned size)
|
||||
vfio_pci_load_rom(vdev);
|
||||
}
|
||||
|
||||
memcpy(&buf, vdev->rom + addr,
|
||||
memcpy(&val, vdev->rom + addr,
|
||||
(addr < vdev->rom_size) ? MIN(size, vdev->rom_size - addr) : 0);
|
||||
|
||||
switch (size) {
|
||||
case 1:
|
||||
data = buf.byte;
|
||||
data = val.byte;
|
||||
break;
|
||||
case 2:
|
||||
data = buf.word;
|
||||
data = le16_to_cpu(val.word);
|
||||
break;
|
||||
case 4:
|
||||
data = buf.dword;
|
||||
data = le32_to_cpu(val.dword);
|
||||
break;
|
||||
default:
|
||||
hw_error("vfio: unsupported read size, %d bytes", size);
|
||||
hw_error("vfio: unsupported read size, %d bytes\n", size);
|
||||
break;
|
||||
}
|
||||
|
||||
@@ -1296,7 +1296,7 @@ static void vfio_rom_write(void *opaque, hwaddr addr,
|
||||
static const MemoryRegionOps vfio_rom_ops = {
|
||||
.read = vfio_rom_read,
|
||||
.write = vfio_rom_write,
|
||||
.endianness = DEVICE_NATIVE_ENDIAN,
|
||||
.endianness = DEVICE_LITTLE_ENDIAN,
|
||||
};
|
||||
|
||||
static bool vfio_blacklist_opt_rom(VFIODevice *vdev)
|
||||
|
@@ -243,17 +243,25 @@ static void copy_sense_id_to_guest(SenseId *dest, SenseId *src)
|
||||
}
|
||||
}
|
||||
|
||||
static CCW1 copy_ccw_from_guest(hwaddr addr)
|
||||
static CCW1 copy_ccw_from_guest(hwaddr addr, bool fmt1)
|
||||
{
|
||||
CCW1 tmp;
|
||||
CCW0 tmp0;
|
||||
CCW1 tmp1;
|
||||
CCW1 ret;
|
||||
|
||||
cpu_physical_memory_read(addr, &tmp, sizeof(tmp));
|
||||
ret.cmd_code = tmp.cmd_code;
|
||||
ret.flags = tmp.flags;
|
||||
ret.count = be16_to_cpu(tmp.count);
|
||||
ret.cda = be32_to_cpu(tmp.cda);
|
||||
|
||||
if (fmt1) {
|
||||
cpu_physical_memory_read(addr, &tmp1, sizeof(tmp1));
|
||||
ret.cmd_code = tmp1.cmd_code;
|
||||
ret.flags = tmp1.flags;
|
||||
ret.count = be16_to_cpu(tmp1.count);
|
||||
ret.cda = be32_to_cpu(tmp1.cda);
|
||||
} else {
|
||||
cpu_physical_memory_read(addr, &tmp0, sizeof(tmp0));
|
||||
ret.cmd_code = tmp0.cmd_code;
|
||||
ret.flags = tmp0.flags;
|
||||
ret.count = be16_to_cpu(tmp0.count);
|
||||
ret.cda = be16_to_cpu(tmp0.cda1) | (tmp0.cda0 << 16);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -268,7 +276,8 @@ static int css_interpret_ccw(SubchDev *sch, hwaddr ccw_addr)
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
ccw = copy_ccw_from_guest(ccw_addr);
|
||||
/* Translate everything to format-1 ccws - the information is the same. */
|
||||
ccw = copy_ccw_from_guest(ccw_addr, sch->ccw_fmt_1);
|
||||
|
||||
/* Check for invalid command codes. */
|
||||
if ((ccw.cmd_code & 0x0f) == 0) {
|
||||
@@ -285,6 +294,13 @@ static int css_interpret_ccw(SubchDev *sch, hwaddr ccw_addr)
|
||||
|
||||
check_len = !((ccw.flags & CCW_FLAG_SLI) && !(ccw.flags & CCW_FLAG_DC));
|
||||
|
||||
if (!ccw.cda) {
|
||||
if (sch->ccw_no_data_cnt == 255) {
|
||||
return -EINVAL;
|
||||
}
|
||||
sch->ccw_no_data_cnt++;
|
||||
}
|
||||
|
||||
/* Look at the command. */
|
||||
switch (ccw.cmd_code) {
|
||||
case CCW_CMD_NOOP:
|
||||
@@ -386,6 +402,8 @@ static void sch_handle_start_func(SubchDev *sch, ORB *orb)
|
||||
s->ctrl |= (SCSW_STCTL_ALERT | SCSW_STCTL_STATUS_PEND);
|
||||
return;
|
||||
}
|
||||
sch->ccw_fmt_1 = !!(orb->ctrl0 & ORB_CTRL0_MASK_FMT);
|
||||
sch->ccw_no_data_cnt = 0;
|
||||
} else {
|
||||
s->ctrl &= ~(SCSW_ACTL_SUSP | SCSW_ACTL_RESUME_PEND);
|
||||
}
|
||||
@@ -1347,6 +1365,8 @@ void subch_device_save(SubchDev *s, QEMUFile *f)
|
||||
qemu_put_byte(f, s->id.ciw[i].command);
|
||||
qemu_put_be16(f, s->id.ciw[i].count);
|
||||
}
|
||||
qemu_put_byte(f, s->ccw_fmt_1);
|
||||
qemu_put_byte(f, s->ccw_no_data_cnt);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -1402,6 +1422,8 @@ int subch_device_load(SubchDev *s, QEMUFile *f)
|
||||
s->id.ciw[i].command = qemu_get_byte(f);
|
||||
s->id.ciw[i].count = qemu_get_be16(f);
|
||||
}
|
||||
s->ccw_fmt_1 = qemu_get_byte(f);
|
||||
s->ccw_no_data_cnt = qemu_get_byte(f);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@@ -76,7 +76,9 @@ struct SubchDev {
|
||||
hwaddr channel_prog;
|
||||
CCW1 last_cmd;
|
||||
bool last_cmd_valid;
|
||||
bool ccw_fmt_1;
|
||||
bool thinint_active;
|
||||
uint8_t ccw_no_data_cnt;
|
||||
/* transport-provided data: */
|
||||
int (*ccw_cb) (SubchDev *, CCW1);
|
||||
SenseId id;
|
||||
|
@@ -159,9 +159,8 @@ static int s390_virtio_net_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_virtio_net_instance_init(Object *obj)
|
||||
{
|
||||
VirtIONetS390 *dev = VIRTIO_NET_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_NET);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_NET);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static int s390_virtio_blk_init(VirtIOS390Device *s390_dev)
|
||||
@@ -178,9 +177,10 @@ static int s390_virtio_blk_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_virtio_blk_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOBlkS390 *dev = VIRTIO_BLK_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_BLK);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_BLK);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_unref(OBJECT(&dev->vdev));
|
||||
qdev_alias_all_properties(DEVICE(&dev->vdev), obj);
|
||||
object_property_add_alias(obj, "iothread", OBJECT(&dev->vdev),"iothread",
|
||||
&error_abort);
|
||||
}
|
||||
@@ -222,9 +222,8 @@ static int s390_virtio_serial_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_virtio_serial_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSerialS390 *dev = VIRTIO_SERIAL_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SERIAL);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SERIAL);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static int s390_virtio_scsi_init(VirtIOS390Device *s390_dev)
|
||||
@@ -255,9 +254,8 @@ static int s390_virtio_scsi_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_virtio_scsi_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSCSIS390 *dev = VIRTIO_SCSI_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SCSI);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_VHOST_SCSI
|
||||
@@ -277,9 +275,8 @@ static int s390_vhost_scsi_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_vhost_scsi_instance_init(Object *obj)
|
||||
{
|
||||
VHostSCSIS390 *dev = VHOST_SCSI_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VHOST_SCSI);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VHOST_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
#endif
|
||||
|
||||
@@ -304,9 +301,8 @@ static int s390_virtio_rng_init(VirtIOS390Device *s390_dev)
|
||||
static void s390_virtio_rng_instance_init(Object *obj)
|
||||
{
|
||||
VirtIORNGS390 *dev = VIRTIO_RNG_S390(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_RNG);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_RNG);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_property_add_link(obj, "rng", TYPE_RNG_BACKEND,
|
||||
(Object **)&dev->vdev.conf.rng,
|
||||
qdev_prop_allow_set_link_before_realize,
|
||||
@@ -497,8 +493,10 @@ static unsigned virtio_s390_get_features(DeviceState *d)
|
||||
/**************** S390 Virtio Bus Device Descriptions *******************/
|
||||
|
||||
static Property s390_virtio_net_properties[] = {
|
||||
DEFINE_NIC_PROPERTIES(VirtIONetS390, vdev.nic_conf),
|
||||
DEFINE_VIRTIO_COMMON_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_VIRTIO_NET_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_VIRTIO_NET_PROPERTIES(VirtIONetS390, vdev.net_conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -535,6 +533,7 @@ static const TypeInfo s390_virtio_blk = {
|
||||
};
|
||||
|
||||
static Property s390_virtio_serial_properties[] = {
|
||||
DEFINE_VIRTIO_SERIAL_PROPERTIES(VirtIOSerialS390, vdev.serial),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -557,6 +556,7 @@ static const TypeInfo s390_virtio_serial = {
|
||||
|
||||
static Property s390_virtio_rng_properties[] = {
|
||||
DEFINE_VIRTIO_COMMON_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_VIRTIO_RNG_PROPERTIES(VirtIORNGS390, vdev.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -614,6 +614,7 @@ static const TypeInfo virtio_s390_device_info = {
|
||||
};
|
||||
|
||||
static Property s390_virtio_scsi_properties[] = {
|
||||
DEFINE_VIRTIO_SCSI_PROPERTIES(VirtIOSCSIS390, vdev.parent_obj.conf),
|
||||
DEFINE_VIRTIO_COMMON_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_VIRTIO_SCSI_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
@@ -639,6 +640,7 @@ static const TypeInfo s390_virtio_scsi = {
|
||||
#ifdef CONFIG_VHOST_SCSI
|
||||
static Property s390_vhost_scsi_properties[] = {
|
||||
DEFINE_VIRTIO_COMMON_FEATURES(VirtIOS390Device, host_features),
|
||||
DEFINE_VHOST_SCSI_PROPERTIES(VHostSCSIS390, vdev.parent_obj.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
|
@@ -792,9 +792,8 @@ static int virtio_ccw_net_init(VirtioCcwDevice *ccw_dev)
|
||||
static void virtio_ccw_net_instance_init(Object *obj)
|
||||
{
|
||||
VirtIONetCcw *dev = VIRTIO_NET_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_NET);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_NET);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static int virtio_ccw_blk_init(VirtioCcwDevice *ccw_dev)
|
||||
@@ -812,9 +811,10 @@ static int virtio_ccw_blk_init(VirtioCcwDevice *ccw_dev)
|
||||
static void virtio_ccw_blk_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOBlkCcw *dev = VIRTIO_BLK_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_BLK);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_BLK);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_unref(OBJECT(&dev->vdev));
|
||||
qdev_alias_all_properties(DEVICE(&dev->vdev), obj);
|
||||
object_property_add_alias(obj, "iothread", OBJECT(&dev->vdev),"iothread",
|
||||
&error_abort);
|
||||
}
|
||||
@@ -848,9 +848,8 @@ static int virtio_ccw_serial_init(VirtioCcwDevice *ccw_dev)
|
||||
static void virtio_ccw_serial_instance_init(Object *obj)
|
||||
{
|
||||
VirtioSerialCcw *dev = VIRTIO_SERIAL_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SERIAL);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SERIAL);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static int virtio_ccw_balloon_init(VirtioCcwDevice *ccw_dev)
|
||||
@@ -897,7 +896,7 @@ static void virtio_ccw_balloon_instance_init(Object *obj)
|
||||
VirtIOBalloonCcw *dev = VIRTIO_BALLOON_CCW(obj);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_BALLOON);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_unref(OBJECT(&dev->vdev));
|
||||
|
||||
object_property_add(obj, "guest-stats", "guest statistics",
|
||||
balloon_ccw_stats_get_all, NULL, NULL, dev, NULL);
|
||||
|
||||
@@ -935,11 +934,8 @@ static int virtio_ccw_scsi_init(VirtioCcwDevice *ccw_dev)
|
||||
static void virtio_ccw_scsi_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSCSICcw *dev = VIRTIO_SCSI_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SCSI);
|
||||
object_property_add_alias(obj, "iothread", OBJECT(&dev->vdev), "iothread",
|
||||
&error_abort);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_VHOST_SCSI
|
||||
@@ -959,9 +955,8 @@ static int vhost_ccw_scsi_init(VirtioCcwDevice *ccw_dev)
|
||||
static void vhost_ccw_scsi_instance_init(Object *obj)
|
||||
{
|
||||
VHostSCSICcw *dev = VHOST_SCSI_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VHOST_SCSI);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VHOST_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
#endif
|
||||
|
||||
@@ -1379,6 +1374,8 @@ static int virtio_ccw_load_config(DeviceState *d, QEMUFile *f)
|
||||
static Property virtio_ccw_net_properties[] = {
|
||||
DEFINE_PROP_STRING("devno", VirtioCcwDevice, bus_id),
|
||||
DEFINE_VIRTIO_NET_FEATURES(VirtioCcwDevice, host_features[0]),
|
||||
DEFINE_VIRTIO_NET_PROPERTIES(VirtIONetCcw, vdev.net_conf),
|
||||
DEFINE_NIC_PROPERTIES(VirtIONetCcw, vdev.nic_conf),
|
||||
DEFINE_PROP_BIT("ioeventfd", VirtioCcwDevice, flags,
|
||||
VIRTIO_CCW_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
@@ -1431,6 +1428,7 @@ static const TypeInfo virtio_ccw_blk = {
|
||||
|
||||
static Property virtio_ccw_serial_properties[] = {
|
||||
DEFINE_PROP_STRING("devno", VirtioCcwDevice, bus_id),
|
||||
DEFINE_VIRTIO_SERIAL_PROPERTIES(VirtioSerialCcw, vdev.serial),
|
||||
DEFINE_PROP_BIT("ioeventfd", VirtioCcwDevice, flags,
|
||||
VIRTIO_CCW_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
@@ -1483,6 +1481,7 @@ static const TypeInfo virtio_ccw_balloon = {
|
||||
|
||||
static Property virtio_ccw_scsi_properties[] = {
|
||||
DEFINE_PROP_STRING("devno", VirtioCcwDevice, bus_id),
|
||||
DEFINE_VIRTIO_SCSI_PROPERTIES(VirtIOSCSICcw, vdev.parent_obj.conf),
|
||||
DEFINE_VIRTIO_SCSI_FEATURES(VirtioCcwDevice, host_features[0]),
|
||||
DEFINE_PROP_BIT("ioeventfd", VirtioCcwDevice, flags,
|
||||
VIRTIO_CCW_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
@@ -1511,6 +1510,7 @@ static const TypeInfo virtio_ccw_scsi = {
|
||||
#ifdef CONFIG_VHOST_SCSI
|
||||
static Property vhost_ccw_scsi_properties[] = {
|
||||
DEFINE_PROP_STRING("devno", VirtioCcwDevice, bus_id),
|
||||
DEFINE_VHOST_SCSI_PROPERTIES(VirtIOSCSICcw, vdev.parent_obj.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1537,9 +1537,8 @@ static const TypeInfo vhost_ccw_scsi = {
|
||||
static void virtio_ccw_rng_instance_init(Object *obj)
|
||||
{
|
||||
VirtIORNGCcw *dev = VIRTIO_RNG_CCW(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_RNG);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_RNG);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_property_add_link(obj, "rng", TYPE_RNG_BACKEND,
|
||||
(Object **)&dev->vdev.conf.rng,
|
||||
qdev_prop_allow_set_link_before_realize,
|
||||
@@ -1548,6 +1547,7 @@ static void virtio_ccw_rng_instance_init(Object *obj)
|
||||
|
||||
static Property virtio_ccw_rng_properties[] = {
|
||||
DEFINE_PROP_STRING("devno", VirtioCcwDevice, bus_id),
|
||||
DEFINE_VIRTIO_RNG_PROPERTIES(VirtIORNGCcw, vdev.conf),
|
||||
DEFINE_PROP_BIT("ioeventfd", VirtioCcwDevice, flags,
|
||||
VIRTIO_CCW_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
|
@@ -8,6 +8,6 @@ common-obj-$(CONFIG_ESP_PCI) += esp-pci.o
|
||||
obj-$(CONFIG_PSERIES) += spapr_vscsi.o
|
||||
|
||||
ifeq ($(CONFIG_VIRTIO),y)
|
||||
obj-y += virtio-scsi.o virtio-scsi-dataplane.o
|
||||
obj-y += virtio-scsi.o
|
||||
obj-$(CONFIG_VHOST_SCSI) += vhost-scsi.o
|
||||
endif
|
||||
|
@@ -551,11 +551,8 @@ SCSIRequest *scsi_req_alloc(const SCSIReqOps *reqops, SCSIDevice *d,
|
||||
SCSIRequest *req;
|
||||
SCSIBus *bus = scsi_bus_from_device(d);
|
||||
BusState *qbus = BUS(bus);
|
||||
const int memset_off = offsetof(SCSIRequest, sense)
|
||||
+ sizeof(req->sense);
|
||||
|
||||
req = g_slice_alloc(reqops->size);
|
||||
memset((uint8_t *)req + memset_off, 0, reqops->size - memset_off);
|
||||
req = g_malloc0(reqops->size);
|
||||
req->refcount = 1;
|
||||
req->bus = bus;
|
||||
req->dev = d;
|
||||
@@ -563,10 +560,10 @@ SCSIRequest *scsi_req_alloc(const SCSIReqOps *reqops, SCSIDevice *d,
|
||||
req->lun = lun;
|
||||
req->hba_private = hba_private;
|
||||
req->status = -1;
|
||||
req->sense_len = 0;
|
||||
req->ops = reqops;
|
||||
object_ref(OBJECT(d));
|
||||
object_ref(OBJECT(qbus->parent));
|
||||
notifier_list_init(&req->cancel_notifiers);
|
||||
trace_scsi_req_alloc(req->dev->id, req->lun, req->tag);
|
||||
return req;
|
||||
}
|
||||
@@ -1606,7 +1603,7 @@ void scsi_req_unref(SCSIRequest *req)
|
||||
}
|
||||
object_unref(OBJECT(req->dev));
|
||||
object_unref(OBJECT(qbus->parent));
|
||||
g_slice_free1(req->ops->size, req);
|
||||
g_free(req);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1716,44 +1713,9 @@ void scsi_req_complete(SCSIRequest *req, int status)
|
||||
scsi_req_ref(req);
|
||||
scsi_req_dequeue(req);
|
||||
req->bus->info->complete(req, req->status, req->resid);
|
||||
|
||||
/* Cancelled requests might end up being completed instead of cancelled */
|
||||
notifier_list_notify(&req->cancel_notifiers, req);
|
||||
scsi_req_unref(req);
|
||||
}
|
||||
|
||||
/* Called by the devices when the request is canceled. */
|
||||
void scsi_req_cancel_complete(SCSIRequest *req)
|
||||
{
|
||||
assert(req->io_canceled);
|
||||
if (req->bus->info->cancel) {
|
||||
req->bus->info->cancel(req);
|
||||
}
|
||||
notifier_list_notify(&req->cancel_notifiers, req);
|
||||
scsi_req_unref(req);
|
||||
}
|
||||
|
||||
/* Cancel @req asynchronously. @notifier is added to @req's cancellation
|
||||
* notifier list, the bus will be notified the requests cancellation is
|
||||
* completed.
|
||||
* */
|
||||
void scsi_req_cancel_async(SCSIRequest *req, Notifier *notifier)
|
||||
{
|
||||
trace_scsi_req_cancel(req->dev->id, req->lun, req->tag);
|
||||
if (notifier) {
|
||||
notifier_list_add(&req->cancel_notifiers, notifier);
|
||||
}
|
||||
if (req->io_canceled) {
|
||||
return;
|
||||
}
|
||||
scsi_req_ref(req);
|
||||
scsi_req_dequeue(req);
|
||||
req->io_canceled = true;
|
||||
if (req->aiocb) {
|
||||
bdrv_aio_cancel_async(req->aiocb);
|
||||
}
|
||||
}
|
||||
|
||||
void scsi_req_cancel(SCSIRequest *req)
|
||||
{
|
||||
trace_scsi_req_cancel(req->dev->id, req->lun, req->tag);
|
||||
@@ -1763,9 +1725,28 @@ void scsi_req_cancel(SCSIRequest *req)
|
||||
scsi_req_ref(req);
|
||||
scsi_req_dequeue(req);
|
||||
req->io_canceled = true;
|
||||
if (req->aiocb) {
|
||||
bdrv_aio_cancel(req->aiocb);
|
||||
if (req->ops->cancel_io) {
|
||||
req->ops->cancel_io(req);
|
||||
}
|
||||
if (req->bus->info->cancel) {
|
||||
req->bus->info->cancel(req);
|
||||
}
|
||||
scsi_req_unref(req);
|
||||
}
|
||||
|
||||
void scsi_req_abort(SCSIRequest *req, int status)
|
||||
{
|
||||
if (!req->enqueued) {
|
||||
return;
|
||||
}
|
||||
scsi_req_ref(req);
|
||||
scsi_req_dequeue(req);
|
||||
req->io_canceled = true;
|
||||
if (req->ops->cancel_io) {
|
||||
req->ops->cancel_io(req);
|
||||
}
|
||||
scsi_req_complete(req, status);
|
||||
scsi_req_unref(req);
|
||||
}
|
||||
|
||||
static int scsi_ua_precedence(SCSISense sense)
|
||||
|
@@ -105,6 +105,23 @@ static void scsi_check_condition(SCSIDiskReq *r, SCSISense sense)
|
||||
scsi_req_complete(&r->req, CHECK_CONDITION);
|
||||
}
|
||||
|
||||
/* Cancel a pending data transfer. */
|
||||
static void scsi_cancel_io(SCSIRequest *req)
|
||||
{
|
||||
SCSIDiskReq *r = DO_UPCAST(SCSIDiskReq, req, req);
|
||||
|
||||
DPRINTF("Cancel tag=0x%x\n", req->tag);
|
||||
if (r->req.aiocb) {
|
||||
bdrv_aio_cancel(r->req.aiocb);
|
||||
|
||||
/* This reference was left in by scsi_*_data. We take ownership of
|
||||
* it the moment scsi_req_cancel is called, independent of whether
|
||||
* bdrv_aio_cancel completes the request or not. */
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
r->req.aiocb = NULL;
|
||||
}
|
||||
|
||||
static uint32_t scsi_init_iovec(SCSIDiskReq *r, size_t size)
|
||||
{
|
||||
SCSIDiskState *s = DO_UPCAST(SCSIDiskState, qdev, r->req.dev);
|
||||
@@ -168,7 +185,6 @@ static void scsi_aio_complete(void *opaque, int ret)
|
||||
r->req.aiocb = NULL;
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -181,7 +197,9 @@ static void scsi_aio_complete(void *opaque, int ret)
|
||||
scsi_req_complete(&r->req, GOOD);
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
static bool scsi_is_cmd_fua(SCSICommand *cmd)
|
||||
@@ -215,7 +233,6 @@ static void scsi_write_do_fua(SCSIDiskReq *r)
|
||||
SCSIDiskState *s = DO_UPCAST(SCSIDiskState, qdev, r->req.dev);
|
||||
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -229,7 +246,9 @@ static void scsi_write_do_fua(SCSIDiskReq *r)
|
||||
scsi_req_complete(&r->req, GOOD);
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
static void scsi_dma_complete_noio(void *opaque, int ret)
|
||||
@@ -242,7 +261,6 @@ static void scsi_dma_complete_noio(void *opaque, int ret)
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
}
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -262,7 +280,9 @@ static void scsi_dma_complete_noio(void *opaque, int ret)
|
||||
}
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
static void scsi_dma_complete(void *opaque, int ret)
|
||||
@@ -283,7 +303,6 @@ static void scsi_read_complete(void * opaque, int ret)
|
||||
r->req.aiocb = NULL;
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -301,7 +320,9 @@ static void scsi_read_complete(void * opaque, int ret)
|
||||
scsi_req_data(&r->req, r->qiov.size);
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
/* Actually issue a read to the block device. */
|
||||
@@ -316,7 +337,6 @@ static void scsi_do_read(void *opaque, int ret)
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
}
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -343,7 +363,9 @@ static void scsi_do_read(void *opaque, int ret)
|
||||
}
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
/* Read more data from scsi device into buffer. */
|
||||
@@ -437,7 +459,6 @@ static void scsi_write_complete(void * opaque, int ret)
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
}
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -460,7 +481,9 @@ static void scsi_write_complete(void * opaque, int ret)
|
||||
}
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
static void scsi_write_data(SCSIRequest *req)
|
||||
@@ -1530,7 +1553,6 @@ static void scsi_unmap_complete(void *opaque, int ret)
|
||||
|
||||
r->req.aiocb = NULL;
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -1560,7 +1582,9 @@ static void scsi_unmap_complete(void *opaque, int ret)
|
||||
scsi_req_complete(&r->req, GOOD);
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
g_free(data);
|
||||
}
|
||||
|
||||
@@ -1630,7 +1654,6 @@ static void scsi_write_same_complete(void *opaque, int ret)
|
||||
r->req.aiocb = NULL;
|
||||
block_acct_done(bdrv_get_stats(s->qdev.conf.bs), &r->acct);
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
|
||||
@@ -1655,7 +1678,9 @@ static void scsi_write_same_complete(void *opaque, int ret)
|
||||
scsi_req_complete(&r->req, GOOD);
|
||||
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
qemu_vfree(data->iov.iov_base);
|
||||
g_free(data);
|
||||
}
|
||||
@@ -2321,6 +2346,7 @@ static const SCSIReqOps scsi_disk_emulate_reqops = {
|
||||
.send_command = scsi_disk_emulate_command,
|
||||
.read_data = scsi_disk_emulate_read_data,
|
||||
.write_data = scsi_disk_emulate_write_data,
|
||||
.cancel_io = scsi_cancel_io,
|
||||
.get_buf = scsi_get_buf,
|
||||
};
|
||||
|
||||
@@ -2330,6 +2356,7 @@ static const SCSIReqOps scsi_disk_dma_reqops = {
|
||||
.send_command = scsi_disk_dma_command,
|
||||
.read_data = scsi_read_data,
|
||||
.write_data = scsi_write_data,
|
||||
.cancel_io = scsi_cancel_io,
|
||||
.get_buf = scsi_get_buf,
|
||||
.load_request = scsi_disk_load_request,
|
||||
.save_request = scsi_disk_save_request,
|
||||
|
@@ -93,10 +93,6 @@ static void scsi_command_complete(void *opaque, int ret)
|
||||
SCSIGenericReq *r = (SCSIGenericReq *)opaque;
|
||||
|
||||
r->req.aiocb = NULL;
|
||||
if (r->req.io_canceled) {
|
||||
scsi_req_cancel_complete(&r->req);
|
||||
goto done;
|
||||
}
|
||||
if (r->io_header.driver_status & SG_ERR_DRIVER_SENSE) {
|
||||
r->req.sense_len = r->io_header.sb_len_wr;
|
||||
}
|
||||
@@ -137,8 +133,26 @@ static void scsi_command_complete(void *opaque, int ret)
|
||||
r, r->req.tag, status);
|
||||
|
||||
scsi_req_complete(&r->req, status);
|
||||
done:
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
|
||||
/* Cancel a pending data transfer. */
|
||||
static void scsi_cancel_io(SCSIRequest *req)
|
||||
{
|
||||
SCSIGenericReq *r = DO_UPCAST(SCSIGenericReq, req, req);
|
||||
|
||||
DPRINTF("Cancel tag=0x%x\n", req->tag);
|
||||
if (r->req.aiocb) {
|
||||
bdrv_aio_cancel(r->req.aiocb);
|
||||
|
||||
/* This reference was left in by scsi_*_data. We take ownership of
|
||||
* it independent of whether bdrv_aio_cancel completes the request
|
||||
* or not. */
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
r->req.aiocb = NULL;
|
||||
}
|
||||
|
||||
static int execute_command(BlockDriverState *bdrv,
|
||||
@@ -172,7 +186,8 @@ static void scsi_read_complete(void * opaque, int ret)
|
||||
int len;
|
||||
|
||||
r->req.aiocb = NULL;
|
||||
if (ret || r->req.io_canceled) {
|
||||
if (ret) {
|
||||
DPRINTF("IO error ret %d\n", ret);
|
||||
scsi_command_complete(r, ret);
|
||||
return;
|
||||
}
|
||||
@@ -196,7 +211,9 @@ static void scsi_read_complete(void * opaque, int ret)
|
||||
bdrv_set_guest_block_size(s->conf.bs, s->blocksize);
|
||||
|
||||
scsi_req_data(&r->req, len);
|
||||
scsi_req_unref(&r->req);
|
||||
if (!r->req.io_canceled) {
|
||||
scsi_req_unref(&r->req);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -229,7 +246,8 @@ static void scsi_write_complete(void * opaque, int ret)
|
||||
|
||||
DPRINTF("scsi_write_complete() ret = %d\n", ret);
|
||||
r->req.aiocb = NULL;
|
||||
if (ret || r->req.io_canceled) {
|
||||
if (ret) {
|
||||
DPRINTF("IO error\n");
|
||||
scsi_command_complete(r, ret);
|
||||
return;
|
||||
}
|
||||
@@ -447,6 +465,7 @@ const SCSIReqOps scsi_generic_req_ops = {
|
||||
.send_command = scsi_send_command,
|
||||
.read_data = scsi_read_data,
|
||||
.write_data = scsi_write_data,
|
||||
.cancel_io = scsi_cancel_io,
|
||||
.get_buf = scsi_get_buf,
|
||||
.load_request = scsi_generic_load_request,
|
||||
.save_request = scsi_generic_save_request,
|
||||
|
@@ -77,9 +77,8 @@ typedef struct vscsi_req {
|
||||
SCSIRequest *sreq;
|
||||
uint32_t qtag; /* qemu tag != srp tag */
|
||||
bool active;
|
||||
bool writing;
|
||||
bool dma_error;
|
||||
uint32_t data_len;
|
||||
bool writing;
|
||||
uint32_t senselen;
|
||||
uint8_t sense[SCSI_SENSE_BUF_SIZE];
|
||||
|
||||
@@ -537,8 +536,8 @@ static void vscsi_transfer_data(SCSIRequest *sreq, uint32_t len)
|
||||
}
|
||||
if (rc < 0) {
|
||||
fprintf(stderr, "VSCSI: RDMA error rc=%d!\n", rc);
|
||||
req->dma_error = true;
|
||||
scsi_req_cancel(req->sreq);
|
||||
vscsi_makeup_sense(s, req, HARDWARE_ERROR, 0, 0);
|
||||
scsi_req_abort(req->sreq, CHECK_CONDITION);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -592,12 +591,6 @@ static void vscsi_request_cancelled(SCSIRequest *sreq)
|
||||
{
|
||||
vscsi_req *req = sreq->hba_private;
|
||||
|
||||
if (req->dma_error) {
|
||||
VSCSIState *s = VIO_SPAPR_VSCSI_DEVICE(sreq->bus->qbus.parent);
|
||||
|
||||
vscsi_makeup_sense(s, req, HARDWARE_ERROR, 0, 0);
|
||||
vscsi_send_rsp(s, req, CHECK_CONDITION, 0, 0);
|
||||
}
|
||||
vscsi_put_req(req);
|
||||
}
|
||||
|
||||
|
@@ -23,7 +23,6 @@
|
||||
#include "hw/virtio/vhost.h"
|
||||
#include "hw/virtio/virtio-scsi.h"
|
||||
#include "hw/virtio/virtio-bus.h"
|
||||
#include "hw/virtio/virtio-access.h"
|
||||
|
||||
/* Features supported by host kernel. */
|
||||
static const int kernel_feature_bits[] = {
|
||||
@@ -164,8 +163,8 @@ static void vhost_scsi_set_config(VirtIODevice *vdev,
|
||||
VirtIOSCSIConfig *scsiconf = (VirtIOSCSIConfig *)config;
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(vdev);
|
||||
|
||||
if ((uint32_t) virtio_ldl_p(vdev, &scsiconf->sense_size) != vs->sense_size ||
|
||||
(uint32_t) virtio_ldl_p(vdev, &scsiconf->cdb_size) != vs->cdb_size) {
|
||||
if ((uint32_t) ldl_p(&scsiconf->sense_size) != vs->sense_size ||
|
||||
(uint32_t) ldl_p(&scsiconf->cdb_size) != vs->cdb_size) {
|
||||
error_report("vhost-scsi does not support changing the sense data and CDB sizes");
|
||||
exit(1);
|
||||
}
|
||||
|
@@ -1,229 +0,0 @@
|
||||
/*
|
||||
* Virtio SCSI dataplane
|
||||
*
|
||||
* Copyright Red Hat, Inc. 2014
|
||||
*
|
||||
* Authors:
|
||||
* Fam Zheng <famz@redhat.com>
|
||||
*
|
||||
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||
* See the COPYING file in the top-level directory.
|
||||
*
|
||||
*/
|
||||
|
||||
#include "hw/virtio/virtio-scsi.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include <hw/scsi/scsi.h>
|
||||
#include <block/scsi.h>
|
||||
#include <hw/virtio/virtio-bus.h>
|
||||
#include "hw/virtio/virtio-access.h"
|
||||
#include "stdio.h"
|
||||
|
||||
/* Context: QEMU global mutex held */
|
||||
void virtio_scsi_set_iothread(VirtIOSCSI *s, IOThread *iothread)
|
||||
{
|
||||
BusState *qbus = BUS(qdev_get_parent_bus(DEVICE(s)));
|
||||
VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(s);
|
||||
|
||||
assert(!s->ctx);
|
||||
s->ctx = iothread_get_aio_context(vs->conf.iothread);
|
||||
|
||||
/* Don't try if transport does not support notifiers. */
|
||||
if (!k->set_guest_notifiers || !k->set_host_notifier) {
|
||||
fprintf(stderr, "virtio-scsi: Failed to set iothread "
|
||||
"(transport does not support notifiers)");
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
static VirtIOSCSIVring *virtio_scsi_vring_init(VirtIOSCSI *s,
|
||||
VirtQueue *vq,
|
||||
EventNotifierHandler *handler,
|
||||
int n)
|
||||
{
|
||||
BusState *qbus = BUS(qdev_get_parent_bus(DEVICE(s)));
|
||||
VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
|
||||
VirtIOSCSIVring *r = g_slice_new(VirtIOSCSIVring);
|
||||
|
||||
/* Set up virtqueue notify */
|
||||
if (k->set_host_notifier(qbus->parent, n, true) != 0) {
|
||||
fprintf(stderr, "virtio-scsi: Failed to set host notifier\n");
|
||||
exit(1);
|
||||
}
|
||||
r->host_notifier = *virtio_queue_get_host_notifier(vq);
|
||||
r->guest_notifier = *virtio_queue_get_guest_notifier(vq);
|
||||
aio_set_event_notifier(s->ctx, &r->host_notifier, handler);
|
||||
|
||||
r->parent = s;
|
||||
|
||||
if (!vring_setup(&r->vring, VIRTIO_DEVICE(s), n)) {
|
||||
fprintf(stderr, "virtio-scsi: VRing setup failed\n");
|
||||
exit(1);
|
||||
}
|
||||
return r;
|
||||
}
|
||||
|
||||
VirtIOSCSIReq *virtio_scsi_pop_req_vring(VirtIOSCSI *s,
|
||||
VirtIOSCSIVring *vring)
|
||||
{
|
||||
VirtIOSCSIReq *req = virtio_scsi_init_req(s, NULL);
|
||||
int r;
|
||||
|
||||
req->vring = vring;
|
||||
r = vring_pop((VirtIODevice *)s, &vring->vring, &req->elem);
|
||||
if (r < 0) {
|
||||
virtio_scsi_free_req(req);
|
||||
req = NULL;
|
||||
}
|
||||
return req;
|
||||
}
|
||||
|
||||
void virtio_scsi_vring_push_notify(VirtIOSCSIReq *req)
|
||||
{
|
||||
vring_push(&req->vring->vring, &req->elem,
|
||||
req->qsgl.size + req->resp_iov.size);
|
||||
event_notifier_set(&req->vring->guest_notifier);
|
||||
}
|
||||
|
||||
static void virtio_scsi_iothread_handle_ctrl(EventNotifier *notifier)
|
||||
{
|
||||
VirtIOSCSIVring *vring = container_of(notifier,
|
||||
VirtIOSCSIVring, host_notifier);
|
||||
VirtIOSCSI *s = VIRTIO_SCSI(vring->parent);
|
||||
VirtIOSCSIReq *req;
|
||||
|
||||
event_notifier_test_and_clear(notifier);
|
||||
while ((req = virtio_scsi_pop_req_vring(s, vring))) {
|
||||
virtio_scsi_handle_ctrl_req(s, req);
|
||||
}
|
||||
}
|
||||
|
||||
static void virtio_scsi_iothread_handle_event(EventNotifier *notifier)
|
||||
{
|
||||
VirtIOSCSIVring *vring = container_of(notifier,
|
||||
VirtIOSCSIVring, host_notifier);
|
||||
VirtIOSCSI *s = vring->parent;
|
||||
VirtIODevice *vdev = VIRTIO_DEVICE(s);
|
||||
|
||||
event_notifier_test_and_clear(notifier);
|
||||
|
||||
if (!(vdev->status & VIRTIO_CONFIG_S_DRIVER_OK)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (s->events_dropped) {
|
||||
virtio_scsi_push_event(s, NULL, VIRTIO_SCSI_T_NO_EVENT, 0);
|
||||
}
|
||||
}
|
||||
|
||||
static void virtio_scsi_iothread_handle_cmd(EventNotifier *notifier)
|
||||
{
|
||||
VirtIOSCSIVring *vring = container_of(notifier,
|
||||
VirtIOSCSIVring, host_notifier);
|
||||
VirtIOSCSI *s = (VirtIOSCSI *)vring->parent;
|
||||
VirtIOSCSIReq *req, *next;
|
||||
QTAILQ_HEAD(, VirtIOSCSIReq) reqs = QTAILQ_HEAD_INITIALIZER(reqs);
|
||||
|
||||
event_notifier_test_and_clear(notifier);
|
||||
while ((req = virtio_scsi_pop_req_vring(s, vring))) {
|
||||
if (virtio_scsi_handle_cmd_req_prepare(s, req)) {
|
||||
QTAILQ_INSERT_TAIL(&reqs, req, next);
|
||||
}
|
||||
}
|
||||
|
||||
QTAILQ_FOREACH_SAFE(req, &reqs, next, next) {
|
||||
virtio_scsi_handle_cmd_req_submit(s, req);
|
||||
}
|
||||
}
|
||||
|
||||
/* Context: QEMU global mutex held */
|
||||
void virtio_scsi_dataplane_start(VirtIOSCSI *s)
|
||||
{
|
||||
int i;
|
||||
int rc;
|
||||
BusState *qbus = BUS(qdev_get_parent_bus(DEVICE(s)));
|
||||
VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(s);
|
||||
|
||||
if (s->dataplane_started ||
|
||||
s->dataplane_starting ||
|
||||
s->ctx != iothread_get_aio_context(vs->conf.iothread)) {
|
||||
return;
|
||||
}
|
||||
|
||||
s->dataplane_starting = true;
|
||||
|
||||
/* Set up guest notifier (irq) */
|
||||
rc = k->set_guest_notifiers(qbus->parent, vs->conf.num_queues + 2, true);
|
||||
if (rc != 0) {
|
||||
fprintf(stderr, "virtio-scsi: Failed to set guest notifiers, "
|
||||
"ensure -enable-kvm is set\n");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
aio_context_acquire(s->ctx);
|
||||
s->ctrl_vring = virtio_scsi_vring_init(s, vs->ctrl_vq,
|
||||
virtio_scsi_iothread_handle_ctrl,
|
||||
0);
|
||||
s->event_vring = virtio_scsi_vring_init(s, vs->event_vq,
|
||||
virtio_scsi_iothread_handle_event,
|
||||
1);
|
||||
s->cmd_vrings = g_malloc0(sizeof(VirtIOSCSIVring) * vs->conf.num_queues);
|
||||
for (i = 0; i < vs->conf.num_queues; i++) {
|
||||
s->cmd_vrings[i] =
|
||||
virtio_scsi_vring_init(s, vs->cmd_vqs[i],
|
||||
virtio_scsi_iothread_handle_cmd,
|
||||
i + 2);
|
||||
}
|
||||
|
||||
aio_context_release(s->ctx);
|
||||
s->dataplane_starting = false;
|
||||
s->dataplane_started = true;
|
||||
}
|
||||
|
||||
/* Context: QEMU global mutex held */
|
||||
void virtio_scsi_dataplane_stop(VirtIOSCSI *s)
|
||||
{
|
||||
BusState *qbus = BUS(qdev_get_parent_bus(DEVICE(s)));
|
||||
VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
|
||||
VirtIODevice *vdev = VIRTIO_DEVICE(s);
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(s);
|
||||
int i;
|
||||
|
||||
if (!s->dataplane_started || s->dataplane_stopping) {
|
||||
return;
|
||||
}
|
||||
s->dataplane_stopping = true;
|
||||
assert(s->ctx == iothread_get_aio_context(vs->conf.iothread));
|
||||
|
||||
aio_context_acquire(s->ctx);
|
||||
|
||||
aio_set_event_notifier(s->ctx, &s->ctrl_vring->host_notifier, NULL);
|
||||
aio_set_event_notifier(s->ctx, &s->event_vring->host_notifier, NULL);
|
||||
for (i = 0; i < vs->conf.num_queues; i++) {
|
||||
aio_set_event_notifier(s->ctx, &s->cmd_vrings[i]->host_notifier, NULL);
|
||||
}
|
||||
|
||||
bdrv_drain_all(); /* ensure there are no in-flight requests */
|
||||
|
||||
aio_context_release(s->ctx);
|
||||
|
||||
/* Sync vring state back to virtqueue so that non-dataplane request
|
||||
* processing can continue when we disable the host notifier below.
|
||||
*/
|
||||
vring_teardown(&s->ctrl_vring->vring, vdev, 0);
|
||||
vring_teardown(&s->event_vring->vring, vdev, 1);
|
||||
for (i = 0; i < vs->conf.num_queues; i++) {
|
||||
vring_teardown(&s->cmd_vrings[i]->vring, vdev, 2 + i);
|
||||
}
|
||||
|
||||
for (i = 0; i < vs->conf.num_queues + 2; i++) {
|
||||
k->set_host_notifier(qbus->parent, i, false);
|
||||
}
|
||||
|
||||
/* Clean up guest notifier (irq) */
|
||||
k->set_guest_notifiers(qbus->parent, vs->conf.num_queues + 2, false);
|
||||
s->dataplane_stopping = false;
|
||||
s->dataplane_started = false;
|
||||
}
|
@@ -20,7 +20,34 @@
|
||||
#include <block/scsi.h>
|
||||
#include <hw/virtio/virtio-bus.h>
|
||||
#include "hw/virtio/virtio-access.h"
|
||||
#include "migration/migration.h"
|
||||
|
||||
typedef struct VirtIOSCSIReq {
|
||||
VirtIOSCSI *dev;
|
||||
VirtQueue *vq;
|
||||
VirtQueueElement elem;
|
||||
QEMUSGList qsgl;
|
||||
SCSIRequest *sreq;
|
||||
size_t resp_size;
|
||||
enum SCSIXferMode mode;
|
||||
QEMUIOVector resp_iov;
|
||||
union {
|
||||
VirtIOSCSICmdResp cmd;
|
||||
VirtIOSCSICtrlTMFResp tmf;
|
||||
VirtIOSCSICtrlANResp an;
|
||||
VirtIOSCSIEvent event;
|
||||
} resp;
|
||||
union {
|
||||
struct {
|
||||
VirtIOSCSICmdReq cmd;
|
||||
uint8_t cdb[];
|
||||
} QEMU_PACKED;
|
||||
VirtIOSCSICtrlTMFReq tmf;
|
||||
VirtIOSCSICtrlANReq an;
|
||||
} req;
|
||||
} VirtIOSCSIReq;
|
||||
|
||||
QEMU_BUILD_BUG_ON(offsetof(VirtIOSCSIReq, req.cdb) !=
|
||||
offsetof(VirtIOSCSIReq, req.cmd) + sizeof(VirtIOSCSICmdReq));
|
||||
|
||||
static inline int virtio_scsi_get_lun(uint8_t *lun)
|
||||
{
|
||||
@@ -38,29 +65,26 @@ static inline SCSIDevice *virtio_scsi_device_find(VirtIOSCSI *s, uint8_t *lun)
|
||||
return scsi_device_find(&s->bus, 0, lun[1], virtio_scsi_get_lun(lun));
|
||||
}
|
||||
|
||||
VirtIOSCSIReq *virtio_scsi_init_req(VirtIOSCSI *s, VirtQueue *vq)
|
||||
static VirtIOSCSIReq *virtio_scsi_init_req(VirtIOSCSI *s, VirtQueue *vq)
|
||||
{
|
||||
VirtIOSCSIReq *req;
|
||||
VirtIOSCSICommon *vs = (VirtIOSCSICommon *)s;
|
||||
const size_t zero_skip = offsetof(VirtIOSCSIReq, elem)
|
||||
+ sizeof(VirtQueueElement);
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(s);
|
||||
|
||||
req = g_malloc0(sizeof(*req) + vs->cdb_size);
|
||||
|
||||
req = g_slice_alloc(sizeof(*req) + vs->cdb_size);
|
||||
req->vq = vq;
|
||||
req->dev = s;
|
||||
req->sreq = NULL;
|
||||
qemu_sglist_init(&req->qsgl, DEVICE(s), 8, &address_space_memory);
|
||||
qemu_iovec_init(&req->resp_iov, 1);
|
||||
memset((uint8_t *)req + zero_skip, 0, sizeof(*req) - zero_skip);
|
||||
return req;
|
||||
}
|
||||
|
||||
void virtio_scsi_free_req(VirtIOSCSIReq *req)
|
||||
static void virtio_scsi_free_req(VirtIOSCSIReq *req)
|
||||
{
|
||||
VirtIOSCSICommon *vs = (VirtIOSCSICommon *)req->dev;
|
||||
|
||||
qemu_iovec_destroy(&req->resp_iov);
|
||||
qemu_sglist_destroy(&req->qsgl);
|
||||
g_slice_free1(sizeof(*req) + vs->cdb_size, req);
|
||||
g_free(req);
|
||||
}
|
||||
|
||||
static void virtio_scsi_complete_req(VirtIOSCSIReq *req)
|
||||
@@ -70,19 +94,13 @@ static void virtio_scsi_complete_req(VirtIOSCSIReq *req)
|
||||
VirtIODevice *vdev = VIRTIO_DEVICE(s);
|
||||
|
||||
qemu_iovec_from_buf(&req->resp_iov, 0, &req->resp, req->resp_size);
|
||||
if (req->vring) {
|
||||
assert(req->vq == NULL);
|
||||
virtio_scsi_vring_push_notify(req);
|
||||
} else {
|
||||
virtqueue_push(vq, &req->elem, req->qsgl.size + req->resp_iov.size);
|
||||
virtio_notify(vdev, vq);
|
||||
}
|
||||
|
||||
virtqueue_push(vq, &req->elem, req->qsgl.size + req->resp_iov.size);
|
||||
if (req->sreq) {
|
||||
req->sreq->hba_private = NULL;
|
||||
scsi_req_unref(req->sreq);
|
||||
}
|
||||
virtio_scsi_free_req(req);
|
||||
virtio_notify(vdev, vq);
|
||||
}
|
||||
|
||||
static void virtio_scsi_bad_req(void)
|
||||
@@ -208,39 +226,13 @@ static void *virtio_scsi_load_request(QEMUFile *f, SCSIRequest *sreq)
|
||||
return req;
|
||||
}
|
||||
|
||||
typedef struct {
|
||||
Notifier notifier;
|
||||
VirtIOSCSIReq *tmf_req;
|
||||
} VirtIOSCSICancelNotifier;
|
||||
|
||||
static void virtio_scsi_cancel_notify(Notifier *notifier, void *data)
|
||||
{
|
||||
VirtIOSCSICancelNotifier *n = container_of(notifier,
|
||||
VirtIOSCSICancelNotifier,
|
||||
notifier);
|
||||
|
||||
if (--n->tmf_req->remaining == 0) {
|
||||
virtio_scsi_complete_req(n->tmf_req);
|
||||
}
|
||||
g_slice_free(VirtIOSCSICancelNotifier, n);
|
||||
}
|
||||
|
||||
/* Return 0 if the request is ready to be completed and return to guest;
|
||||
* -EINPROGRESS if the request is submitted and will be completed later, in the
|
||||
* case of async cancellation. */
|
||||
static int virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
static void virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
{
|
||||
SCSIDevice *d = virtio_scsi_device_find(s, req->req.tmf.lun);
|
||||
SCSIRequest *r, *next;
|
||||
BusChild *kid;
|
||||
int target;
|
||||
int ret = 0;
|
||||
|
||||
if (s->dataplane_started && bdrv_get_aio_context(d->conf.bs) != s->ctx) {
|
||||
aio_context_acquire(s->ctx);
|
||||
bdrv_set_aio_context(d->conf.bs, s->ctx);
|
||||
aio_context_release(s->ctx);
|
||||
}
|
||||
/* Here VIRTIO_SCSI_S_OK means "FUNCTION COMPLETE". */
|
||||
req->resp.tmf.response = VIRTIO_SCSI_S_OK;
|
||||
|
||||
@@ -272,14 +264,7 @@ static int virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
*/
|
||||
req->resp.tmf.response = VIRTIO_SCSI_S_FUNCTION_SUCCEEDED;
|
||||
} else {
|
||||
VirtIOSCSICancelNotifier *notifier;
|
||||
|
||||
req->remaining = 1;
|
||||
notifier = g_slice_new(VirtIOSCSICancelNotifier);
|
||||
notifier->tmf_req = req;
|
||||
notifier->notifier.notify = virtio_scsi_cancel_notify;
|
||||
scsi_req_cancel_async(r, ¬ifier->notifier);
|
||||
ret = -EINPROGRESS;
|
||||
scsi_req_cancel(r);
|
||||
}
|
||||
}
|
||||
break;
|
||||
@@ -305,13 +290,6 @@ static int virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
if (d->lun != virtio_scsi_get_lun(req->req.tmf.lun)) {
|
||||
goto incorrect_lun;
|
||||
}
|
||||
|
||||
/* Add 1 to "remaining" until virtio_scsi_do_tmf returns.
|
||||
* This way, if the bus starts calling back to the notifiers
|
||||
* even before we finish the loop, virtio_scsi_cancel_notify
|
||||
* will not complete the TMF too early.
|
||||
*/
|
||||
req->remaining = 1;
|
||||
QTAILQ_FOREACH_SAFE(r, &d->requests, next, next) {
|
||||
if (r->hba_private) {
|
||||
if (req->req.tmf.subtype == VIRTIO_SCSI_T_TMF_QUERY_TASK_SET) {
|
||||
@@ -321,19 +299,10 @@ static int virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
req->resp.tmf.response = VIRTIO_SCSI_S_FUNCTION_SUCCEEDED;
|
||||
break;
|
||||
} else {
|
||||
VirtIOSCSICancelNotifier *notifier;
|
||||
|
||||
req->remaining++;
|
||||
notifier = g_slice_new(VirtIOSCSICancelNotifier);
|
||||
notifier->notifier.notify = virtio_scsi_cancel_notify;
|
||||
notifier->tmf_req = req;
|
||||
scsi_req_cancel_async(r, ¬ifier->notifier);
|
||||
scsi_req_cancel(r);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (--req->remaining > 0) {
|
||||
ret = -EINPROGRESS;
|
||||
}
|
||||
break;
|
||||
|
||||
case VIRTIO_SCSI_T_TMF_I_T_NEXUS_RESET:
|
||||
@@ -354,53 +323,14 @@ static int virtio_scsi_do_tmf(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
break;
|
||||
}
|
||||
|
||||
return ret;
|
||||
return;
|
||||
|
||||
incorrect_lun:
|
||||
req->resp.tmf.response = VIRTIO_SCSI_S_INCORRECT_LUN;
|
||||
return ret;
|
||||
return;
|
||||
|
||||
fail:
|
||||
req->resp.tmf.response = VIRTIO_SCSI_S_BAD_TARGET;
|
||||
return ret;
|
||||
}
|
||||
|
||||
void virtio_scsi_handle_ctrl_req(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
{
|
||||
VirtIODevice *vdev = (VirtIODevice *)s;
|
||||
int type;
|
||||
int r = 0;
|
||||
|
||||
if (iov_to_buf(req->elem.out_sg, req->elem.out_num, 0,
|
||||
&type, sizeof(type)) < sizeof(type)) {
|
||||
virtio_scsi_bad_req();
|
||||
return;
|
||||
}
|
||||
|
||||
virtio_tswap32s(vdev, &req->req.tmf.type);
|
||||
if (req->req.tmf.type == VIRTIO_SCSI_T_TMF) {
|
||||
if (virtio_scsi_parse_req(req, sizeof(VirtIOSCSICtrlTMFReq),
|
||||
sizeof(VirtIOSCSICtrlTMFResp)) < 0) {
|
||||
virtio_scsi_bad_req();
|
||||
} else {
|
||||
r = virtio_scsi_do_tmf(s, req);
|
||||
}
|
||||
|
||||
} else if (req->req.tmf.type == VIRTIO_SCSI_T_AN_QUERY ||
|
||||
req->req.tmf.type == VIRTIO_SCSI_T_AN_SUBSCRIBE) {
|
||||
if (virtio_scsi_parse_req(req, sizeof(VirtIOSCSICtrlANReq),
|
||||
sizeof(VirtIOSCSICtrlANResp)) < 0) {
|
||||
virtio_scsi_bad_req();
|
||||
} else {
|
||||
req->resp.an.event_actual = 0;
|
||||
req->resp.an.response = VIRTIO_SCSI_S_OK;
|
||||
}
|
||||
}
|
||||
if (r == 0) {
|
||||
virtio_scsi_complete_req(req);
|
||||
} else {
|
||||
assert(r == -EINPROGRESS);
|
||||
}
|
||||
}
|
||||
|
||||
static void virtio_scsi_handle_ctrl(VirtIODevice *vdev, VirtQueue *vq)
|
||||
@@ -408,12 +338,35 @@ static void virtio_scsi_handle_ctrl(VirtIODevice *vdev, VirtQueue *vq)
|
||||
VirtIOSCSI *s = (VirtIOSCSI *)vdev;
|
||||
VirtIOSCSIReq *req;
|
||||
|
||||
if (s->ctx && !s->dataplane_disabled) {
|
||||
virtio_scsi_dataplane_start(s);
|
||||
return;
|
||||
}
|
||||
while ((req = virtio_scsi_pop_req(s, vq))) {
|
||||
virtio_scsi_handle_ctrl_req(s, req);
|
||||
int type;
|
||||
|
||||
if (iov_to_buf(req->elem.out_sg, req->elem.out_num, 0,
|
||||
&type, sizeof(type)) < sizeof(type)) {
|
||||
virtio_scsi_bad_req();
|
||||
continue;
|
||||
}
|
||||
|
||||
virtio_tswap32s(vdev, &req->req.tmf.type);
|
||||
if (req->req.tmf.type == VIRTIO_SCSI_T_TMF) {
|
||||
if (virtio_scsi_parse_req(req, sizeof(VirtIOSCSICtrlTMFReq),
|
||||
sizeof(VirtIOSCSICtrlTMFResp)) < 0) {
|
||||
virtio_scsi_bad_req();
|
||||
} else {
|
||||
virtio_scsi_do_tmf(s, req);
|
||||
}
|
||||
|
||||
} else if (req->req.tmf.type == VIRTIO_SCSI_T_AN_QUERY ||
|
||||
req->req.tmf.type == VIRTIO_SCSI_T_AN_SUBSCRIBE) {
|
||||
if (virtio_scsi_parse_req(req, sizeof(VirtIOSCSICtrlANReq),
|
||||
sizeof(VirtIOSCSICtrlANResp)) < 0) {
|
||||
virtio_scsi_bad_req();
|
||||
} else {
|
||||
req->resp.an.event_actual = 0;
|
||||
req->resp.an.response = VIRTIO_SCSI_S_OK;
|
||||
}
|
||||
}
|
||||
virtio_scsi_complete_req(req);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -467,7 +420,13 @@ static int virtio_scsi_parse_cdb(SCSIDevice *dev, SCSICommand *cmd,
|
||||
* host device passthrough.
|
||||
*/
|
||||
cmd->xfer = req->qsgl.size;
|
||||
cmd->mode = req->mode;
|
||||
if (cmd->xfer == 0) {
|
||||
cmd->mode = SCSI_XFER_NONE;
|
||||
} else if (iov_size(req->elem.in_sg, req->elem.in_num) > req->resp_size) {
|
||||
cmd->mode = SCSI_XFER_FROM_DEV;
|
||||
} else {
|
||||
cmd->mode = SCSI_XFER_TO_DEV;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -499,78 +458,52 @@ static void virtio_scsi_fail_cmd_req(VirtIOSCSIReq *req)
|
||||
virtio_scsi_complete_cmd_req(req);
|
||||
}
|
||||
|
||||
bool virtio_scsi_handle_cmd_req_prepare(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
{
|
||||
VirtIOSCSICommon *vs = &s->parent_obj;
|
||||
SCSIDevice *d;
|
||||
int rc;
|
||||
|
||||
rc = virtio_scsi_parse_req(req, sizeof(VirtIOSCSICmdReq) + vs->cdb_size,
|
||||
sizeof(VirtIOSCSICmdResp) + vs->sense_size);
|
||||
if (rc < 0) {
|
||||
if (rc == -ENOTSUP) {
|
||||
virtio_scsi_fail_cmd_req(req);
|
||||
} else {
|
||||
virtio_scsi_bad_req();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
d = virtio_scsi_device_find(s, req->req.cmd.lun);
|
||||
if (!d) {
|
||||
req->resp.cmd.response = VIRTIO_SCSI_S_BAD_TARGET;
|
||||
virtio_scsi_complete_cmd_req(req);
|
||||
return false;
|
||||
}
|
||||
if (s->dataplane_started && bdrv_get_aio_context(d->conf.bs) != s->ctx) {
|
||||
aio_context_acquire(s->ctx);
|
||||
bdrv_set_aio_context(d->conf.bs, s->ctx);
|
||||
aio_context_release(s->ctx);
|
||||
}
|
||||
req->sreq = scsi_req_new(d, req->req.cmd.tag,
|
||||
virtio_scsi_get_lun(req->req.cmd.lun),
|
||||
req->req.cdb, req);
|
||||
|
||||
if (req->sreq->cmd.mode != SCSI_XFER_NONE
|
||||
&& (req->sreq->cmd.mode != req->mode ||
|
||||
req->sreq->cmd.xfer > req->qsgl.size)) {
|
||||
req->resp.cmd.response = VIRTIO_SCSI_S_OVERRUN;
|
||||
virtio_scsi_complete_cmd_req(req);
|
||||
return false;
|
||||
}
|
||||
scsi_req_ref(req->sreq);
|
||||
bdrv_io_plug(d->conf.bs);
|
||||
return true;
|
||||
}
|
||||
|
||||
void virtio_scsi_handle_cmd_req_submit(VirtIOSCSI *s, VirtIOSCSIReq *req)
|
||||
{
|
||||
if (scsi_req_enqueue(req->sreq)) {
|
||||
scsi_req_continue(req->sreq);
|
||||
}
|
||||
bdrv_io_unplug(req->sreq->dev->conf.bs);
|
||||
scsi_req_unref(req->sreq);
|
||||
}
|
||||
|
||||
static void virtio_scsi_handle_cmd(VirtIODevice *vdev, VirtQueue *vq)
|
||||
{
|
||||
/* use non-QOM casts in the data path */
|
||||
VirtIOSCSI *s = (VirtIOSCSI *)vdev;
|
||||
VirtIOSCSIReq *req, *next;
|
||||
QTAILQ_HEAD(, VirtIOSCSIReq) reqs = QTAILQ_HEAD_INITIALIZER(reqs);
|
||||
VirtIOSCSICommon *vs = &s->parent_obj;
|
||||
|
||||
VirtIOSCSIReq *req;
|
||||
int n;
|
||||
|
||||
if (s->ctx && !s->dataplane_disabled) {
|
||||
virtio_scsi_dataplane_start(s);
|
||||
return;
|
||||
}
|
||||
while ((req = virtio_scsi_pop_req(s, vq))) {
|
||||
if (virtio_scsi_handle_cmd_req_prepare(s, req)) {
|
||||
QTAILQ_INSERT_TAIL(&reqs, req, next);
|
||||
}
|
||||
}
|
||||
SCSIDevice *d;
|
||||
int rc;
|
||||
|
||||
QTAILQ_FOREACH_SAFE(req, &reqs, next, next) {
|
||||
virtio_scsi_handle_cmd_req_submit(s, req);
|
||||
rc = virtio_scsi_parse_req(req, sizeof(VirtIOSCSICmdReq) + vs->cdb_size,
|
||||
sizeof(VirtIOSCSICmdResp) + vs->sense_size);
|
||||
if (rc < 0) {
|
||||
if (rc == -ENOTSUP) {
|
||||
virtio_scsi_fail_cmd_req(req);
|
||||
} else {
|
||||
virtio_scsi_bad_req();
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
d = virtio_scsi_device_find(s, req->req.cmd.lun);
|
||||
if (!d) {
|
||||
req->resp.cmd.response = VIRTIO_SCSI_S_BAD_TARGET;
|
||||
virtio_scsi_complete_cmd_req(req);
|
||||
continue;
|
||||
}
|
||||
req->sreq = scsi_req_new(d, req->req.cmd.tag,
|
||||
virtio_scsi_get_lun(req->req.cmd.lun),
|
||||
req->req.cdb, req);
|
||||
|
||||
if (req->sreq->cmd.mode != SCSI_XFER_NONE
|
||||
&& (req->sreq->cmd.mode != req->mode ||
|
||||
req->sreq->cmd.xfer > req->qsgl.size)) {
|
||||
req->resp.cmd.response = VIRTIO_SCSI_S_OVERRUN;
|
||||
virtio_scsi_complete_cmd_req(req);
|
||||
continue;
|
||||
}
|
||||
|
||||
n = scsi_req_enqueue(req->sreq);
|
||||
if (n) {
|
||||
scsi_req_continue(req->sreq);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -619,9 +552,6 @@ static void virtio_scsi_reset(VirtIODevice *vdev)
|
||||
VirtIOSCSI *s = VIRTIO_SCSI(vdev);
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(vdev);
|
||||
|
||||
if (s->ctx) {
|
||||
virtio_scsi_dataplane_stop(s);
|
||||
}
|
||||
s->resetting++;
|
||||
qbus_reset_all(&s->bus.qbus);
|
||||
s->resetting--;
|
||||
@@ -652,8 +582,8 @@ static int virtio_scsi_load(QEMUFile *f, void *opaque, int version_id)
|
||||
return 0;
|
||||
}
|
||||
|
||||
void virtio_scsi_push_event(VirtIOSCSI *s, SCSIDevice *dev,
|
||||
uint32_t event, uint32_t reason)
|
||||
static void virtio_scsi_push_event(VirtIOSCSI *s, SCSIDevice *dev,
|
||||
uint32_t event, uint32_t reason)
|
||||
{
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(s);
|
||||
VirtIOSCSIReq *req;
|
||||
@@ -664,19 +594,10 @@ void virtio_scsi_push_event(VirtIOSCSI *s, SCSIDevice *dev,
|
||||
return;
|
||||
}
|
||||
|
||||
if (s->dataplane_started) {
|
||||
assert(s->ctx);
|
||||
aio_context_acquire(s->ctx);
|
||||
}
|
||||
|
||||
if (s->dataplane_started) {
|
||||
req = virtio_scsi_pop_req_vring(s, s->event_vring);
|
||||
} else {
|
||||
req = virtio_scsi_pop_req(s, vs->event_vq);
|
||||
}
|
||||
req = virtio_scsi_pop_req(s, vs->event_vq);
|
||||
if (!req) {
|
||||
s->events_dropped = true;
|
||||
goto out;
|
||||
return;
|
||||
}
|
||||
|
||||
if (s->events_dropped) {
|
||||
@@ -705,20 +626,12 @@ void virtio_scsi_push_event(VirtIOSCSI *s, SCSIDevice *dev,
|
||||
evt->lun[3] = dev->lun & 0xFF;
|
||||
}
|
||||
virtio_scsi_complete_req(req);
|
||||
out:
|
||||
if (s->dataplane_started) {
|
||||
aio_context_release(s->ctx);
|
||||
}
|
||||
}
|
||||
|
||||
static void virtio_scsi_handle_event(VirtIODevice *vdev, VirtQueue *vq)
|
||||
{
|
||||
VirtIOSCSI *s = VIRTIO_SCSI(vdev);
|
||||
|
||||
if (s->ctx && !s->dataplane_disabled) {
|
||||
virtio_scsi_dataplane_start(s);
|
||||
return;
|
||||
}
|
||||
if (s->events_dropped) {
|
||||
virtio_scsi_push_event(s, NULL, VIRTIO_SCSI_T_NO_EVENT, 0);
|
||||
}
|
||||
@@ -804,35 +717,6 @@ void virtio_scsi_common_realize(DeviceState *dev, Error **errp,
|
||||
s->cmd_vqs[i] = virtio_add_queue(vdev, VIRTIO_SCSI_VQ_SIZE,
|
||||
cmd);
|
||||
}
|
||||
|
||||
if (s->conf.iothread) {
|
||||
virtio_scsi_set_iothread(VIRTIO_SCSI(s), s->conf.iothread);
|
||||
}
|
||||
}
|
||||
|
||||
/* Disable dataplane thread during live migration since it does not
|
||||
* update the dirty memory bitmap yet.
|
||||
*/
|
||||
static void virtio_scsi_migration_state_changed(Notifier *notifier, void *data)
|
||||
{
|
||||
VirtIOSCSI *s = container_of(notifier, VirtIOSCSI,
|
||||
migration_state_notifier);
|
||||
MigrationState *mig = data;
|
||||
|
||||
if (migration_in_setup(mig)) {
|
||||
if (!s->dataplane_started) {
|
||||
return;
|
||||
}
|
||||
virtio_scsi_dataplane_stop(s);
|
||||
s->dataplane_disabled = true;
|
||||
} else if (migration_has_finished(mig) ||
|
||||
migration_has_failed(mig)) {
|
||||
if (s->dataplane_started) {
|
||||
return;
|
||||
}
|
||||
bdrv_drain_all(); /* complete in-flight non-dataplane requests */
|
||||
s->dataplane_disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
static void virtio_scsi_device_realize(DeviceState *dev, Error **errp)
|
||||
@@ -863,18 +747,6 @@ static void virtio_scsi_device_realize(DeviceState *dev, Error **errp)
|
||||
|
||||
register_savevm(dev, "virtio-scsi", virtio_scsi_id++, 1,
|
||||
virtio_scsi_save, virtio_scsi_load, s);
|
||||
s->migration_state_notifier.notify = virtio_scsi_migration_state_changed;
|
||||
add_migration_state_change_notifier(&s->migration_state_notifier);
|
||||
}
|
||||
|
||||
static void virtio_scsi_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSCSICommon *vs = VIRTIO_SCSI_COMMON(obj);
|
||||
|
||||
object_property_add_link(obj, "iothread", TYPE_IOTHREAD,
|
||||
(Object **)&vs->conf.iothread,
|
||||
qdev_prop_allow_set_link_before_realize,
|
||||
OBJ_PROP_LINK_UNREF_ON_RELEASE, &error_abort);
|
||||
}
|
||||
|
||||
void virtio_scsi_common_unrealize(DeviceState *dev, Error **errp)
|
||||
@@ -891,7 +763,6 @@ static void virtio_scsi_device_unrealize(DeviceState *dev, Error **errp)
|
||||
VirtIOSCSI *s = VIRTIO_SCSI(dev);
|
||||
|
||||
unregister_savevm(dev, "virtio-scsi", s);
|
||||
remove_migration_state_change_notifier(&s->migration_state_notifier);
|
||||
|
||||
virtio_scsi_common_unrealize(dev, errp);
|
||||
}
|
||||
@@ -936,7 +807,6 @@ static const TypeInfo virtio_scsi_info = {
|
||||
.name = TYPE_VIRTIO_SCSI,
|
||||
.parent = TYPE_VIRTIO_SCSI_COMMON,
|
||||
.instance_size = sizeof(VirtIOSCSI),
|
||||
.instance_init = virtio_scsi_instance_init,
|
||||
.class_init = virtio_scsi_class_init,
|
||||
};
|
||||
|
||||
|
@@ -527,7 +527,7 @@ static void apc_init(hwaddr power_base, qemu_irq cpu_halt)
|
||||
sysbus_connect_irq(s, 0, cpu_halt);
|
||||
}
|
||||
|
||||
static void tcx_init(hwaddr addr, int vram_size, int width,
|
||||
static void tcx_init(hwaddr addr, qemu_irq irq, int vram_size, int width,
|
||||
int height, int depth)
|
||||
{
|
||||
DeviceState *dev;
|
||||
@@ -541,25 +541,43 @@ static void tcx_init(hwaddr addr, int vram_size, int width,
|
||||
qdev_prop_set_uint64(dev, "prom_addr", addr);
|
||||
qdev_init_nofail(dev);
|
||||
s = SYS_BUS_DEVICE(dev);
|
||||
/* FCode ROM */
|
||||
|
||||
/* 10/ROM : FCode ROM */
|
||||
sysbus_mmio_map(s, 0, addr);
|
||||
/* DAC */
|
||||
sysbus_mmio_map(s, 1, addr + 0x00200000ULL);
|
||||
/* TEC (dummy) */
|
||||
sysbus_mmio_map(s, 2, addr + 0x00700000ULL);
|
||||
/* THC 24 bit: NetBSD writes here even with 8-bit display: dummy */
|
||||
sysbus_mmio_map(s, 3, addr + 0x00301000ULL);
|
||||
/* 8-bit plane */
|
||||
sysbus_mmio_map(s, 4, addr + 0x00800000ULL);
|
||||
if (depth == 24) {
|
||||
/* 24-bit plane */
|
||||
sysbus_mmio_map(s, 5, addr + 0x02000000ULL);
|
||||
/* Control plane */
|
||||
sysbus_mmio_map(s, 6, addr + 0x0a000000ULL);
|
||||
/* 2/STIP : Stipple */
|
||||
sysbus_mmio_map(s, 1, addr + 0x04000000ULL);
|
||||
/* 3/BLIT : Blitter */
|
||||
sysbus_mmio_map(s, 2, addr + 0x06000000ULL);
|
||||
/* 5/RSTIP : Raw Stipple */
|
||||
sysbus_mmio_map(s, 3, addr + 0x0c000000ULL);
|
||||
/* 6/RBLIT : Raw Blitter */
|
||||
sysbus_mmio_map(s, 4, addr + 0x0e000000ULL);
|
||||
/* 7/TEC : Transform Engine */
|
||||
sysbus_mmio_map(s, 5, addr + 0x00700000ULL);
|
||||
/* 8/CMAP : DAC */
|
||||
sysbus_mmio_map(s, 6, addr + 0x00200000ULL);
|
||||
/* 9/THC : */
|
||||
if (depth == 8) {
|
||||
sysbus_mmio_map(s, 7, addr + 0x00300000ULL);
|
||||
} else {
|
||||
/* THC 8 bit (dummy) */
|
||||
sysbus_mmio_map(s, 5, addr + 0x00300000ULL);
|
||||
sysbus_mmio_map(s, 7, addr + 0x00301000ULL);
|
||||
}
|
||||
/* 11/DHC : */
|
||||
sysbus_mmio_map(s, 8, addr + 0x00240000ULL);
|
||||
/* 12/ALT : */
|
||||
sysbus_mmio_map(s, 9, addr + 0x00280000ULL);
|
||||
/* 0/DFB8 : 8-bit plane */
|
||||
sysbus_mmio_map(s, 10, addr + 0x00800000ULL);
|
||||
/* 1/DFB24 : 24bit plane */
|
||||
sysbus_mmio_map(s, 11, addr + 0x02000000ULL);
|
||||
/* 4/RDFB32: Raw framebuffer. Control plane */
|
||||
sysbus_mmio_map(s, 12, addr + 0x0a000000ULL);
|
||||
/* 9/THC24bits : NetBSD writes here even with 8-bit display: dummy */
|
||||
if (depth == 8) {
|
||||
sysbus_mmio_map(s, 13, addr + 0x00301000ULL);
|
||||
}
|
||||
|
||||
sysbus_connect_irq(s, 0, irq);
|
||||
}
|
||||
|
||||
static void cg3_init(hwaddr addr, qemu_irq irq, int vram_size, int width,
|
||||
@@ -976,8 +994,8 @@ static void sun4m_hw_init(const struct sun4m_hwdef *hwdef,
|
||||
exit(1);
|
||||
}
|
||||
|
||||
tcx_init(hwdef->tcx_base, 0x00100000, graphic_width, graphic_height,
|
||||
graphic_depth);
|
||||
tcx_init(hwdef->tcx_base, slavio_irq[11], 0x00100000,
|
||||
graphic_width, graphic_height, graphic_depth);
|
||||
}
|
||||
}
|
||||
|
||||
|
91
hw/usb/bus.c
91
hw/usb/bus.c
@@ -9,7 +9,7 @@ static void usb_bus_dev_print(Monitor *mon, DeviceState *qdev, int indent);
|
||||
|
||||
static char *usb_get_dev_path(DeviceState *dev);
|
||||
static char *usb_get_fw_dev_path(DeviceState *qdev);
|
||||
static int usb_qdev_exit(DeviceState *qdev);
|
||||
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp);
|
||||
|
||||
static Property usb_props[] = {
|
||||
DEFINE_PROP_STRING("port", USBDevice, port_path),
|
||||
@@ -107,13 +107,13 @@ USBBus *usb_bus_find(int busnr)
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int usb_device_init(USBDevice *dev)
|
||||
static void usb_device_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBDeviceClass *klass = USB_DEVICE_GET_CLASS(dev);
|
||||
if (klass->init) {
|
||||
return klass->init(dev);
|
||||
|
||||
if (klass->realize) {
|
||||
klass->realize(dev, errp);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
USBDevice *usb_device_find_device(USBDevice *dev, uint8_t addr)
|
||||
@@ -232,36 +232,41 @@ void usb_device_free_streams(USBDevice *dev, USBEndpoint **eps, int nr_eps)
|
||||
}
|
||||
}
|
||||
|
||||
static int usb_qdev_init(DeviceState *qdev)
|
||||
static void usb_qdev_realize(DeviceState *qdev, Error **errp)
|
||||
{
|
||||
USBDevice *dev = USB_DEVICE(qdev);
|
||||
int rc;
|
||||
Error *local_err = NULL;
|
||||
|
||||
pstrcpy(dev->product_desc, sizeof(dev->product_desc),
|
||||
usb_device_get_product_desc(dev));
|
||||
dev->auto_attach = 1;
|
||||
QLIST_INIT(&dev->strings);
|
||||
usb_ep_init(dev);
|
||||
rc = usb_claim_port(dev);
|
||||
if (rc != 0) {
|
||||
return rc;
|
||||
|
||||
usb_claim_port(dev, &local_err);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
return;
|
||||
}
|
||||
rc = usb_device_init(dev);
|
||||
if (rc != 0) {
|
||||
|
||||
usb_device_realize(dev, &local_err);
|
||||
if (local_err) {
|
||||
usb_release_port(dev);
|
||||
return rc;
|
||||
error_propagate(errp, local_err);
|
||||
return;
|
||||
}
|
||||
|
||||
if (dev->auto_attach) {
|
||||
rc = usb_device_attach(dev);
|
||||
if (rc != 0) {
|
||||
usb_qdev_exit(qdev);
|
||||
return rc;
|
||||
usb_device_attach(dev, &local_err);
|
||||
if (local_err) {
|
||||
usb_qdev_unrealize(qdev, NULL);
|
||||
error_propagate(errp, local_err);
|
||||
return;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int usb_qdev_exit(DeviceState *qdev)
|
||||
static void usb_qdev_unrealize(DeviceState *qdev, Error **errp)
|
||||
{
|
||||
USBDevice *dev = USB_DEVICE(qdev);
|
||||
|
||||
@@ -272,7 +277,6 @@ static int usb_qdev_exit(DeviceState *qdev)
|
||||
if (dev->port) {
|
||||
usb_release_port(dev);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
typedef struct LegacyUSBFactory
|
||||
@@ -392,7 +396,7 @@ void usb_unregister_port(USBBus *bus, USBPort *port)
|
||||
bus->nfree--;
|
||||
}
|
||||
|
||||
int usb_claim_port(USBDevice *dev)
|
||||
void usb_claim_port(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBBus *bus = usb_bus_from_device(dev);
|
||||
USBPort *port;
|
||||
@@ -406,9 +410,9 @@ int usb_claim_port(USBDevice *dev)
|
||||
}
|
||||
}
|
||||
if (port == NULL) {
|
||||
error_report("Error: usb port %s (bus %s) not found (in use?)",
|
||||
dev->port_path, bus->qbus.name);
|
||||
return -1;
|
||||
error_setg(errp, "Error: usb port %s (bus %s) not found (in use?)",
|
||||
dev->port_path, bus->qbus.name);
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
if (bus->nfree == 1 && strcmp(object_get_typename(OBJECT(dev)), "usb-hub") != 0) {
|
||||
@@ -416,9 +420,9 @@ int usb_claim_port(USBDevice *dev)
|
||||
usb_create_simple(bus, "usb-hub");
|
||||
}
|
||||
if (bus->nfree == 0) {
|
||||
error_report("Error: tried to attach usb device %s to a bus "
|
||||
"with no free ports", dev->product_desc);
|
||||
return -1;
|
||||
error_setg(errp, "Error: tried to attach usb device %s to a bus "
|
||||
"with no free ports", dev->product_desc);
|
||||
return;
|
||||
}
|
||||
port = QTAILQ_FIRST(&bus->free);
|
||||
}
|
||||
@@ -432,7 +436,6 @@ int usb_claim_port(USBDevice *dev)
|
||||
|
||||
QTAILQ_INSERT_TAIL(&bus->used, port, next);
|
||||
bus->nused++;
|
||||
return 0;
|
||||
}
|
||||
|
||||
void usb_release_port(USBDevice *dev)
|
||||
@@ -475,7 +478,7 @@ static void usb_mask_to_str(char *dest, size_t size,
|
||||
}
|
||||
}
|
||||
|
||||
int usb_device_attach(USBDevice *dev)
|
||||
void usb_check_attach(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBBus *bus = usb_bus_from_device(dev);
|
||||
USBPort *port = dev->port;
|
||||
@@ -489,18 +492,28 @@ int usb_device_attach(USBDevice *dev)
|
||||
devspeed, portspeed);
|
||||
|
||||
if (!(port->speedmask & dev->speedmask)) {
|
||||
error_report("Warning: speed mismatch trying to attach"
|
||||
" usb device \"%s\" (%s speed)"
|
||||
" to bus \"%s\", port \"%s\" (%s speed)",
|
||||
dev->product_desc, devspeed,
|
||||
bus->qbus.name, port->path, portspeed);
|
||||
return -1;
|
||||
error_setg(errp, "Warning: speed mismatch trying to attach"
|
||||
" usb device \"%s\" (%s speed)"
|
||||
" to bus \"%s\", port \"%s\" (%s speed)",
|
||||
dev->product_desc, devspeed,
|
||||
bus->qbus.name, port->path, portspeed);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
void usb_device_attach(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBPort *port = dev->port;
|
||||
Error *local_err = NULL;
|
||||
|
||||
usb_check_attach(dev, &local_err);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
return;
|
||||
}
|
||||
|
||||
dev->attached++;
|
||||
usb_attach(port);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int usb_device_detach(USBDevice *dev)
|
||||
@@ -688,9 +701,9 @@ static void usb_device_class_init(ObjectClass *klass, void *data)
|
||||
{
|
||||
DeviceClass *k = DEVICE_CLASS(klass);
|
||||
k->bus_type = TYPE_USB_BUS;
|
||||
k->init = usb_qdev_init;
|
||||
k->unplug = qdev_simple_unplug_cb;
|
||||
k->exit = usb_qdev_exit;
|
||||
k->realize = usb_qdev_realize;
|
||||
k->unrealize = usb_qdev_unrealize;
|
||||
k->props = usb_props;
|
||||
}
|
||||
|
||||
|
@@ -628,7 +628,7 @@ static void usb_audio_handle_destroy(USBDevice *dev)
|
||||
streambuf_fini(&s->out.buf);
|
||||
}
|
||||
|
||||
static int usb_audio_initfn(USBDevice *dev)
|
||||
static void usb_audio_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBAudioState *s = DO_UPCAST(USBAudioState, dev, dev);
|
||||
|
||||
@@ -651,7 +651,6 @@ static int usb_audio_initfn(USBDevice *dev)
|
||||
s, output_callback, &s->out.as);
|
||||
AUD_set_volume_out(s->out.voice, s->out.mute, s->out.vol[0], s->out.vol[1]);
|
||||
AUD_set_active_out(s->out.voice, 0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_usb_audio = {
|
||||
@@ -676,7 +675,7 @@ static void usb_audio_class_init(ObjectClass *klass, void *data)
|
||||
set_bit(DEVICE_CATEGORY_SOUND, dc->categories);
|
||||
k->product_desc = "QEMU USB Audio Interface";
|
||||
k->usb_desc = &desc_audio;
|
||||
k->init = usb_audio_initfn;
|
||||
k->realize = usb_audio_realize;
|
||||
k->handle_reset = usb_audio_handle_reset;
|
||||
k->handle_control = usb_audio_handle_control;
|
||||
k->handle_data = usb_audio_handle_data;
|
||||
|
@@ -501,7 +501,7 @@ static void usb_bt_handle_destroy(USBDevice *dev)
|
||||
s->hci->acl_recv = NULL;
|
||||
}
|
||||
|
||||
static int usb_bt_initfn(USBDevice *dev)
|
||||
static void usb_bt_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
struct USBBtState *s = DO_UPCAST(struct USBBtState, dev, dev);
|
||||
|
||||
@@ -516,8 +516,6 @@ static int usb_bt_initfn(USBDevice *dev)
|
||||
s->hci->acl_recv = usb_bt_out_hci_packet_acl;
|
||||
usb_bt_handle_reset(&s->dev);
|
||||
s->intr = usb_ep_get(dev, USB_TOKEN_IN, USB_EVT_EP);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static USBDevice *usb_bt_init(USBBus *bus, const char *cmdline)
|
||||
@@ -560,7 +558,7 @@ static void usb_bt_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_bt_initfn;
|
||||
uc->realize = usb_bt_realize;
|
||||
uc->product_desc = "QEMU BT dongle";
|
||||
uc->usb_desc = &desc_bluetooth;
|
||||
uc->handle_reset = usb_bt_handle_reset;
|
||||
|
@@ -566,7 +566,7 @@ static void usb_hid_handle_destroy(USBDevice *dev)
|
||||
hid_free(&us->hid);
|
||||
}
|
||||
|
||||
static int usb_hid_initfn(USBDevice *dev, int kind)
|
||||
static void usb_hid_initfn(USBDevice *dev, int kind)
|
||||
{
|
||||
USBHIDState *us = DO_UPCAST(USBHIDState, dev, dev);
|
||||
|
||||
@@ -579,10 +579,9 @@ static int usb_hid_initfn(USBDevice *dev, int kind)
|
||||
if (us->display && us->hid.s) {
|
||||
qemu_input_handler_bind(us->hid.s, us->display, us->head, NULL);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int usb_tablet_initfn(USBDevice *dev)
|
||||
static void usb_tablet_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBHIDState *us = DO_UPCAST(USBHIDState, dev, dev);
|
||||
|
||||
@@ -594,22 +593,22 @@ static int usb_tablet_initfn(USBDevice *dev)
|
||||
dev->usb_desc = &desc_tablet2;
|
||||
break;
|
||||
default:
|
||||
error_report("Invalid usb version %d for usb-tabler (must be 1 or 2)",
|
||||
us->usb_version);
|
||||
return -1;
|
||||
error_setg(errp, "Invalid usb version %d for usb-tablet "
|
||||
"(must be 1 or 2)", us->usb_version);
|
||||
return;
|
||||
}
|
||||
|
||||
return usb_hid_initfn(dev, HID_TABLET);
|
||||
usb_hid_initfn(dev, HID_TABLET);
|
||||
}
|
||||
|
||||
static int usb_mouse_initfn(USBDevice *dev)
|
||||
static void usb_mouse_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
return usb_hid_initfn(dev, HID_MOUSE);
|
||||
usb_hid_initfn(dev, HID_MOUSE);
|
||||
}
|
||||
|
||||
static int usb_keyboard_initfn(USBDevice *dev)
|
||||
static void usb_keyboard_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
return usb_hid_initfn(dev, HID_KEYBOARD);
|
||||
usb_hid_initfn(dev, HID_KEYBOARD);
|
||||
}
|
||||
|
||||
static int usb_ptr_post_load(void *opaque, int version_id)
|
||||
@@ -669,7 +668,7 @@ static void usb_tablet_class_initfn(ObjectClass *klass, void *data)
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
usb_hid_class_initfn(klass, data);
|
||||
uc->init = usb_tablet_initfn;
|
||||
uc->realize = usb_tablet_realize;
|
||||
uc->product_desc = "QEMU USB Tablet";
|
||||
dc->vmsd = &vmstate_usb_ptr;
|
||||
dc->props = usb_tablet_properties;
|
||||
@@ -689,7 +688,7 @@ static void usb_mouse_class_initfn(ObjectClass *klass, void *data)
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
usb_hid_class_initfn(klass, data);
|
||||
uc->init = usb_mouse_initfn;
|
||||
uc->realize = usb_mouse_realize;
|
||||
uc->product_desc = "QEMU USB Mouse";
|
||||
uc->usb_desc = &desc_mouse;
|
||||
dc->vmsd = &vmstate_usb_ptr;
|
||||
@@ -714,7 +713,7 @@ static void usb_keyboard_class_initfn(ObjectClass *klass, void *data)
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
usb_hid_class_initfn(klass, data);
|
||||
uc->init = usb_keyboard_initfn;
|
||||
uc->realize = usb_keyboard_realize;
|
||||
uc->product_desc = "QEMU USB Keyboard";
|
||||
uc->usb_desc = &desc_keyboard;
|
||||
dc->vmsd = &vmstate_usb_kbd;
|
||||
|
@@ -511,15 +511,15 @@ static USBPortOps usb_hub_port_ops = {
|
||||
.complete = usb_hub_complete,
|
||||
};
|
||||
|
||||
static int usb_hub_initfn(USBDevice *dev)
|
||||
static void usb_hub_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBHubState *s = DO_UPCAST(USBHubState, dev, dev);
|
||||
USBHubPort *port;
|
||||
int i;
|
||||
|
||||
if (dev->port->hubcount == 5) {
|
||||
error_report("usb hub chain too deep");
|
||||
return -1;
|
||||
error_setg(errp, "usb hub chain too deep");
|
||||
return;
|
||||
}
|
||||
|
||||
usb_desc_create_serial(dev);
|
||||
@@ -533,7 +533,6 @@ static int usb_hub_initfn(USBDevice *dev)
|
||||
usb_port_location(&port->port, dev->port, i+1);
|
||||
}
|
||||
usb_hub_handle_reset(dev);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_usb_hub_port = {
|
||||
@@ -564,7 +563,7 @@ static void usb_hub_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_hub_initfn;
|
||||
uc->realize = usb_hub_realize;
|
||||
uc->product_desc = "QEMU USB Hub";
|
||||
uc->usb_desc = &desc_hub;
|
||||
uc->find_device = usb_hub_find_device;
|
||||
|
@@ -1060,7 +1060,7 @@ static void usb_mtp_handle_data(USBDevice *dev, USBPacket *p)
|
||||
}
|
||||
}
|
||||
|
||||
static int usb_mtp_initfn(USBDevice *dev)
|
||||
static void usb_mtp_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
MTPState *s = DO_UPCAST(MTPState, dev, dev);
|
||||
|
||||
@@ -1075,7 +1075,6 @@ static int usb_mtp_initfn(USBDevice *dev)
|
||||
s->desc = g_strdup("none");
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_usb_mtp = {
|
||||
@@ -1100,7 +1099,7 @@ static void usb_mtp_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_mtp_initfn;
|
||||
uc->realize = usb_mtp_realize;
|
||||
uc->product_desc = "QEMU USB MTP";
|
||||
uc->usb_desc = &desc;
|
||||
uc->cancel_packet = usb_mtp_cancel_packet;
|
||||
|
@@ -27,7 +27,7 @@
|
||||
#include "hw/usb.h"
|
||||
#include "hw/usb/desc.h"
|
||||
#include "net/net.h"
|
||||
#include "qapi/qmp/qerror.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include "qemu/queue.h"
|
||||
#include "qemu/config-file.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
@@ -1341,7 +1341,7 @@ static NetClientInfo net_usbnet_info = {
|
||||
.cleanup = usbnet_cleanup,
|
||||
};
|
||||
|
||||
static int usb_net_initfn(USBDevice *dev)
|
||||
static void usb_net_realize(USBDevice *dev, Error **errrp)
|
||||
{
|
||||
USBNetState *s = DO_UPCAST(USBNetState, dev, dev);
|
||||
|
||||
@@ -1373,7 +1373,6 @@ static int usb_net_initfn(USBDevice *dev)
|
||||
usb_desc_set_string(dev, STRING_ETHADDR, s->usbstring_mac);
|
||||
|
||||
add_boot_device_path(s->conf.bootindex, &dev->qdev, "/ethernet@0");
|
||||
return 0;
|
||||
}
|
||||
|
||||
static USBDevice *usb_net_init(USBBus *bus, const char *cmdline)
|
||||
@@ -1392,7 +1391,7 @@ static USBDevice *usb_net_init(USBBus *bus, const char *cmdline)
|
||||
|
||||
idx = net_client_init(opts, 0, &local_err);
|
||||
if (local_err) {
|
||||
qerror_report_err(local_err);
|
||||
error_report("%s", error_get_pretty(local_err));
|
||||
error_free(local_err);
|
||||
return NULL;
|
||||
}
|
||||
@@ -1421,7 +1420,7 @@ static void usb_net_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_net_initfn;
|
||||
uc->realize = usb_net_realize;
|
||||
uc->product_desc = "QEMU USB Network Interface";
|
||||
uc->usb_desc = &desc_net;
|
||||
uc->handle_reset = usb_net_handle_reset;
|
||||
|
@@ -9,7 +9,7 @@
|
||||
*/
|
||||
|
||||
#include "qemu-common.h"
|
||||
#include "qemu/error-report.h"
|
||||
#include "monitor/monitor.h"
|
||||
#include "hw/usb.h"
|
||||
#include "hw/usb/desc.h"
|
||||
#include "sysemu/char.h"
|
||||
@@ -460,7 +460,7 @@ static void usb_serial_event(void *opaque, int event)
|
||||
break;
|
||||
case CHR_EVENT_OPENED:
|
||||
if (!s->dev.attached) {
|
||||
usb_device_attach(&s->dev);
|
||||
usb_device_attach(&s->dev, &error_abort);
|
||||
}
|
||||
break;
|
||||
case CHR_EVENT_CLOSED:
|
||||
@@ -471,17 +471,24 @@ static void usb_serial_event(void *opaque, int event)
|
||||
}
|
||||
}
|
||||
|
||||
static int usb_serial_initfn(USBDevice *dev)
|
||||
static void usb_serial_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBSerialState *s = DO_UPCAST(USBSerialState, dev, dev);
|
||||
Error *local_err = NULL;
|
||||
|
||||
usb_desc_create_serial(dev);
|
||||
usb_desc_init(dev);
|
||||
dev->auto_attach = 0;
|
||||
|
||||
if (!s->cs) {
|
||||
error_report("Property chardev is required");
|
||||
return -1;
|
||||
error_setg(errp, "Property chardev is required");
|
||||
return;
|
||||
}
|
||||
|
||||
usb_check_attach(dev, &local_err);
|
||||
if (local_err) {
|
||||
error_propagate(errp, local_err);
|
||||
return;
|
||||
}
|
||||
|
||||
qemu_chr_add_handlers(s->cs, usb_serial_can_read, usb_serial_read,
|
||||
@@ -489,9 +496,8 @@ static int usb_serial_initfn(USBDevice *dev)
|
||||
usb_serial_handle_reset(dev);
|
||||
|
||||
if (s->cs->be_open && !dev->attached) {
|
||||
usb_device_attach(dev);
|
||||
usb_device_attach(dev, &error_abort);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static USBDevice *usb_serial_init(USBBus *bus, const char *filename)
|
||||
@@ -582,7 +588,7 @@ static void usb_serial_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_serial_initfn;
|
||||
uc->realize = usb_serial_realize;
|
||||
uc->product_desc = "QEMU USB Serial";
|
||||
uc->usb_desc = &desc_serial;
|
||||
uc->handle_reset = usb_serial_handle_reset;
|
||||
@@ -610,7 +616,7 @@ static void usb_braille_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_serial_initfn;
|
||||
uc->realize = usb_serial_realize;
|
||||
uc->product_desc = "QEMU USB Braille";
|
||||
uc->usb_desc = &desc_braille;
|
||||
uc->handle_reset = usb_serial_handle_reset;
|
||||
|
@@ -1304,7 +1304,7 @@ static int ccid_card_init(DeviceState *qdev)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ccid_initfn(USBDevice *dev)
|
||||
static void ccid_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBCCIDState *s = DO_UPCAST(USBCCIDState, dev, dev);
|
||||
|
||||
@@ -1332,7 +1332,6 @@ static int ccid_initfn(USBDevice *dev)
|
||||
ccid_reset_parameters(s);
|
||||
ccid_reset(s);
|
||||
s->debug = parse_debug_env("QEMU_CCID_DEBUG", D_VERBOSE, s->debug);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ccid_post_load(void *opaque, int version_id)
|
||||
@@ -1441,7 +1440,7 @@ static void ccid_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = ccid_initfn;
|
||||
uc->realize = ccid_realize;
|
||||
uc->product_desc = "QEMU USB CCID";
|
||||
uc->usb_desc = &desc_ccid;
|
||||
uc->handle_reset = ccid_handle_reset;
|
||||
|
@@ -409,19 +409,19 @@ static void usb_msd_handle_data(USBDevice *dev, USBPacket *p)
|
||||
switch (s->mode) {
|
||||
case USB_MSDM_CBW:
|
||||
if (p->iov.size != 31) {
|
||||
fprintf(stderr, "usb-msd: Bad CBW size");
|
||||
error_report("usb-msd: Bad CBW size");
|
||||
goto fail;
|
||||
}
|
||||
usb_packet_copy(p, &cbw, 31);
|
||||
if (le32_to_cpu(cbw.sig) != 0x43425355) {
|
||||
fprintf(stderr, "usb-msd: Bad signature %08x\n",
|
||||
le32_to_cpu(cbw.sig));
|
||||
error_report("usb-msd: Bad signature %08x",
|
||||
le32_to_cpu(cbw.sig));
|
||||
goto fail;
|
||||
}
|
||||
DPRINTF("Command on LUN %d\n", cbw.lun);
|
||||
scsi_dev = scsi_device_find(&s->bus, 0, 0, cbw.lun);
|
||||
if (scsi_dev == NULL) {
|
||||
fprintf(stderr, "usb-msd: Bad LUN %d\n", cbw.lun);
|
||||
error_report("usb-msd: Bad LUN %d", cbw.lun);
|
||||
goto fail;
|
||||
}
|
||||
tag = le32_to_cpu(cbw.tag);
|
||||
@@ -549,12 +549,17 @@ static void usb_msd_handle_data(USBDevice *dev, USBPacket *p)
|
||||
static void usb_msd_password_cb(void *opaque, int err)
|
||||
{
|
||||
MSDState *s = opaque;
|
||||
Error *local_err = NULL;
|
||||
|
||||
if (!err)
|
||||
err = usb_device_attach(&s->dev);
|
||||
if (!err) {
|
||||
usb_device_attach(&s->dev, &local_err);
|
||||
}
|
||||
|
||||
if (err)
|
||||
if (local_err) {
|
||||
qerror_report_err(local_err);
|
||||
error_free(local_err);
|
||||
qdev_unplug(&s->dev.qdev, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
static void *usb_msd_load_request(QEMUFile *f, SCSIRequest *req)
|
||||
@@ -590,7 +595,7 @@ static const struct SCSIBusInfo usb_msd_scsi_info_bot = {
|
||||
.load_request = usb_msd_load_request,
|
||||
};
|
||||
|
||||
static int usb_msd_initfn_storage(USBDevice *dev)
|
||||
static void usb_msd_realize_storage(USBDevice *dev, Error **errp)
|
||||
{
|
||||
MSDState *s = DO_UPCAST(MSDState, dev, dev);
|
||||
BlockDriverState *bs = s->conf.bs;
|
||||
@@ -598,8 +603,8 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
||||
Error *err = NULL;
|
||||
|
||||
if (!bs) {
|
||||
error_report("drive property not set");
|
||||
return -1;
|
||||
error_setg(errp, "drive property not set");
|
||||
return;
|
||||
}
|
||||
|
||||
blkconf_serial(&s->conf, &dev->serial);
|
||||
@@ -624,7 +629,8 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
||||
s->conf.bootindex, dev->serial,
|
||||
&err);
|
||||
if (!scsi_dev) {
|
||||
return -1;
|
||||
error_propagate(errp, err);
|
||||
return;
|
||||
}
|
||||
s->bus.qbus.allow_hotplug = 0;
|
||||
usb_msd_handle_reset(dev);
|
||||
@@ -637,11 +643,9 @@ static int usb_msd_initfn_storage(USBDevice *dev)
|
||||
autostart = 0;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int usb_msd_initfn_bot(USBDevice *dev)
|
||||
static void usb_msd_realize_bot(USBDevice *dev, Error **errp)
|
||||
{
|
||||
MSDState *s = DO_UPCAST(MSDState, dev, dev);
|
||||
|
||||
@@ -651,8 +655,6 @@ static int usb_msd_initfn_bot(USBDevice *dev)
|
||||
&usb_msd_scsi_info_bot, NULL);
|
||||
s->bus.qbus.allow_hotplug = 0;
|
||||
usb_msd_handle_reset(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static USBDevice *usb_msd_init(USBBus *bus, const char *filename)
|
||||
@@ -666,8 +668,10 @@ static USBDevice *usb_msd_init(USBBus *bus, const char *filename)
|
||||
char fmt[32];
|
||||
|
||||
/* parse -usbdevice disk: syntax into drive opts */
|
||||
snprintf(id, sizeof(id), "usb%d", nr++);
|
||||
opts = qemu_opts_create(qemu_find_opts("drive"), id, 0, NULL);
|
||||
do {
|
||||
snprintf(id, sizeof(id), "usb%d", nr++);
|
||||
opts = qemu_opts_create(qemu_find_opts("drive"), id, 1, NULL);
|
||||
} while (!opts);
|
||||
|
||||
p1 = strchr(filename, ':');
|
||||
if (p1++) {
|
||||
@@ -678,13 +682,13 @@ static USBDevice *usb_msd_init(USBBus *bus, const char *filename)
|
||||
pstrcpy(fmt, len, p2);
|
||||
qemu_opt_set(opts, "format", fmt);
|
||||
} else if (*filename != ':') {
|
||||
printf("unrecognized USB mass-storage option %s\n", filename);
|
||||
error_report("unrecognized USB mass-storage option %s", filename);
|
||||
return NULL;
|
||||
}
|
||||
filename = p1;
|
||||
}
|
||||
if (!*filename) {
|
||||
printf("block device specification needed\n");
|
||||
error_report("block device specification needed");
|
||||
return NULL;
|
||||
}
|
||||
qemu_opt_set(opts, "file", filename);
|
||||
@@ -758,7 +762,7 @@ static void usb_msd_class_initfn_storage(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_msd_initfn_storage;
|
||||
uc->realize = usb_msd_realize_storage;
|
||||
dc->props = msd_properties;
|
||||
usb_msd_class_initfn_common(klass);
|
||||
}
|
||||
@@ -767,7 +771,7 @@ static void usb_msd_class_initfn_bot(ObjectClass *klass, void *data)
|
||||
{
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_msd_initfn_bot;
|
||||
uc->realize = usb_msd_realize_bot;
|
||||
usb_msd_class_initfn_common(klass);
|
||||
}
|
||||
|
||||
|
@@ -13,6 +13,7 @@
|
||||
#include "qemu/option.h"
|
||||
#include "qemu/config-file.h"
|
||||
#include "trace.h"
|
||||
#include "qemu/error-report.h"
|
||||
|
||||
#include "hw/usb.h"
|
||||
#include "hw/usb/desc.h"
|
||||
@@ -648,7 +649,7 @@ static void usb_uas_handle_control(USBDevice *dev, USBPacket *p,
|
||||
if (ret >= 0) {
|
||||
return;
|
||||
}
|
||||
fprintf(stderr, "%s: unhandled control request\n", __func__);
|
||||
error_report("%s: unhandled control request", __func__);
|
||||
p->status = USB_RET_STALL;
|
||||
}
|
||||
|
||||
@@ -814,8 +815,8 @@ static void usb_uas_handle_data(USBDevice *dev, USBPacket *p)
|
||||
usb_uas_task(uas, &iu);
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "%s: unknown command iu: id 0x%x\n",
|
||||
__func__, iu.hdr.id);
|
||||
error_report("%s: unknown command iu: id 0x%x",
|
||||
__func__, iu.hdr.id);
|
||||
p->status = USB_RET_STALL;
|
||||
break;
|
||||
}
|
||||
@@ -861,7 +862,7 @@ static void usb_uas_handle_data(USBDevice *dev, USBPacket *p)
|
||||
p->status = USB_RET_ASYNC;
|
||||
break;
|
||||
} else {
|
||||
fprintf(stderr, "%s: no inflight request\n", __func__);
|
||||
error_report("%s: no inflight request", __func__);
|
||||
p->status = USB_RET_STALL;
|
||||
break;
|
||||
}
|
||||
@@ -879,7 +880,7 @@ static void usb_uas_handle_data(USBDevice *dev, USBPacket *p)
|
||||
usb_uas_start_next_transfer(uas);
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "%s: invalid endpoint %d\n", __func__, p->ep->nr);
|
||||
error_report("%s: invalid endpoint %d", __func__, p->ep->nr);
|
||||
p->status = USB_RET_STALL;
|
||||
break;
|
||||
}
|
||||
@@ -892,7 +893,7 @@ static void usb_uas_handle_destroy(USBDevice *dev)
|
||||
qemu_bh_delete(uas->status_bh);
|
||||
}
|
||||
|
||||
static int usb_uas_init(USBDevice *dev)
|
||||
static void usb_uas_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
UASDevice *uas = DO_UPCAST(UASDevice, dev, dev);
|
||||
|
||||
@@ -905,8 +906,6 @@ static int usb_uas_init(USBDevice *dev)
|
||||
|
||||
scsi_bus_new(&uas->bus, sizeof(uas->bus), DEVICE(dev),
|
||||
&usb_uas_scsi_info, NULL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_usb_uas = {
|
||||
@@ -928,7 +927,7 @@ static void usb_uas_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_uas_init;
|
||||
uc->realize = usb_uas_realize;
|
||||
uc->product_desc = desc_strings[STR_PRODUCT];
|
||||
uc->usb_desc = &desc;
|
||||
uc->cancel_packet = usb_uas_cancel_io;
|
||||
|
@@ -335,14 +335,13 @@ static void usb_wacom_handle_destroy(USBDevice *dev)
|
||||
}
|
||||
}
|
||||
|
||||
static int usb_wacom_initfn(USBDevice *dev)
|
||||
static void usb_wacom_realize(USBDevice *dev, Error **errp)
|
||||
{
|
||||
USBWacomState *s = DO_UPCAST(USBWacomState, dev, dev);
|
||||
usb_desc_create_serial(dev);
|
||||
usb_desc_init(dev);
|
||||
s->intr = usb_ep_get(dev, USB_TOKEN_IN, 1);
|
||||
s->changed = 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const VMStateDescription vmstate_usb_wacom = {
|
||||
@@ -357,7 +356,7 @@ static void usb_wacom_class_init(ObjectClass *klass, void *data)
|
||||
|
||||
uc->product_desc = "QEMU PenPartner Tablet";
|
||||
uc->usb_desc = &desc_wacom;
|
||||
uc->init = usb_wacom_initfn;
|
||||
uc->realize = usb_wacom_realize;
|
||||
uc->handle_reset = usb_wacom_handle_reset;
|
||||
uc->handle_control = usb_wacom_handle_control;
|
||||
uc->handle_data = usb_wacom_handle_data;
|
||||
|
@@ -23,6 +23,7 @@ typedef struct EHCIPCIInfo {
|
||||
uint16_t vendor_id;
|
||||
uint16_t device_id;
|
||||
uint8_t revision;
|
||||
bool companion;
|
||||
} EHCIPCIInfo;
|
||||
|
||||
static int usb_ehci_pci_initfn(PCIDevice *dev)
|
||||
@@ -71,6 +72,7 @@ static int usb_ehci_pci_initfn(PCIDevice *dev)
|
||||
|
||||
static void usb_ehci_pci_init(Object *obj)
|
||||
{
|
||||
DeviceClass *dc = OBJECT_GET_CLASS(DeviceClass, obj, TYPE_DEVICE);
|
||||
EHCIPCIState *i = PCI_EHCI(obj);
|
||||
EHCIState *s = &i->ehci;
|
||||
|
||||
@@ -81,6 +83,10 @@ static void usb_ehci_pci_init(Object *obj)
|
||||
s->portscbase = 0x44;
|
||||
s->portnr = NB_PORTS;
|
||||
|
||||
if (!dc->hotpluggable) {
|
||||
s->companion_enable = true;
|
||||
}
|
||||
|
||||
usb_ehci_init(s, DEVICE(obj));
|
||||
}
|
||||
|
||||
@@ -137,7 +143,6 @@ static void ehci_class_init(ObjectClass *klass, void *data)
|
||||
k->exit = usb_ehci_pci_exit;
|
||||
k->class_id = PCI_CLASS_SERIAL_USB;
|
||||
k->config_write = usb_ehci_pci_write_config;
|
||||
dc->hotpluggable = false;
|
||||
dc->vmsd = &vmstate_ehci_pci;
|
||||
dc->props = ehci_pci_properties;
|
||||
}
|
||||
@@ -161,6 +166,9 @@ static void ehci_data_class_init(ObjectClass *klass, void *data)
|
||||
k->device_id = i->device_id;
|
||||
k->revision = i->revision;
|
||||
set_bit(DEVICE_CATEGORY_USB, dc->categories);
|
||||
if (i->companion) {
|
||||
dc->hotpluggable = false;
|
||||
}
|
||||
}
|
||||
|
||||
static struct EHCIPCIInfo ehci_pci_info[] = {
|
||||
@@ -174,11 +182,13 @@ static struct EHCIPCIInfo ehci_pci_info[] = {
|
||||
.vendor_id = PCI_VENDOR_ID_INTEL,
|
||||
.device_id = PCI_DEVICE_ID_INTEL_82801I_EHCI1,
|
||||
.revision = 0x03,
|
||||
.companion = true,
|
||||
},{
|
||||
.name = "ich9-usb-ehci2", /* 00:1a.7 */
|
||||
.vendor_id = PCI_VENDOR_ID_INTEL,
|
||||
.device_id = PCI_DEVICE_ID_INTEL_82801I_EHCI2,
|
||||
.revision = 0x03,
|
||||
.companion = true,
|
||||
}
|
||||
};
|
||||
|
||||
|
@@ -2347,10 +2347,13 @@ static USBPortOps ehci_port_ops = {
|
||||
.complete = ehci_async_complete_packet,
|
||||
};
|
||||
|
||||
static USBBusOps ehci_bus_ops = {
|
||||
static USBBusOps ehci_bus_ops_companion = {
|
||||
.register_companion = ehci_register_companion,
|
||||
.wakeup_endpoint = ehci_wakeup_endpoint,
|
||||
};
|
||||
static USBBusOps ehci_bus_ops_standalone = {
|
||||
.wakeup_endpoint = ehci_wakeup_endpoint,
|
||||
};
|
||||
|
||||
static void usb_ehci_pre_save(void *opaque)
|
||||
{
|
||||
@@ -2456,7 +2459,8 @@ void usb_ehci_realize(EHCIState *s, DeviceState *dev, Error **errp)
|
||||
return;
|
||||
}
|
||||
|
||||
usb_bus_new(&s->bus, sizeof(s->bus), &ehci_bus_ops, dev);
|
||||
usb_bus_new(&s->bus, sizeof(s->bus), s->companion_enable ?
|
||||
&ehci_bus_ops_companion : &ehci_bus_ops_standalone, dev);
|
||||
for (i = 0; i < s->portnr; i++) {
|
||||
usb_register_port(&s->bus, &s->ports[i], s, i, &ehci_port_ops,
|
||||
USB_SPEED_MASK_HIGH);
|
||||
|
@@ -262,6 +262,7 @@ struct EHCIState {
|
||||
MemoryRegion mem_opreg;
|
||||
MemoryRegion mem_ports;
|
||||
int companion_count;
|
||||
bool companion_enable;
|
||||
uint16_t capsbase;
|
||||
uint16_t opregbase;
|
||||
uint16_t portscbase;
|
||||
|
@@ -31,20 +31,11 @@
|
||||
#include "hw/pci/pci.h"
|
||||
#include "hw/sysbus.h"
|
||||
#include "hw/qdev-dma.h"
|
||||
#include "trace.h"
|
||||
|
||||
//#define DEBUG_OHCI
|
||||
/* Dump packet contents. */
|
||||
//#define DEBUG_PACKET
|
||||
//#define DEBUG_ISOCH
|
||||
/* This causes frames to occur 1000x slower */
|
||||
//#define OHCI_TIME_WARP 1
|
||||
|
||||
#ifdef DEBUG_OHCI
|
||||
#define DPRINTF printf
|
||||
#else
|
||||
#define DPRINTF(...)
|
||||
#endif
|
||||
|
||||
/* Number of Downstream Ports on the root hub. */
|
||||
|
||||
#define OHCI_MAX_PORTS 15
|
||||
@@ -350,7 +341,7 @@ static void ohci_attach(USBPort *port1)
|
||||
ohci_set_interrupt(s, OHCI_INTR_RD);
|
||||
}
|
||||
|
||||
DPRINTF("usb-ohci: Attached port %d\n", port1->index);
|
||||
trace_usb_ohci_port_attach(port1->index);
|
||||
|
||||
if (old_state != port->ctrl) {
|
||||
ohci_set_interrupt(s, OHCI_INTR_RHSC);
|
||||
@@ -375,7 +366,7 @@ static void ohci_detach(USBPort *port1)
|
||||
port->ctrl &= ~OHCI_PORT_PES;
|
||||
port->ctrl |= OHCI_PORT_PESC;
|
||||
}
|
||||
DPRINTF("usb-ohci: Detached port %d\n", port1->index);
|
||||
trace_usb_ohci_port_detach(port1->index);
|
||||
|
||||
if (old_state != port->ctrl) {
|
||||
ohci_set_interrupt(s, OHCI_INTR_RHSC);
|
||||
@@ -388,14 +379,14 @@ static void ohci_wakeup(USBPort *port1)
|
||||
OHCIPort *port = &s->rhport[port1->index];
|
||||
uint32_t intr = 0;
|
||||
if (port->ctrl & OHCI_PORT_PSS) {
|
||||
DPRINTF("usb-ohci: port %d: wakeup\n", port1->index);
|
||||
trace_usb_ohci_port_wakeup(port1->index);
|
||||
port->ctrl |= OHCI_PORT_PSSC;
|
||||
port->ctrl &= ~OHCI_PORT_PSS;
|
||||
intr = OHCI_INTR_RHSC;
|
||||
}
|
||||
/* Note that the controller can be suspended even if this port is not */
|
||||
if ((s->ctl & OHCI_CTL_HCFS) == OHCI_USB_SUSPEND) {
|
||||
DPRINTF("usb-ohci: remote-wakeup: SUSPEND->RESUME\n");
|
||||
trace_usb_ohci_remote_wakeup(s->name);
|
||||
/* This is the one state transition the controller can do by itself */
|
||||
s->ctl &= ~OHCI_CTL_HCFS;
|
||||
s->ctl |= OHCI_USB_RESUME;
|
||||
@@ -497,7 +488,7 @@ static void ohci_reset(void *opaque)
|
||||
ohci->async_td = 0;
|
||||
}
|
||||
ohci_stop_endpoints(ohci);
|
||||
DPRINTF("usb-ohci: Reset %s\n", ohci->name);
|
||||
trace_usb_ohci_reset(ohci->name);
|
||||
}
|
||||
|
||||
/* Get an array of dwords from main memory */
|
||||
@@ -690,9 +681,8 @@ static void ohci_process_lists(OHCIState *ohci, int completion);
|
||||
static void ohci_async_complete_packet(USBPort *port, USBPacket *packet)
|
||||
{
|
||||
OHCIState *ohci = container_of(packet, OHCIState, usb_packet);
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF("Async packet complete\n");
|
||||
#endif
|
||||
|
||||
trace_usb_ohci_async_complete();
|
||||
ohci->async_complete = true;
|
||||
ohci_process_lists(ohci, 1);
|
||||
}
|
||||
@@ -704,9 +694,7 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
{
|
||||
int dir;
|
||||
size_t len = 0;
|
||||
#ifdef DEBUG_ISOCH
|
||||
const char *str = NULL;
|
||||
#endif
|
||||
int pid;
|
||||
int ret;
|
||||
int i;
|
||||
@@ -723,7 +711,7 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
addr = ed->head & OHCI_DPTR_MASK;
|
||||
|
||||
if (ohci_read_iso_td(ohci, addr, &iso_td)) {
|
||||
printf("usb-ohci: ISO_TD read error at %x\n", addr);
|
||||
trace_usb_ohci_iso_td_read_failed(addr);
|
||||
ohci_die(ohci);
|
||||
return 0;
|
||||
}
|
||||
@@ -732,31 +720,25 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
frame_count = OHCI_BM(iso_td.flags, TD_FC);
|
||||
relative_frame_number = USUB(ohci->frame_number, starting_frame);
|
||||
|
||||
#ifdef DEBUG_ISOCH
|
||||
printf("--- ISO_TD ED head 0x%.8x tailp 0x%.8x\n"
|
||||
"0x%.8x 0x%.8x 0x%.8x 0x%.8x\n"
|
||||
"0x%.8x 0x%.8x 0x%.8x 0x%.8x\n"
|
||||
"0x%.8x 0x%.8x 0x%.8x 0x%.8x\n"
|
||||
"frame_number 0x%.8x starting_frame 0x%.8x\n"
|
||||
"frame_count 0x%.8x relative %d\n"
|
||||
"di 0x%.8x cc 0x%.8x\n",
|
||||
trace_usb_ohci_iso_td_head(
|
||||
ed->head & OHCI_DPTR_MASK, ed->tail & OHCI_DPTR_MASK,
|
||||
iso_td.flags, iso_td.bp, iso_td.next, iso_td.be,
|
||||
iso_td.offset[0], iso_td.offset[1], iso_td.offset[2], iso_td.offset[3],
|
||||
iso_td.offset[4], iso_td.offset[5], iso_td.offset[6], iso_td.offset[7],
|
||||
ohci->frame_number, starting_frame,
|
||||
frame_count, relative_frame_number,
|
||||
OHCI_BM(iso_td.flags, TD_DI), OHCI_BM(iso_td.flags, TD_CC));
|
||||
#endif
|
||||
ohci->frame_number, starting_frame,
|
||||
frame_count, relative_frame_number);
|
||||
trace_usb_ohci_iso_td_head_offset(
|
||||
iso_td.offset[0], iso_td.offset[1],
|
||||
iso_td.offset[2], iso_td.offset[3],
|
||||
iso_td.offset[4], iso_td.offset[5],
|
||||
iso_td.offset[6], iso_td.offset[7]);
|
||||
|
||||
if (relative_frame_number < 0) {
|
||||
DPRINTF("usb-ohci: ISO_TD R=%d < 0\n", relative_frame_number);
|
||||
trace_usb_ohci_iso_td_relative_frame_number_neg(relative_frame_number);
|
||||
return 1;
|
||||
} else if (relative_frame_number > frame_count) {
|
||||
/* ISO TD expired - retire the TD to the Done Queue and continue with
|
||||
the next ISO TD of the same ED */
|
||||
DPRINTF("usb-ohci: ISO_TD R=%d > FC=%d\n", relative_frame_number,
|
||||
frame_count);
|
||||
trace_usb_ohci_iso_td_relative_frame_number_big(relative_frame_number,
|
||||
frame_count);
|
||||
OHCI_SET_BM(iso_td.flags, TD_CC, OHCI_CC_DATAOVERRUN);
|
||||
ed->head &= ~OHCI_DPTR_MASK;
|
||||
ed->head |= (iso_td.next & OHCI_DPTR_MASK);
|
||||
@@ -775,30 +757,24 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
dir = OHCI_BM(ed->flags, ED_D);
|
||||
switch (dir) {
|
||||
case OHCI_TD_DIR_IN:
|
||||
#ifdef DEBUG_ISOCH
|
||||
str = "in";
|
||||
#endif
|
||||
pid = USB_TOKEN_IN;
|
||||
break;
|
||||
case OHCI_TD_DIR_OUT:
|
||||
#ifdef DEBUG_ISOCH
|
||||
str = "out";
|
||||
#endif
|
||||
pid = USB_TOKEN_OUT;
|
||||
break;
|
||||
case OHCI_TD_DIR_SETUP:
|
||||
#ifdef DEBUG_ISOCH
|
||||
str = "setup";
|
||||
#endif
|
||||
pid = USB_TOKEN_SETUP;
|
||||
break;
|
||||
default:
|
||||
printf("usb-ohci: Bad direction %d\n", dir);
|
||||
trace_usb_ohci_iso_td_bad_direction(dir);
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (!iso_td.bp || !iso_td.be) {
|
||||
printf("usb-ohci: ISO_TD bp 0x%.8x be 0x%.8x\n", iso_td.bp, iso_td.be);
|
||||
trace_usb_ohci_iso_td_bad_bp_be(iso_td.bp, iso_td.be);
|
||||
return 1;
|
||||
}
|
||||
|
||||
@@ -808,14 +784,12 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
if (!(OHCI_BM(start_offset, TD_PSW_CC) & 0xe) ||
|
||||
((relative_frame_number < frame_count) &&
|
||||
!(OHCI_BM(next_offset, TD_PSW_CC) & 0xe))) {
|
||||
printf("usb-ohci: ISO_TD cc != not accessed 0x%.8x 0x%.8x\n",
|
||||
start_offset, next_offset);
|
||||
trace_usb_ohci_iso_td_bad_cc_not_accessed(start_offset, next_offset);
|
||||
return 1;
|
||||
}
|
||||
|
||||
if ((relative_frame_number < frame_count) && (start_offset > next_offset)) {
|
||||
printf("usb-ohci: ISO_TD start_offset=0x%.8x > next_offset=0x%.8x\n",
|
||||
start_offset, next_offset);
|
||||
trace_usb_ohci_iso_td_bad_cc_overrun(start_offset, next_offset);
|
||||
return 1;
|
||||
}
|
||||
|
||||
@@ -875,10 +849,8 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
ret = ohci->usb_packet.status;
|
||||
}
|
||||
|
||||
#ifdef DEBUG_ISOCH
|
||||
printf("so 0x%.8x eo 0x%.8x\nsa 0x%.8x ea 0x%.8x\ndir %s len %zu ret %d\n",
|
||||
start_offset, end_offset, start_addr, end_addr, str, len, ret);
|
||||
#endif
|
||||
trace_usb_ohci_iso_td_so(start_offset, end_offset, start_addr, end_addr,
|
||||
str, len, ret);
|
||||
|
||||
/* Writeback */
|
||||
if (dir == OHCI_TD_DIR_IN && ret >= 0 && ret <= len) {
|
||||
@@ -898,13 +870,13 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_SIZE, 0);
|
||||
} else {
|
||||
if (ret > (ssize_t) len) {
|
||||
printf("usb-ohci: DataOverrun %d > %zu\n", ret, len);
|
||||
trace_usb_ohci_iso_td_data_overrun(ret, len);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_CC,
|
||||
OHCI_CC_DATAOVERRUN);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_SIZE,
|
||||
len);
|
||||
} else if (ret >= 0) {
|
||||
printf("usb-ohci: DataUnderrun %d\n", ret);
|
||||
trace_usb_ohci_iso_td_data_underrun(ret);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_CC,
|
||||
OHCI_CC_DATAUNDERRUN);
|
||||
} else {
|
||||
@@ -918,14 +890,14 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
break;
|
||||
case USB_RET_NAK:
|
||||
case USB_RET_STALL:
|
||||
printf("usb-ohci: got NAK/STALL %d\n", ret);
|
||||
trace_usb_ohci_iso_td_nak(ret);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_CC,
|
||||
OHCI_CC_STALL);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_SIZE,
|
||||
0);
|
||||
break;
|
||||
default:
|
||||
printf("usb-ohci: Bad device response %d\n", ret);
|
||||
trace_usb_ohci_iso_td_bad_response(ret);
|
||||
OHCI_SET_BM(iso_td.offset[relative_frame_number], TD_PSW_CC,
|
||||
OHCI_CC_UNDEXPETEDPID);
|
||||
break;
|
||||
@@ -950,6 +922,43 @@ static int ohci_service_iso_td(OHCIState *ohci, struct ohci_ed *ed,
|
||||
return 1;
|
||||
}
|
||||
|
||||
#ifdef trace_event_get_state
|
||||
static void ohci_td_pkt(const char *msg, const uint8_t *buf, size_t len)
|
||||
{
|
||||
bool print16 = !!trace_event_get_state(TRACE_USB_OHCI_TD_PKT_SHORT);
|
||||
bool printall = !!trace_event_get_state(TRACE_USB_OHCI_TD_PKT_FULL);
|
||||
const int width = 16;
|
||||
int i;
|
||||
char tmp[3 * width + 1];
|
||||
char *p = tmp;
|
||||
|
||||
if (!printall && !print16) {
|
||||
return;
|
||||
}
|
||||
|
||||
for (i = 0; ; i++) {
|
||||
if (i && (!(i % width) || (i == len))) {
|
||||
if (!printall) {
|
||||
trace_usb_ohci_td_pkt_short(msg, tmp);
|
||||
break;
|
||||
}
|
||||
trace_usb_ohci_td_pkt_full(msg, tmp);
|
||||
p = tmp;
|
||||
*p = 0;
|
||||
}
|
||||
if (i == len) {
|
||||
break;
|
||||
}
|
||||
|
||||
p += sprintf(p, " %.2x", buf[i]);
|
||||
}
|
||||
}
|
||||
#else
|
||||
static void ohci_td_pkt(const char *msg, const uint8_t *buf, size_t len)
|
||||
{
|
||||
}
|
||||
#endif
|
||||
|
||||
/* Service a transport descriptor.
|
||||
Returns nonzero to terminate processing of this endpoint. */
|
||||
|
||||
@@ -957,9 +966,7 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
{
|
||||
int dir;
|
||||
size_t len = 0, pktlen = 0;
|
||||
#ifdef DEBUG_PACKET
|
||||
const char *str = NULL;
|
||||
#endif
|
||||
int pid;
|
||||
int ret;
|
||||
int i;
|
||||
@@ -974,13 +981,11 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
/* See if this TD has already been submitted to the device. */
|
||||
completion = (addr == ohci->async_td);
|
||||
if (completion && !ohci->async_complete) {
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF("Skipping async TD\n");
|
||||
#endif
|
||||
trace_usb_ohci_td_skip_async();
|
||||
return 1;
|
||||
}
|
||||
if (ohci_read_td(ohci, addr, &td)) {
|
||||
fprintf(stderr, "usb-ohci: TD read error at %x\n", addr);
|
||||
trace_usb_ohci_td_read_error(addr);
|
||||
ohci_die(ohci);
|
||||
return 0;
|
||||
}
|
||||
@@ -998,25 +1003,19 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
|
||||
switch (dir) {
|
||||
case OHCI_TD_DIR_IN:
|
||||
#ifdef DEBUG_PACKET
|
||||
str = "in";
|
||||
#endif
|
||||
pid = USB_TOKEN_IN;
|
||||
break;
|
||||
case OHCI_TD_DIR_OUT:
|
||||
#ifdef DEBUG_PACKET
|
||||
str = "out";
|
||||
#endif
|
||||
pid = USB_TOKEN_OUT;
|
||||
break;
|
||||
case OHCI_TD_DIR_SETUP:
|
||||
#ifdef DEBUG_PACKET
|
||||
str = "setup";
|
||||
#endif
|
||||
pid = USB_TOKEN_SETUP;
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "usb-ohci: Bad direction\n");
|
||||
trace_usb_ohci_td_bad_direction(dir);
|
||||
return 1;
|
||||
}
|
||||
if (td.cbp && td.be) {
|
||||
@@ -1043,19 +1042,10 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
}
|
||||
|
||||
flag_r = (td.flags & OHCI_TD_R) != 0;
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF(" TD @ 0x%.8x %" PRId64 " of %" PRId64
|
||||
" bytes %s r=%d cbp=0x%.8x be=0x%.8x\n",
|
||||
addr, (int64_t)pktlen, (int64_t)len, str, flag_r, td.cbp, td.be);
|
||||
trace_usb_ohci_td_pkt_hdr(addr, (int64_t)pktlen, (int64_t)len, str,
|
||||
flag_r, td.cbp, td.be);
|
||||
ohci_td_pkt("OUT", ohci->usb_buf, pktlen);
|
||||
|
||||
if (pktlen > 0 && dir != OHCI_TD_DIR_IN) {
|
||||
DPRINTF(" data:");
|
||||
for (i = 0; i < pktlen; i++) {
|
||||
printf(" %.2x", ohci->usb_buf[i]);
|
||||
}
|
||||
DPRINTF("\n");
|
||||
}
|
||||
#endif
|
||||
if (completion) {
|
||||
ohci->async_td = 0;
|
||||
ohci->async_complete = false;
|
||||
@@ -1066,9 +1056,7 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
This should be sufficient as long as devices respond in a
|
||||
timely manner.
|
||||
*/
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF("Too many pending packets\n");
|
||||
#endif
|
||||
trace_usb_ohci_td_too_many_pending();
|
||||
return 1;
|
||||
}
|
||||
dev = ohci_find_device(ohci, OHCI_BM(ed->flags, ED_FA));
|
||||
@@ -1077,9 +1065,8 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
OHCI_BM(td.flags, TD_DI) == 0);
|
||||
usb_packet_addbuf(&ohci->usb_packet, ohci->usb_buf, pktlen);
|
||||
usb_handle_packet(dev, &ohci->usb_packet);
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF("status=%d\n", ohci->usb_packet.status);
|
||||
#endif
|
||||
trace_usb_ohci_td_packet_status(ohci->usb_packet.status);
|
||||
|
||||
if (ohci->usb_packet.status == USB_RET_ASYNC) {
|
||||
usb_device_flush_ep_queue(dev, ep);
|
||||
ohci->async_td = addr;
|
||||
@@ -1098,12 +1085,7 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
DMA_DIRECTION_FROM_DEVICE)) {
|
||||
ohci_die(ohci);
|
||||
}
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF(" data:");
|
||||
for (i = 0; i < ret; i++)
|
||||
printf(" %.2x", ohci->usb_buf[i]);
|
||||
DPRINTF("\n");
|
||||
#endif
|
||||
ohci_td_pkt("IN", ohci->usb_buf, pktlen);
|
||||
} else {
|
||||
ret = pktlen;
|
||||
}
|
||||
@@ -1137,28 +1119,28 @@ static int ohci_service_td(OHCIState *ohci, struct ohci_ed *ed)
|
||||
ed->head |= OHCI_ED_C;
|
||||
} else {
|
||||
if (ret >= 0) {
|
||||
DPRINTF("usb-ohci: Underrun\n");
|
||||
trace_usb_ohci_td_underrun();
|
||||
OHCI_SET_BM(td.flags, TD_CC, OHCI_CC_DATAUNDERRUN);
|
||||
} else {
|
||||
switch (ret) {
|
||||
case USB_RET_IOERROR:
|
||||
case USB_RET_NODEV:
|
||||
DPRINTF("usb-ohci: got DEV ERROR\n");
|
||||
trace_usb_ohci_td_dev_error();
|
||||
OHCI_SET_BM(td.flags, TD_CC, OHCI_CC_DEVICENOTRESPONDING);
|
||||
break;
|
||||
case USB_RET_NAK:
|
||||
DPRINTF("usb-ohci: got NAK\n");
|
||||
trace_usb_ohci_td_nak();
|
||||
return 1;
|
||||
case USB_RET_STALL:
|
||||
DPRINTF("usb-ohci: got STALL\n");
|
||||
trace_usb_ohci_td_stall();
|
||||
OHCI_SET_BM(td.flags, TD_CC, OHCI_CC_STALL);
|
||||
break;
|
||||
case USB_RET_BABBLE:
|
||||
DPRINTF("usb-ohci: got BABBLE\n");
|
||||
trace_usb_ohci_td_babble();
|
||||
OHCI_SET_BM(td.flags, TD_CC, OHCI_CC_DATAOVERRUN);
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr, "usb-ohci: Bad device response %d\n", ret);
|
||||
trace_usb_ohci_td_bad_device_response(ret);
|
||||
OHCI_SET_BM(td.flags, TD_CC, OHCI_CC_UNDEXPETEDPID);
|
||||
OHCI_SET_BM(td.flags, TD_EC, 3);
|
||||
break;
|
||||
@@ -1198,7 +1180,7 @@ static int ohci_service_ed_list(OHCIState *ohci, uint32_t head, int completion)
|
||||
|
||||
for (cur = head; cur; cur = next_ed) {
|
||||
if (ohci_read_ed(ohci, cur, &ed)) {
|
||||
fprintf(stderr, "usb-ohci: ED read error at %x\n", cur);
|
||||
trace_usb_ohci_ed_read_error(cur);
|
||||
ohci_die(ohci);
|
||||
return 0;
|
||||
}
|
||||
@@ -1219,16 +1201,15 @@ static int ohci_service_ed_list(OHCIState *ohci, uint32_t head, int completion)
|
||||
}
|
||||
|
||||
while ((ed.head & OHCI_DPTR_MASK) != ed.tail) {
|
||||
#ifdef DEBUG_PACKET
|
||||
DPRINTF("ED @ 0x%.8x fa=%u en=%u d=%u s=%u k=%u f=%u mps=%u "
|
||||
"h=%u c=%u\n head=0x%.8x tailp=0x%.8x next=0x%.8x\n", cur,
|
||||
trace_usb_ohci_ed_pkt(cur, (ed.head & OHCI_ED_H) != 0,
|
||||
(ed.head & OHCI_ED_C) != 0, ed.head & OHCI_DPTR_MASK,
|
||||
ed.tail & OHCI_DPTR_MASK, ed.next & OHCI_DPTR_MASK);
|
||||
trace_usb_ohci_ed_pkt_flags(
|
||||
OHCI_BM(ed.flags, ED_FA), OHCI_BM(ed.flags, ED_EN),
|
||||
OHCI_BM(ed.flags, ED_D), (ed.flags & OHCI_ED_S)!= 0,
|
||||
(ed.flags & OHCI_ED_K) != 0, (ed.flags & OHCI_ED_F) != 0,
|
||||
OHCI_BM(ed.flags, ED_MPS), (ed.head & OHCI_ED_H) != 0,
|
||||
(ed.head & OHCI_ED_C) != 0, ed.head & OHCI_DPTR_MASK,
|
||||
ed.tail & OHCI_DPTR_MASK, ed.next & OHCI_DPTR_MASK);
|
||||
#endif
|
||||
OHCI_BM(ed.flags, ED_MPS));
|
||||
|
||||
active = 1;
|
||||
|
||||
if ((ed.flags & OHCI_ED_F) == 0) {
|
||||
@@ -1263,8 +1244,7 @@ static void ohci_process_lists(OHCIState *ohci, int completion)
|
||||
{
|
||||
if ((ohci->ctl & OHCI_CTL_CLE) && (ohci->status & OHCI_STATUS_CLF)) {
|
||||
if (ohci->ctrl_cur && ohci->ctrl_cur != ohci->ctrl_head) {
|
||||
DPRINTF("usb-ohci: head %x, cur %x\n",
|
||||
ohci->ctrl_head, ohci->ctrl_cur);
|
||||
trace_usb_ohci_process_lists(ohci->ctrl_head, ohci->ctrl_cur);
|
||||
}
|
||||
if (!ohci_service_ed_list(ohci, ohci->ctrl_head, completion)) {
|
||||
ohci->ctrl_cur = 0;
|
||||
@@ -1287,7 +1267,7 @@ static void ohci_frame_boundary(void *opaque)
|
||||
struct ohci_hcca hcca;
|
||||
|
||||
if (ohci_read_hcca(ohci, ohci->hcca, &hcca)) {
|
||||
fprintf(stderr, "usb-ohci: HCCA read error at %x\n", ohci->hcca);
|
||||
trace_usb_ohci_hcca_read_error(ohci->hcca);
|
||||
ohci_die(ohci);
|
||||
return;
|
||||
}
|
||||
@@ -1356,12 +1336,12 @@ static int ohci_bus_start(OHCIState *ohci)
|
||||
ohci);
|
||||
|
||||
if (ohci->eof_timer == NULL) {
|
||||
fprintf(stderr, "usb-ohci: %s: timer_new_ns failed\n", ohci->name);
|
||||
trace_usb_ohci_bus_eof_timer_failed(ohci->name);
|
||||
ohci_die(ohci);
|
||||
return 0;
|
||||
}
|
||||
|
||||
DPRINTF("usb-ohci: %s: USB Operational\n", ohci->name);
|
||||
trace_usb_ohci_start(ohci->name);
|
||||
|
||||
ohci_sof(ohci);
|
||||
|
||||
@@ -1371,6 +1351,7 @@ static int ohci_bus_start(OHCIState *ohci)
|
||||
/* Stop sending SOF tokens on the bus */
|
||||
static void ohci_bus_stop(OHCIState *ohci)
|
||||
{
|
||||
trace_usb_ohci_stop(ohci->name);
|
||||
if (ohci->eof_timer) {
|
||||
timer_del(ohci->eof_timer);
|
||||
timer_free(ohci->eof_timer);
|
||||
@@ -1416,8 +1397,7 @@ static void ohci_set_frame_interval(OHCIState *ohci, uint16_t val)
|
||||
val &= OHCI_FMI_FI;
|
||||
|
||||
if (val != ohci->fi) {
|
||||
DPRINTF("usb-ohci: %s: FrameInterval = 0x%x (%u)\n",
|
||||
ohci->name, ohci->fi, ohci->fi);
|
||||
trace_usb_ohci_set_frame_interval(ohci->name, ohci->fi, ohci->fi);
|
||||
}
|
||||
|
||||
ohci->fi = val;
|
||||
@@ -1449,20 +1429,19 @@ static void ohci_set_ctl(OHCIState *ohci, uint32_t val)
|
||||
if (old_state == new_state)
|
||||
return;
|
||||
|
||||
trace_usb_ohci_set_ctl(ohci->name, new_state);
|
||||
switch (new_state) {
|
||||
case OHCI_USB_OPERATIONAL:
|
||||
ohci_bus_start(ohci);
|
||||
break;
|
||||
case OHCI_USB_SUSPEND:
|
||||
ohci_bus_stop(ohci);
|
||||
DPRINTF("usb-ohci: %s: USB Suspended\n", ohci->name);
|
||||
break;
|
||||
case OHCI_USB_RESUME:
|
||||
DPRINTF("usb-ohci: %s: USB Resume\n", ohci->name);
|
||||
trace_usb_ohci_resume(ohci->name);
|
||||
break;
|
||||
case OHCI_USB_RESET:
|
||||
ohci_reset(ohci);
|
||||
DPRINTF("usb-ohci: %s: USB Reset\n", ohci->name);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -1507,7 +1486,7 @@ static void ohci_set_hub_status(OHCIState *ohci, uint32_t val)
|
||||
|
||||
for (i = 0; i < ohci->num_ports; i++)
|
||||
ohci_port_power(ohci, i, 0);
|
||||
DPRINTF("usb-ohci: powered down all ports\n");
|
||||
trace_usb_ohci_hub_power_down();
|
||||
}
|
||||
|
||||
if (val & OHCI_RHS_LPSC) {
|
||||
@@ -1515,7 +1494,7 @@ static void ohci_set_hub_status(OHCIState *ohci, uint32_t val)
|
||||
|
||||
for (i = 0; i < ohci->num_ports; i++)
|
||||
ohci_port_power(ohci, i, 1);
|
||||
DPRINTF("usb-ohci: powered up all ports\n");
|
||||
trace_usb_ohci_hub_power_up();
|
||||
}
|
||||
|
||||
if (val & OHCI_RHS_DRWE)
|
||||
@@ -1547,11 +1526,11 @@ static void ohci_port_set_status(OHCIState *ohci, int portnum, uint32_t val)
|
||||
ohci_port_set_if_connected(ohci, portnum, val & OHCI_PORT_PES);
|
||||
|
||||
if (ohci_port_set_if_connected(ohci, portnum, val & OHCI_PORT_PSS)) {
|
||||
DPRINTF("usb-ohci: port %d: SUSPEND\n", portnum);
|
||||
trace_usb_ohci_port_suspend(portnum);
|
||||
}
|
||||
|
||||
if (ohci_port_set_if_connected(ohci, portnum, val & OHCI_PORT_PRS)) {
|
||||
DPRINTF("usb-ohci: port %d: RESET\n", portnum);
|
||||
trace_usb_ohci_port_reset(portnum);
|
||||
usb_device_reset(port->port.dev);
|
||||
port->ctrl &= ~OHCI_PORT_PRS;
|
||||
/* ??? Should this also set OHCI_PORT_PESC. */
|
||||
@@ -1579,7 +1558,7 @@ static uint64_t ohci_mem_read(void *opaque,
|
||||
|
||||
/* Only aligned reads are allowed on OHCI */
|
||||
if (addr & 3) {
|
||||
fprintf(stderr, "usb-ohci: Mis-aligned read\n");
|
||||
trace_usb_ohci_mem_read_unaligned(addr);
|
||||
return 0xffffffff;
|
||||
} else if (addr >= 0x54 && addr < 0x54 + ohci->num_ports * 4) {
|
||||
/* HcRhPortStatus */
|
||||
@@ -1685,7 +1664,7 @@ static uint64_t ohci_mem_read(void *opaque,
|
||||
break;
|
||||
|
||||
default:
|
||||
fprintf(stderr, "ohci_read: Bad offset %x\n", (int)addr);
|
||||
trace_usb_ohci_mem_read_bad_offset(addr);
|
||||
retval = 0xffffffff;
|
||||
}
|
||||
}
|
||||
@@ -1702,7 +1681,7 @@ static void ohci_mem_write(void *opaque,
|
||||
|
||||
/* Only aligned reads are allowed on OHCI */
|
||||
if (addr & 3) {
|
||||
fprintf(stderr, "usb-ohci: Mis-aligned write\n");
|
||||
trace_usb_ohci_mem_write_unaligned(addr);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -1816,7 +1795,7 @@ static void ohci_mem_write(void *opaque,
|
||||
break;
|
||||
|
||||
default:
|
||||
fprintf(stderr, "ohci_write: Bad offset %x\n", (int)addr);
|
||||
trace_usb_ohci_mem_write_bad_offset(addr);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -1869,8 +1848,7 @@ static int usb_ohci_init(OHCIState *ohci, DeviceState *dev,
|
||||
usb_bit_time = 1;
|
||||
}
|
||||
#endif
|
||||
DPRINTF("usb-ohci: usb_bit_time=%" PRId64 " usb_frame_time=%" PRId64 "\n",
|
||||
usb_frame_time, usb_bit_time);
|
||||
trace_usb_ohci_init_time(usb_frame_time, usb_bit_time);
|
||||
}
|
||||
|
||||
ohci->num_ports = num_ports;
|
||||
@@ -1928,7 +1906,7 @@ static void ohci_die(OHCIState *ohci)
|
||||
{
|
||||
OHCIPCIState *dev = container_of(ohci, OHCIPCIState, state);
|
||||
|
||||
fprintf(stderr, "%s: DMA error\n", __func__);
|
||||
trace_usb_ohci_die();
|
||||
|
||||
ohci_set_interrupt(ohci, OHCI_INTR_UE);
|
||||
ohci_bus_stop(ohci);
|
||||
@@ -1959,6 +1937,7 @@ static void usb_ohci_exit(PCIDevice *dev)
|
||||
OHCIPCIState *ohci = PCI_OHCI(dev);
|
||||
OHCIState *s = &ohci->state;
|
||||
|
||||
trace_usb_ohci_exit(s->name);
|
||||
ohci_bus_stop(s);
|
||||
|
||||
if (s->async_td) {
|
||||
|
@@ -1279,13 +1279,18 @@ static void usb_uhci_exit(PCIDevice *dev)
|
||||
}
|
||||
}
|
||||
|
||||
static Property uhci_properties[] = {
|
||||
static Property uhci_properties_companion[] = {
|
||||
DEFINE_PROP_STRING("masterbus", UHCIState, masterbus),
|
||||
DEFINE_PROP_UINT32("firstport", UHCIState, firstport, 0),
|
||||
DEFINE_PROP_UINT32("bandwidth", UHCIState, frame_bandwidth, 1280),
|
||||
DEFINE_PROP_UINT32("maxframes", UHCIState, maxframes, 128),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
static Property uhci_properties_standalone[] = {
|
||||
DEFINE_PROP_UINT32("bandwidth", UHCIState, frame_bandwidth, 1280),
|
||||
DEFINE_PROP_UINT32("maxframes", UHCIState, maxframes, 128),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
static void uhci_class_init(ObjectClass *klass, void *data)
|
||||
{
|
||||
@@ -1300,9 +1305,14 @@ static void uhci_class_init(ObjectClass *klass, void *data)
|
||||
k->device_id = info->device_id;
|
||||
k->revision = info->revision;
|
||||
k->class_id = PCI_CLASS_SERIAL_USB;
|
||||
dc->hotpluggable = false;
|
||||
dc->vmsd = &vmstate_uhci;
|
||||
dc->props = uhci_properties;
|
||||
if (!info->unplug) {
|
||||
/* uhci controllers in companion setups can't be hotplugged */
|
||||
dc->hotpluggable = false;
|
||||
dc->props = uhci_properties_companion;
|
||||
} else {
|
||||
dc->props = uhci_properties_standalone;
|
||||
}
|
||||
set_bit(DEVICE_CATEGORY_USB, dc->categories);
|
||||
u->info = *info;
|
||||
}
|
||||
|
@@ -3874,7 +3874,6 @@ static void xhci_class_init(ObjectClass *klass, void *data)
|
||||
dc->vmsd = &vmstate_xhci;
|
||||
dc->props = xhci_properties;
|
||||
dc->reset = xhci_reset;
|
||||
dc->hotpluggable = false;
|
||||
set_bit(DEVICE_CATEGORY_USB, dc->categories);
|
||||
k->init = usb_xhci_initfn;
|
||||
k->exit = usb_xhci_exit;
|
||||
|
@@ -275,7 +275,7 @@ static void usb_host_libusb_error(const char *func, int rc)
|
||||
} else {
|
||||
errname = "?";
|
||||
}
|
||||
fprintf(stderr, "%s: %d [%s]\n", func, rc, errname);
|
||||
error_report("%s: %d [%s]", func, rc, errname);
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------------ */
|
||||
@@ -834,6 +834,7 @@ static int usb_host_open(USBHostDevice *s, libusb_device *dev)
|
||||
int bus_num = libusb_get_bus_number(dev);
|
||||
int addr = libusb_get_device_address(dev);
|
||||
int rc;
|
||||
Error *local_err = NULL;
|
||||
|
||||
trace_usb_host_open_started(bus_num, addr);
|
||||
|
||||
@@ -869,8 +870,10 @@ static int usb_host_open(USBHostDevice *s, libusb_device *dev)
|
||||
"host:%d.%d", bus_num, addr);
|
||||
}
|
||||
|
||||
rc = usb_device_attach(udev);
|
||||
if (rc) {
|
||||
usb_device_attach(udev, &local_err);
|
||||
if (local_err) {
|
||||
error_report("%s", error_get_pretty(local_err));
|
||||
error_free(local_err);
|
||||
goto fail;
|
||||
}
|
||||
|
||||
@@ -948,21 +951,21 @@ static void usb_host_exit_notifier(struct Notifier *n, void *data)
|
||||
}
|
||||
}
|
||||
|
||||
static int usb_host_initfn(USBDevice *udev)
|
||||
static void usb_host_realize(USBDevice *udev, Error **errp)
|
||||
{
|
||||
USBHostDevice *s = USB_HOST_DEVICE(udev);
|
||||
|
||||
if (s->match.vendor_id > 0xffff) {
|
||||
error_report("vendorid out of range");
|
||||
return -1;
|
||||
error_setg(errp, "vendorid out of range");
|
||||
return;
|
||||
}
|
||||
if (s->match.product_id > 0xffff) {
|
||||
error_report("productid out of range");
|
||||
return -1;
|
||||
error_setg(errp, "productid out of range");
|
||||
return;
|
||||
}
|
||||
if (s->match.addr > 127) {
|
||||
error_report("hostaddr out of range");
|
||||
return -1;
|
||||
error_setg(errp, "hostaddr out of range");
|
||||
return;
|
||||
}
|
||||
|
||||
loglevel = s->loglevel;
|
||||
@@ -977,7 +980,6 @@ static int usb_host_initfn(USBDevice *udev)
|
||||
QTAILQ_INSERT_TAIL(&hostdevs, s, next);
|
||||
add_boot_device_path(s->bootindex, &udev->qdev, NULL);
|
||||
usb_host_auto_check(NULL);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void usb_host_handle_destroy(USBDevice *udev)
|
||||
@@ -1374,14 +1376,13 @@ static int usb_host_alloc_streams(USBDevice *udev, USBEndpoint **eps,
|
||||
if (rc < 0) {
|
||||
usb_host_libusb_error("libusb_alloc_streams", rc);
|
||||
} else if (rc != streams) {
|
||||
fprintf(stderr,
|
||||
"libusb_alloc_streams: got less streams then requested %d < %d\n",
|
||||
rc, streams);
|
||||
error_report("libusb_alloc_streams: got less streams "
|
||||
"then requested %d < %d", rc, streams);
|
||||
}
|
||||
|
||||
return (rc == streams) ? 0 : -1;
|
||||
#else
|
||||
fprintf(stderr, "libusb_alloc_streams: error not implemented\n");
|
||||
error_report("libusb_alloc_streams: error not implemented");
|
||||
return -1;
|
||||
#endif
|
||||
}
|
||||
@@ -1477,7 +1478,7 @@ static void usb_host_class_initfn(ObjectClass *klass, void *data)
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usb_host_initfn;
|
||||
uc->realize = usb_host_realize;
|
||||
uc->product_desc = "USB Host Device";
|
||||
uc->cancel_packet = usb_host_cancel_packet;
|
||||
uc->handle_data = usb_host_handle_data;
|
||||
|
@@ -1256,6 +1256,7 @@ static void usbredir_device_reject_bh(void *opaque)
|
||||
static void usbredir_do_attach(void *opaque)
|
||||
{
|
||||
USBRedirDevice *dev = opaque;
|
||||
Error *local_err = NULL;
|
||||
|
||||
/* In order to work properly with XHCI controllers we need these caps */
|
||||
if ((dev->dev.port->speedmask & USB_SPEED_MASK_SUPER) && !(
|
||||
@@ -1270,7 +1271,10 @@ static void usbredir_do_attach(void *opaque)
|
||||
return;
|
||||
}
|
||||
|
||||
if (usb_device_attach(&dev->dev) != 0) {
|
||||
usb_device_attach(&dev->dev, &local_err);
|
||||
if (local_err) {
|
||||
error_report("%s", error_get_pretty(local_err));
|
||||
error_free(local_err);
|
||||
WARNING("rejecting device due to speed mismatch\n");
|
||||
usbredir_reject_device(dev);
|
||||
}
|
||||
@@ -1357,14 +1361,14 @@ static void usbredir_init_endpoints(USBRedirDevice *dev)
|
||||
}
|
||||
}
|
||||
|
||||
static int usbredir_initfn(USBDevice *udev)
|
||||
static void usbredir_realize(USBDevice *udev, Error **errp)
|
||||
{
|
||||
USBRedirDevice *dev = DO_UPCAST(USBRedirDevice, dev, udev);
|
||||
int i;
|
||||
|
||||
if (dev->cs == NULL) {
|
||||
qerror_report(QERR_MISSING_PARAMETER, "chardev");
|
||||
return -1;
|
||||
error_set(errp, QERR_MISSING_PARAMETER, "chardev");
|
||||
return;
|
||||
}
|
||||
|
||||
if (dev->filter_str) {
|
||||
@@ -1372,9 +1376,9 @@ static int usbredir_initfn(USBDevice *udev)
|
||||
&dev->filter_rules,
|
||||
&dev->filter_rules_count);
|
||||
if (i) {
|
||||
qerror_report(QERR_INVALID_PARAMETER_VALUE, "filter",
|
||||
"a usb device filter string");
|
||||
return -1;
|
||||
error_set(errp, QERR_INVALID_PARAMETER_VALUE, "filter",
|
||||
"a usb device filter string");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1398,7 +1402,6 @@ static int usbredir_initfn(USBDevice *udev)
|
||||
|
||||
qemu_add_vm_change_state_handler(usbredir_vm_state_change, dev);
|
||||
add_boot_device_path(dev->bootindex, &udev->qdev, NULL);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void usbredir_cleanup_device_queues(USBRedirDevice *dev)
|
||||
@@ -2477,7 +2480,7 @@ static void usbredir_class_initfn(ObjectClass *klass, void *data)
|
||||
USBDeviceClass *uc = USB_DEVICE_CLASS(klass);
|
||||
DeviceClass *dc = DEVICE_CLASS(klass);
|
||||
|
||||
uc->init = usbredir_initfn;
|
||||
uc->realize = usbredir_realize;
|
||||
uc->product_desc = "USB Redirection Device";
|
||||
uc->handle_destroy = usbredir_handle_destroy;
|
||||
uc->cancel_packet = usbredir_cancel_packet;
|
||||
|
@@ -87,7 +87,7 @@ static void balloon_stats_destroy_timer(VirtIOBalloon *s)
|
||||
}
|
||||
}
|
||||
|
||||
static void balloon_stats_change_timer(VirtIOBalloon *s, int secs)
|
||||
static void balloon_stats_change_timer(VirtIOBalloon *s, int64_t secs)
|
||||
{
|
||||
timer_mod(s->stats_timer, qemu_clock_get_ms(QEMU_CLOCK_VIRTUAL) + secs * 1000);
|
||||
}
|
||||
@@ -170,6 +170,11 @@ static void balloon_stats_set_poll_interval(Object *obj, struct Visitor *v,
|
||||
return;
|
||||
}
|
||||
|
||||
if (value > UINT_MAX) {
|
||||
error_setg(errp, "timer value is too big");
|
||||
return;
|
||||
}
|
||||
|
||||
if (value == s->stats_poll_interval) {
|
||||
return;
|
||||
}
|
||||
|
@@ -926,6 +926,7 @@ static Property virtio_9p_pci_properties[] = {
|
||||
DEFINE_PROP_BIT("ioeventfd", VirtIOPCIProxy, flags,
|
||||
VIRTIO_PCI_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
DEFINE_PROP_UINT32("vectors", VirtIOPCIProxy, nvectors, 2),
|
||||
DEFINE_VIRTIO_9P_PROPERTIES(V9fsPCIState, vdev.fsconf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -947,9 +948,8 @@ static void virtio_9p_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_9p_pci_instance_init(Object *obj)
|
||||
{
|
||||
V9fsPCIState *dev = VIRTIO_9P_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_9P);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_9P);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static const TypeInfo virtio_9p_pci_info = {
|
||||
@@ -1111,9 +1111,10 @@ static void virtio_blk_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_blk_pci_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOBlkPCI *dev = VIRTIO_BLK_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_BLK);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_BLK);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_unref(OBJECT(&dev->vdev));
|
||||
qdev_alias_all_properties(DEVICE(&dev->vdev), obj);
|
||||
object_property_add_alias(obj, "iothread", OBJECT(&dev->vdev),"iothread",
|
||||
&error_abort);
|
||||
}
|
||||
@@ -1134,6 +1135,7 @@ static Property virtio_scsi_pci_properties[] = {
|
||||
DEFINE_PROP_UINT32("vectors", VirtIOPCIProxy, nvectors,
|
||||
DEV_NVECTORS_UNSPECIFIED),
|
||||
DEFINE_VIRTIO_SCSI_FEATURES(VirtIOPCIProxy, host_features),
|
||||
DEFINE_VIRTIO_SCSI_PROPERTIES(VirtIOSCSIPCI, vdev.parent_obj.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1183,11 +1185,8 @@ static void virtio_scsi_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_scsi_pci_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSCSIPCI *dev = VIRTIO_SCSI_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SCSI);
|
||||
object_property_add_alias(obj, "iothread", OBJECT(&dev->vdev), "iothread",
|
||||
&error_abort);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static const TypeInfo virtio_scsi_pci_info = {
|
||||
@@ -1204,6 +1203,7 @@ static const TypeInfo virtio_scsi_pci_info = {
|
||||
static Property vhost_scsi_pci_properties[] = {
|
||||
DEFINE_PROP_UINT32("vectors", VirtIOPCIProxy, nvectors,
|
||||
DEV_NVECTORS_UNSPECIFIED),
|
||||
DEFINE_VHOST_SCSI_PROPERTIES(VHostSCSIPCI, vdev.parent_obj.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1241,9 +1241,8 @@ static void vhost_scsi_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void vhost_scsi_pci_instance_init(Object *obj)
|
||||
{
|
||||
VHostSCSIPCI *dev = VHOST_SCSI_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VHOST_SCSI);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VHOST_SCSI);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static const TypeInfo vhost_scsi_pci_info = {
|
||||
@@ -1324,7 +1323,7 @@ static void virtio_balloon_pci_instance_init(Object *obj)
|
||||
VirtIOBalloonPCI *dev = VIRTIO_BALLOON_PCI(obj);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_BALLOON);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_unref(OBJECT(&dev->vdev));
|
||||
|
||||
object_property_add(obj, "guest-stats", "guest statistics",
|
||||
balloon_pci_stats_get_all, NULL, NULL, dev,
|
||||
NULL);
|
||||
@@ -1386,6 +1385,7 @@ static Property virtio_serial_pci_properties[] = {
|
||||
VIRTIO_PCI_FLAG_USE_IOEVENTFD_BIT, true),
|
||||
DEFINE_PROP_UINT32("vectors", VirtIOPCIProxy, nvectors, 2),
|
||||
DEFINE_PROP_UINT32("class", VirtIOPCIProxy, class_code, 0),
|
||||
DEFINE_VIRTIO_SERIAL_PROPERTIES(VirtIOSerialPCI, vdev.serial),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1406,9 +1406,8 @@ static void virtio_serial_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_serial_pci_instance_init(Object *obj)
|
||||
{
|
||||
VirtIOSerialPCI *dev = VIRTIO_SERIAL_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_SERIAL);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_SERIAL);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static const TypeInfo virtio_serial_pci_info = {
|
||||
@@ -1426,6 +1425,8 @@ static Property virtio_net_properties[] = {
|
||||
VIRTIO_PCI_FLAG_USE_IOEVENTFD_BIT, false),
|
||||
DEFINE_PROP_UINT32("vectors", VirtIOPCIProxy, nvectors, 3),
|
||||
DEFINE_VIRTIO_NET_FEATURES(VirtIOPCIProxy, host_features),
|
||||
DEFINE_NIC_PROPERTIES(VirtIONetPCI, vdev.nic_conf),
|
||||
DEFINE_VIRTIO_NET_PROPERTIES(VirtIONetPCI, vdev.net_conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1464,9 +1465,8 @@ static void virtio_net_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_net_pci_instance_init(Object *obj)
|
||||
{
|
||||
VirtIONetPCI *dev = VIRTIO_NET_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_NET);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_NET);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
}
|
||||
|
||||
static const TypeInfo virtio_net_pci_info = {
|
||||
@@ -1480,6 +1480,7 @@ static const TypeInfo virtio_net_pci_info = {
|
||||
/* virtio-rng-pci */
|
||||
|
||||
static Property virtio_rng_pci_properties[] = {
|
||||
DEFINE_VIRTIO_RNG_PROPERTIES(VirtIORngPCI, vdev.conf),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
@@ -1519,9 +1520,8 @@ static void virtio_rng_pci_class_init(ObjectClass *klass, void *data)
|
||||
static void virtio_rng_initfn(Object *obj)
|
||||
{
|
||||
VirtIORngPCI *dev = VIRTIO_RNG_PCI(obj);
|
||||
|
||||
virtio_instance_init_common(obj, &dev->vdev, sizeof(dev->vdev),
|
||||
TYPE_VIRTIO_RNG);
|
||||
object_initialize(&dev->vdev, sizeof(dev->vdev), TYPE_VIRTIO_RNG);
|
||||
object_property_add_child(obj, "virtio-backend", OBJECT(&dev->vdev), NULL);
|
||||
object_property_add_link(obj, "rng", TYPE_RNG_BACKEND,
|
||||
(Object **)&dev->vdev.conf.rng,
|
||||
qdev_prop_allow_set_link_before_realize,
|
||||
|
@@ -1123,17 +1123,6 @@ static void virtio_vmstate_change(void *opaque, int running, RunState state)
|
||||
}
|
||||
}
|
||||
|
||||
void virtio_instance_init_common(Object *proxy_obj, void *data,
|
||||
size_t vdev_size, const char *vdev_name)
|
||||
{
|
||||
DeviceState *vdev = data;
|
||||
|
||||
object_initialize(vdev, vdev_size, vdev_name);
|
||||
object_property_add_child(proxy_obj, "virtio-backend", OBJECT(vdev), NULL);
|
||||
object_unref(OBJECT(vdev));
|
||||
qdev_alias_all_properties(vdev, proxy_obj);
|
||||
}
|
||||
|
||||
void virtio_init(VirtIODevice *vdev, const char *name,
|
||||
uint16_t device_id, size_t config_size)
|
||||
{
|
||||
|
@@ -28,8 +28,6 @@
|
||||
#define SCLP_UNASSIGN_STORAGE 0x000C0001
|
||||
#define SCLP_CMD_READ_EVENT_DATA 0x00770005
|
||||
#define SCLP_CMD_WRITE_EVENT_DATA 0x00760005
|
||||
#define SCLP_CMD_READ_EVENT_DATA 0x00770005
|
||||
#define SCLP_CMD_WRITE_EVENT_DATA 0x00760005
|
||||
#define SCLP_CMD_WRITE_EVENT_MASK 0x00780005
|
||||
|
||||
/* SCLP Memory hotplug codes */
|
||||
|
@@ -5,7 +5,6 @@
|
||||
#include "block/block.h"
|
||||
#include "hw/block/block.h"
|
||||
#include "sysemu/sysemu.h"
|
||||
#include "qemu/notify.h"
|
||||
|
||||
#define MAX_SCSI_DEVS 255
|
||||
|
||||
@@ -51,25 +50,17 @@ struct SCSIRequest {
|
||||
uint32_t tag;
|
||||
uint32_t lun;
|
||||
uint32_t status;
|
||||
void *hba_private;
|
||||
size_t resid;
|
||||
SCSICommand cmd;
|
||||
NotifierList cancel_notifiers;
|
||||
|
||||
/* Note:
|
||||
* - fields before sense are initialized by scsi_req_alloc;
|
||||
* - sense[] is uninitialized;
|
||||
* - fields after sense are memset to 0 by scsi_req_alloc.
|
||||
* */
|
||||
|
||||
uint8_t sense[SCSI_SENSE_BUF_SIZE];
|
||||
uint32_t sense_len;
|
||||
bool enqueued;
|
||||
bool io_canceled;
|
||||
bool retry;
|
||||
bool dma_started;
|
||||
BlockDriverAIOCB *aiocb;
|
||||
QEMUSGList *sg;
|
||||
bool dma_started;
|
||||
uint8_t sense[SCSI_SENSE_BUF_SIZE];
|
||||
uint32_t sense_len;
|
||||
bool enqueued;
|
||||
bool io_canceled;
|
||||
bool retry;
|
||||
void *hba_private;
|
||||
QTAILQ_ENTRY(SCSIRequest) next;
|
||||
};
|
||||
|
||||
@@ -132,6 +123,7 @@ struct SCSIReqOps {
|
||||
int32_t (*send_command)(SCSIRequest *req, uint8_t *buf);
|
||||
void (*read_data)(SCSIRequest *req);
|
||||
void (*write_data)(SCSIRequest *req);
|
||||
void (*cancel_io)(SCSIRequest *req);
|
||||
uint8_t *(*get_buf)(SCSIRequest *req);
|
||||
|
||||
void (*save_request)(QEMUFile *f, SCSIRequest *req);
|
||||
@@ -266,9 +258,8 @@ void scsi_req_data(SCSIRequest *req, int len);
|
||||
void scsi_req_complete(SCSIRequest *req, int status);
|
||||
uint8_t *scsi_req_get_buf(SCSIRequest *req);
|
||||
int scsi_req_get_sense(SCSIRequest *req, uint8_t *buf, int len);
|
||||
void scsi_req_cancel_complete(SCSIRequest *req);
|
||||
void scsi_req_abort(SCSIRequest *req, int status);
|
||||
void scsi_req_cancel(SCSIRequest *req);
|
||||
void scsi_req_cancel_async(SCSIRequest *req, Notifier *notifier);
|
||||
void scsi_req_retry(SCSIRequest *req);
|
||||
void scsi_device_purge_requests(SCSIDevice *sdev, SCSISense sense);
|
||||
void scsi_device_set_ua(SCSIDevice *sdev, SCSISense sense);
|
||||
|
@@ -267,10 +267,14 @@ struct USBDevice {
|
||||
#define USB_DEVICE_GET_CLASS(obj) \
|
||||
OBJECT_GET_CLASS(USBDeviceClass, (obj), TYPE_USB_DEVICE)
|
||||
|
||||
typedef void (*USBDeviceRealize)(USBDevice *dev, Error **errp);
|
||||
typedef void (*USBDeviceUnrealize)(USBDevice *dev, Error **errp);
|
||||
|
||||
typedef struct USBDeviceClass {
|
||||
DeviceClass parent_class;
|
||||
|
||||
int (*init)(USBDevice *dev);
|
||||
USBDeviceRealize realize;
|
||||
USBDeviceUnrealize unrealize;
|
||||
|
||||
/*
|
||||
* Walk (enabled) downstream ports, check for a matching device.
|
||||
@@ -544,11 +548,12 @@ int usb_register_companion(const char *masterbus, USBPort *ports[],
|
||||
void *opaque, USBPortOps *ops, int speedmask);
|
||||
void usb_port_location(USBPort *downstream, USBPort *upstream, int portnr);
|
||||
void usb_unregister_port(USBBus *bus, USBPort *port);
|
||||
int usb_claim_port(USBDevice *dev);
|
||||
void usb_claim_port(USBDevice *dev, Error **errp);
|
||||
void usb_release_port(USBDevice *dev);
|
||||
int usb_device_attach(USBDevice *dev);
|
||||
void usb_device_attach(USBDevice *dev, Error **errp);
|
||||
int usb_device_detach(USBDevice *dev);
|
||||
int usb_device_delete_addr(int busnr, int addr);
|
||||
void usb_check_attach(USBDevice *dev, Error **errp);
|
||||
|
||||
static inline USBBus *usb_bus_from_device(USBDevice *d)
|
||||
{
|
||||
|
@@ -17,8 +17,6 @@
|
||||
#include "hw/virtio/virtio.h"
|
||||
#include "hw/pci/pci.h"
|
||||
#include "hw/scsi/scsi.h"
|
||||
#include "sysemu/iothread.h"
|
||||
#include "hw/virtio/dataplane/vring.h"
|
||||
|
||||
#define TYPE_VIRTIO_SCSI_COMMON "virtio-scsi-common"
|
||||
#define VIRTIO_SCSI_COMMON(obj) \
|
||||
@@ -153,18 +151,8 @@ struct VirtIOSCSIConf {
|
||||
uint32_t cmd_per_lun;
|
||||
char *vhostfd;
|
||||
char *wwpn;
|
||||
IOThread *iothread;
|
||||
};
|
||||
|
||||
struct VirtIOSCSI;
|
||||
|
||||
typedef struct {
|
||||
struct VirtIOSCSI *parent;
|
||||
Vring vring;
|
||||
EventNotifier host_notifier;
|
||||
EventNotifier guest_notifier;
|
||||
} VirtIOSCSIVring;
|
||||
|
||||
typedef struct VirtIOSCSICommon {
|
||||
VirtIODevice parent_obj;
|
||||
VirtIOSCSIConf conf;
|
||||
@@ -176,74 +164,14 @@ typedef struct VirtIOSCSICommon {
|
||||
VirtQueue **cmd_vqs;
|
||||
} VirtIOSCSICommon;
|
||||
|
||||
typedef struct VirtIOSCSI {
|
||||
typedef struct {
|
||||
VirtIOSCSICommon parent_obj;
|
||||
|
||||
SCSIBus bus;
|
||||
int resetting;
|
||||
bool events_dropped;
|
||||
|
||||
/* Fields for dataplane below */
|
||||
AioContext *ctx; /* one iothread per virtio-scsi-pci for now */
|
||||
|
||||
/* Vring is used instead of vq in dataplane code, because of the underlying
|
||||
* memory layer thread safety */
|
||||
VirtIOSCSIVring *ctrl_vring;
|
||||
VirtIOSCSIVring *event_vring;
|
||||
VirtIOSCSIVring **cmd_vrings;
|
||||
bool dataplane_started;
|
||||
bool dataplane_starting;
|
||||
bool dataplane_stopping;
|
||||
bool dataplane_disabled;
|
||||
Notifier migration_state_notifier;
|
||||
} VirtIOSCSI;
|
||||
|
||||
typedef struct VirtIOSCSIReq {
|
||||
VirtIOSCSI *dev;
|
||||
VirtQueue *vq;
|
||||
QEMUSGList qsgl;
|
||||
QEMUIOVector resp_iov;
|
||||
|
||||
/* Note:
|
||||
* - fields before elem are initialized by virtio_scsi_init_req;
|
||||
* - elem is uninitialized at the time of allocation.
|
||||
* - fields after elem are zeroed by virtio_scsi_init_req.
|
||||
* */
|
||||
|
||||
VirtQueueElement elem;
|
||||
/* Set by dataplane code. */
|
||||
VirtIOSCSIVring *vring;
|
||||
|
||||
union {
|
||||
/* Used for two-stage request submission */
|
||||
QTAILQ_ENTRY(VirtIOSCSIReq) next;
|
||||
|
||||
/* Used for cancellation of request during TMFs */
|
||||
int remaining;
|
||||
};
|
||||
|
||||
SCSIRequest *sreq;
|
||||
size_t resp_size;
|
||||
enum SCSIXferMode mode;
|
||||
union {
|
||||
VirtIOSCSICmdResp cmd;
|
||||
VirtIOSCSICtrlTMFResp tmf;
|
||||
VirtIOSCSICtrlANResp an;
|
||||
VirtIOSCSIEvent event;
|
||||
} resp;
|
||||
union {
|
||||
struct {
|
||||
VirtIOSCSICmdReq cmd;
|
||||
uint8_t cdb[];
|
||||
} QEMU_PACKED;
|
||||
VirtIOSCSICtrlTMFReq tmf;
|
||||
VirtIOSCSICtrlANReq an;
|
||||
} req;
|
||||
} VirtIOSCSIReq;
|
||||
|
||||
QEMU_BUILD_BUG_ON(offsetof(VirtIOSCSIReq, req.cdb) !=
|
||||
offsetof(VirtIOSCSIReq, req.cmd) + sizeof(VirtIOSCSICmdReq));
|
||||
|
||||
#define DEFINE_VIRTIO_SCSI_PROPERTIES(_state, _conf_field) \
|
||||
DEFINE_PROP_UINT32("num_queues", _state, _conf_field.num_queues, 1), \
|
||||
DEFINE_PROP_UINT32("max_sectors", _state, _conf_field.max_sectors, 0xFFFF),\
|
||||
@@ -264,19 +192,5 @@ void virtio_scsi_common_realize(DeviceState *dev, Error **errp,
|
||||
HandleOutput cmd);
|
||||
|
||||
void virtio_scsi_common_unrealize(DeviceState *dev, Error **errp);
|
||||
void virtio_scsi_handle_ctrl_req(VirtIOSCSI *s, VirtIOSCSIReq *req);
|
||||
bool virtio_scsi_handle_cmd_req_prepare(VirtIOSCSI *s, VirtIOSCSIReq *req);
|
||||
void virtio_scsi_handle_cmd_req_submit(VirtIOSCSI *s, VirtIOSCSIReq *req);
|
||||
VirtIOSCSIReq *virtio_scsi_init_req(VirtIOSCSI *s, VirtQueue *vq);
|
||||
void virtio_scsi_free_req(VirtIOSCSIReq *req);
|
||||
void virtio_scsi_push_event(VirtIOSCSI *s, SCSIDevice *dev,
|
||||
uint32_t event, uint32_t reason);
|
||||
|
||||
void virtio_scsi_set_iothread(VirtIOSCSI *s, IOThread *iothread);
|
||||
void virtio_scsi_dataplane_start(VirtIOSCSI *s);
|
||||
void virtio_scsi_dataplane_stop(VirtIOSCSI *s);
|
||||
void virtio_scsi_vring_push_notify(VirtIOSCSIReq *req);
|
||||
VirtIOSCSIReq *virtio_scsi_pop_req_vring(VirtIOSCSI *s,
|
||||
VirtIOSCSIVring *vring);
|
||||
|
||||
#endif /* _QEMU_VIRTIO_SCSI_H */
|
||||
|
@@ -161,9 +161,6 @@ typedef struct VirtioDeviceClass {
|
||||
int (*load)(VirtIODevice *vdev, QEMUFile *f, int version_id);
|
||||
} VirtioDeviceClass;
|
||||
|
||||
void virtio_instance_init_common(Object *proxy_obj, void *data,
|
||||
size_t vdev_size, const char *vdev_name);
|
||||
|
||||
void virtio_init(VirtIODevice *vdev, const char *name,
|
||||
uint16_t device_id, size_t config_size);
|
||||
void virtio_cleanup(VirtIODevice *vdev);
|
||||
|
@@ -139,8 +139,8 @@ static inline void vring_init(struct vring *vr, unsigned int num, void *p,
|
||||
vr->num = num;
|
||||
vr->desc = p;
|
||||
vr->avail = p + num*sizeof(struct vring_desc);
|
||||
vr->used = (void *)(((unsigned long)&vr->avail->ring[num] + sizeof(uint16_t)
|
||||
+ align-1) & ~(align - 1));
|
||||
vr->used = (void *)(((uintptr_t)&vr->avail->ring[num] + sizeof(uint16_t)
|
||||
+ align - 1) & ~(align - 1));
|
||||
}
|
||||
|
||||
static inline unsigned vring_size(unsigned int num, unsigned long align)
|
||||
|
@@ -154,16 +154,4 @@ void qerror_report_err(Error *err);
|
||||
#define QERR_UNSUPPORTED \
|
||||
ERROR_CLASS_GENERIC_ERROR, "this feature or command is not currently supported"
|
||||
|
||||
#define QERR_SOCKET_CONNECT_FAILED \
|
||||
ERROR_CLASS_GENERIC_ERROR, "Failed to connect to socket"
|
||||
|
||||
#define QERR_SOCKET_LISTEN_FAILED \
|
||||
ERROR_CLASS_GENERIC_ERROR, "Failed to set socket to listening mode"
|
||||
|
||||
#define QERR_SOCKET_BIND_FAILED \
|
||||
ERROR_CLASS_GENERIC_ERROR, "Failed to bind socket"
|
||||
|
||||
#define QERR_SOCKET_CREATE_FAILED \
|
||||
ERROR_CLASS_GENERIC_ERROR, "Failed to create socket"
|
||||
|
||||
#endif /* QERROR_H */
|
||||
|
@@ -55,6 +55,8 @@ struct Visitor
|
||||
void (*type_int64)(Visitor *v, int64_t *obj, const char *name, Error **errp);
|
||||
/* visit_type_size() falls back to (*type_uint64)() if type_size is unset */
|
||||
void (*type_size)(Visitor *v, uint64_t *obj, const char *name, Error **errp);
|
||||
bool (*start_union)(Visitor *v, bool data_present, Error **errp);
|
||||
void (*end_union)(Visitor *v, bool data_present, Error **errp);
|
||||
};
|
||||
|
||||
void input_type_enum(Visitor *v, int *obj, const char *strings[],
|
||||
|
@@ -58,5 +58,7 @@ void visit_type_size(Visitor *v, uint64_t *obj, const char *name, Error **errp);
|
||||
void visit_type_bool(Visitor *v, bool *obj, const char *name, Error **errp);
|
||||
void visit_type_str(Visitor *v, char **obj, const char *name, Error **errp);
|
||||
void visit_type_number(Visitor *v, double *obj, const char *name, Error **errp);
|
||||
bool visit_start_union(Visitor *v, bool data_present, Error **errp);
|
||||
void visit_end_union(Visitor *v, bool data_present, Error **errp);
|
||||
|
||||
#endif
|
||||
|
@@ -88,19 +88,10 @@ int slow_bitmap_andnot(unsigned long *dst, const unsigned long *bitmap1,
|
||||
int slow_bitmap_intersects(const unsigned long *bitmap1,
|
||||
const unsigned long *bitmap2, long bits);
|
||||
|
||||
static inline unsigned long *bitmap_try_new(long nbits)
|
||||
{
|
||||
long len = BITS_TO_LONGS(nbits) * sizeof(unsigned long);
|
||||
return g_try_malloc0(len);
|
||||
}
|
||||
|
||||
static inline unsigned long *bitmap_new(long nbits)
|
||||
{
|
||||
unsigned long *ptr = bitmap_try_new(nbits);
|
||||
if (ptr == NULL) {
|
||||
abort();
|
||||
}
|
||||
return ptr;
|
||||
long len = BITS_TO_LONGS(nbits) * sizeof(unsigned long);
|
||||
return g_malloc0(len);
|
||||
}
|
||||
|
||||
static inline void bitmap_zero(unsigned long *dst, long nbits)
|
||||
|
@@ -103,6 +103,7 @@ typedef int (*qemu_opt_loopfunc)(const char *name, const char *value, void *opaq
|
||||
int qemu_opt_foreach(QemuOpts *opts, qemu_opt_loopfunc func, void *opaque,
|
||||
int abort_on_failure);
|
||||
|
||||
int qemu_opts_id_wellformed(const char *id);
|
||||
QemuOpts *qemu_opts_find(QemuOptsList *list, const char *id);
|
||||
QemuOpts *qemu_opts_create(QemuOptsList *list, const char *id,
|
||||
int fail_if_exists, Error **errp);
|
||||
|
@@ -99,6 +99,9 @@ struct TranslationBlock;
|
||||
* @vmsd: State description for migration.
|
||||
* @gdb_num_core_regs: Number of core registers accessible to GDB.
|
||||
* @gdb_core_xml_file: File name for core registers GDB XML description.
|
||||
* @cpu_exec_enter: Callback for cpu_exec preparation.
|
||||
* @cpu_exec_exit: Callback for cpu_exec cleanup.
|
||||
* @cpu_exec_interrupt: Callback for processing interrupts in cpu_exec.
|
||||
*
|
||||
* Represents a CPU family or model.
|
||||
*/
|
||||
@@ -149,6 +152,10 @@ typedef struct CPUClass {
|
||||
const struct VMStateDescription *vmsd;
|
||||
int gdb_num_core_regs;
|
||||
const char *gdb_core_xml_file;
|
||||
|
||||
void (*cpu_exec_enter)(CPUState *cpu);
|
||||
void (*cpu_exec_exit)(CPUState *cpu);
|
||||
bool (*cpu_exec_interrupt)(CPUState *cpu, int interrupt_request);
|
||||
} CPUClass;
|
||||
|
||||
#ifdef HOST_WORDS_BIGENDIAN
|
||||
|
@@ -56,6 +56,7 @@ QemuOpts *drive_add(BlockInterfaceType type, int index, const char *file,
|
||||
const char *optstr);
|
||||
DriveInfo *drive_new(QemuOpts *arg, BlockInterfaceType block_default_type);
|
||||
void drive_del(DriveInfo *dinfo);
|
||||
void drive_info_del(DriveInfo *dinfo);
|
||||
|
||||
/* device-hotplug */
|
||||
|
||||
|
@@ -10,6 +10,7 @@ void cpu_stop_current(void);
|
||||
void cpu_synchronize_all_states(void);
|
||||
void cpu_synchronize_all_post_reset(void);
|
||||
void cpu_synchronize_all_post_init(void);
|
||||
void cpu_clean_all_dirty(void);
|
||||
|
||||
void qtest_clock_warp(int64_t dest);
|
||||
|
||||
|
@@ -350,6 +350,7 @@ int kvm_physical_memory_addr_from_host(KVMState *s, void *ram_addr,
|
||||
void kvm_cpu_synchronize_state(CPUState *cpu);
|
||||
void kvm_cpu_synchronize_post_reset(CPUState *cpu);
|
||||
void kvm_cpu_synchronize_post_init(CPUState *cpu);
|
||||
void kvm_cpu_clean_state(CPUState *cpu);
|
||||
|
||||
/* generic hooks - to be moved/refactored once there are more users */
|
||||
|
||||
@@ -374,6 +375,13 @@ static inline void cpu_synchronize_post_init(CPUState *cpu)
|
||||
}
|
||||
}
|
||||
|
||||
static inline void cpu_clean_state(CPUState *cpu)
|
||||
{
|
||||
if (kvm_enabled()) {
|
||||
kvm_cpu_clean_state(cpu);
|
||||
}
|
||||
}
|
||||
|
||||
int kvm_irqchip_add_msi_route(KVMState *s, MSIMessage msg);
|
||||
int kvm_irqchip_update_msi_route(KVMState *s, int virq, MSIMessage msg);
|
||||
void kvm_irqchip_release_virq(KVMState *s, int virq);
|
||||
|
@@ -292,6 +292,9 @@ typedef struct GraphicHwOps {
|
||||
QemuConsole *graphic_console_init(DeviceState *dev, uint32_t head,
|
||||
const GraphicHwOps *ops,
|
||||
void *opaque);
|
||||
void graphic_console_set_hwops(QemuConsole *con,
|
||||
const GraphicHwOps *hw_ops,
|
||||
void *opaque);
|
||||
|
||||
void graphic_hw_update(QemuConsole *con);
|
||||
void graphic_hw_invalidate(QemuConsole *con);
|
||||
|
12
kvm-all.c
12
kvm-all.c
@@ -44,10 +44,6 @@
|
||||
#include <sys/eventfd.h>
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_VALGRIND_H
|
||||
#include <valgrind/memcheck.h>
|
||||
#endif
|
||||
|
||||
/* KVM uses PAGE_SIZE in its definition of COALESCED_MMIO_MAX */
|
||||
#define PAGE_SIZE TARGET_PAGE_SIZE
|
||||
|
||||
@@ -1708,6 +1704,11 @@ void kvm_cpu_synchronize_post_init(CPUState *cpu)
|
||||
run_on_cpu(cpu, do_kvm_cpu_synchronize_post_init, cpu);
|
||||
}
|
||||
|
||||
void kvm_cpu_clean_state(CPUState *cpu)
|
||||
{
|
||||
cpu->kvm_vcpu_dirty = false;
|
||||
}
|
||||
|
||||
int kvm_cpu_exec(CPUState *cpu)
|
||||
{
|
||||
struct kvm_run *run = cpu->kvm_run;
|
||||
@@ -1954,9 +1955,6 @@ int kvm_has_intx_set_mask(void)
|
||||
|
||||
void kvm_setup_guest_memory(void *start, size_t size)
|
||||
{
|
||||
#ifdef CONFIG_VALGRIND_H
|
||||
VALGRIND_MAKE_MEM_DEFINED(start, size);
|
||||
#endif
|
||||
if (!kvm_has_sync_mmu()) {
|
||||
int ret = qemu_madvise(start, size, QEMU_MADV_DONTFORK);
|
||||
|
||||
|
@@ -84,6 +84,9 @@ static int qemu_signal_init(void)
|
||||
sigaddset(&set, SIGIO);
|
||||
sigaddset(&set, SIGALRM);
|
||||
sigaddset(&set, SIGBUS);
|
||||
sigaddset(&set, SIGINT);
|
||||
sigaddset(&set, SIGHUP);
|
||||
sigaddset(&set, SIGTERM);
|
||||
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
||||
|
||||
sigdelset(&set, SIG_IPI);
|
||||
|
35
monitor.c
35
monitor.c
@@ -886,19 +886,12 @@ static void do_trace_event_set_state(Monitor *mon, const QDict *qdict)
|
||||
{
|
||||
const char *tp_name = qdict_get_str(qdict, "name");
|
||||
bool new_state = qdict_get_bool(qdict, "option");
|
||||
Error *local_err = NULL;
|
||||
|
||||
bool found = false;
|
||||
TraceEvent *ev = NULL;
|
||||
while ((ev = trace_event_pattern(tp_name, ev)) != NULL) {
|
||||
found = true;
|
||||
if (!trace_event_get_state_static(ev)) {
|
||||
monitor_printf(mon, "event \"%s\" is not traceable\n", tp_name);
|
||||
} else {
|
||||
trace_event_set_state_dynamic(ev, new_state);
|
||||
}
|
||||
}
|
||||
if (!trace_event_is_pattern(tp_name) && !found) {
|
||||
monitor_printf(mon, "unknown event name \"%s\"\n", tp_name);
|
||||
qmp_trace_event_set_state(tp_name, new_state, true, true, &local_err);
|
||||
if (local_err) {
|
||||
qerror_report_err(local_err);
|
||||
error_free(local_err);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1079,7 +1072,15 @@ static void do_info_cpu_stats(Monitor *mon, const QDict *qdict)
|
||||
|
||||
static void do_trace_print_events(Monitor *mon, const QDict *qdict)
|
||||
{
|
||||
trace_print_events((FILE *)mon, &monitor_fprintf);
|
||||
TraceEventInfoList *events = qmp_trace_event_get_state("*", NULL);
|
||||
TraceEventInfoList *elem;
|
||||
|
||||
for (elem = events; elem != NULL; elem = elem->next) {
|
||||
monitor_printf(mon, "%s : state %u\n",
|
||||
elem->value->name,
|
||||
elem->value->state == TRACE_EVENT_STATE_ENABLED ? 1 : 0);
|
||||
}
|
||||
qapi_free_TraceEventInfoList(events);
|
||||
}
|
||||
|
||||
static int client_migrate_info(Monitor *mon, const QDict *qdict,
|
||||
@@ -2920,6 +2921,13 @@ static mon_cmd_t info_cmds[] = {
|
||||
.help = "show memory backends",
|
||||
.mhandler.cmd = hmp_info_memdev,
|
||||
},
|
||||
{
|
||||
.name = "memory-devices",
|
||||
.args_type = "",
|
||||
.params = "",
|
||||
.help = "show memory devices",
|
||||
.mhandler.cmd = hmp_info_memory_devices,
|
||||
},
|
||||
{
|
||||
.name = NULL,
|
||||
},
|
||||
@@ -5248,6 +5256,7 @@ static void monitor_event(void *opaque, int event)
|
||||
monitor_printf(mon, "QEMU %s monitor - type 'help' for more "
|
||||
"information\n", QEMU_VERSION);
|
||||
if (!mon->mux_out) {
|
||||
readline_restart(mon->rs);
|
||||
readline_show_prompt(mon->rs);
|
||||
}
|
||||
mon->reset_seen = 1;
|
||||
|
97
os-posix.c
97
os-posix.c
@@ -204,45 +204,49 @@ static void change_root(void)
|
||||
void os_daemonize(void)
|
||||
{
|
||||
if (daemonize) {
|
||||
pid_t pid;
|
||||
pid_t pid;
|
||||
|
||||
if (pipe(fds) == -1)
|
||||
exit(1);
|
||||
|
||||
pid = fork();
|
||||
if (pid > 0) {
|
||||
uint8_t status;
|
||||
ssize_t len;
|
||||
|
||||
close(fds[1]);
|
||||
|
||||
again:
|
||||
len = read(fds[0], &status, 1);
|
||||
if (len == -1 && (errno == EINTR))
|
||||
goto again;
|
||||
|
||||
if (len != 1)
|
||||
exit(1);
|
||||
else if (status == 1) {
|
||||
fprintf(stderr, "Could not acquire pidfile: %s\n", strerror(errno));
|
||||
exit(1);
|
||||
} else
|
||||
exit(0);
|
||||
} else if (pid < 0)
|
||||
if (pipe(fds) == -1) {
|
||||
exit(1);
|
||||
}
|
||||
|
||||
close(fds[0]);
|
||||
qemu_set_cloexec(fds[1]);
|
||||
pid = fork();
|
||||
if (pid > 0) {
|
||||
uint8_t status;
|
||||
ssize_t len;
|
||||
|
||||
setsid();
|
||||
close(fds[1]);
|
||||
|
||||
pid = fork();
|
||||
if (pid > 0)
|
||||
exit(0);
|
||||
else if (pid < 0)
|
||||
exit(1);
|
||||
again:
|
||||
len = read(fds[0], &status, 1);
|
||||
if (len == -1 && (errno == EINTR)) {
|
||||
goto again;
|
||||
}
|
||||
if (len != 1) {
|
||||
exit(1);
|
||||
}
|
||||
else if (status == 1) {
|
||||
fprintf(stderr, "Could not acquire pidfile\n");
|
||||
exit(1);
|
||||
} else {
|
||||
exit(0);
|
||||
}
|
||||
} else if (pid < 0) {
|
||||
exit(1);
|
||||
}
|
||||
|
||||
umask(027);
|
||||
close(fds[0]);
|
||||
qemu_set_cloexec(fds[1]);
|
||||
|
||||
setsid();
|
||||
|
||||
pid = fork();
|
||||
if (pid > 0) {
|
||||
exit(0);
|
||||
} else if (pid < 0) {
|
||||
exit(1);
|
||||
}
|
||||
umask(027);
|
||||
|
||||
signal(SIGTSTP, SIG_IGN);
|
||||
signal(SIGTTOU, SIG_IGN);
|
||||
@@ -255,24 +259,25 @@ void os_setup_post(void)
|
||||
int fd = 0;
|
||||
|
||||
if (daemonize) {
|
||||
uint8_t status = 0;
|
||||
ssize_t len;
|
||||
uint8_t status = 0;
|
||||
ssize_t len;
|
||||
|
||||
again1:
|
||||
len = write(fds[1], &status, 1);
|
||||
if (len == -1 && (errno == EINTR))
|
||||
goto again1;
|
||||
|
||||
if (len != 1)
|
||||
exit(1);
|
||||
|
||||
len = write(fds[1], &status, 1);
|
||||
if (len == -1 && (errno == EINTR)) {
|
||||
goto again1;
|
||||
}
|
||||
if (len != 1) {
|
||||
exit(1);
|
||||
}
|
||||
if (chdir("/")) {
|
||||
perror("not able to chdir to /");
|
||||
exit(1);
|
||||
}
|
||||
TFR(fd = qemu_open("/dev/null", O_RDWR));
|
||||
if (fd == -1)
|
||||
exit(1);
|
||||
TFR(fd = qemu_open("/dev/null", O_RDWR));
|
||||
if (fd == -1) {
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
change_root();
|
||||
@@ -314,6 +319,8 @@ int qemu_create_pidfile(const char *filename)
|
||||
return -1;
|
||||
}
|
||||
if (lockf(fd, F_TLOCK, 0) == -1) {
|
||||
fprintf(stderr, "lock file '%s' failed: %s\n",
|
||||
filename, strerror(errno));
|
||||
close(fd);
|
||||
return -1;
|
||||
}
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -9,6 +9,9 @@ all:
|
||||
|
||||
.PHONY: all build clean install update
|
||||
|
||||
%.mo: %.po
|
||||
$(call quiet-command, msgfmt -o $@ $<, " GEN $@")
|
||||
|
||||
-include ../config-host.mak
|
||||
include $(SRC_PATH)/rules.mak
|
||||
|
||||
@@ -38,9 +41,6 @@ install: $(OBJS)
|
||||
$(INSTALL) -m644 $$obj $(DESTDIR)$(prefix)/share/locale/$$base/LC_MESSAGES/qemu.mo; \
|
||||
done
|
||||
|
||||
%.mo: %.po
|
||||
$(call quiet-command, msgfmt -o $@ $<, " GEN $@")
|
||||
|
||||
$(PO_PATH)/messages.po: $(SRC_PATH)/ui/gtk.c
|
||||
$(call quiet-command, ( cd $(SRC_PATH) && \
|
||||
xgettext -o - --from-code=UTF-8 --foreign-user \
|
||||
|
@@ -11,6 +11,9 @@
|
||||
# QAPI event definitions
|
||||
{ 'include': 'qapi/event.json' }
|
||||
|
||||
# Tracing commands
|
||||
{ 'include': 'qapi/trace.json' }
|
||||
|
||||
##
|
||||
# LostTickPolicy:
|
||||
#
|
||||
@@ -3229,6 +3232,23 @@
|
||||
'rel' : 'InputMoveEvent',
|
||||
'abs' : 'InputMoveEvent' } }
|
||||
|
||||
##
|
||||
# @input-send-event
|
||||
#
|
||||
# Send input event(s) to guest.
|
||||
#
|
||||
# @console: Which console to send event(s) to.
|
||||
#
|
||||
# @events: List of InputEvent union.
|
||||
#
|
||||
# Returns: Nothing on success.
|
||||
#
|
||||
# Since: 2.2
|
||||
#
|
||||
##
|
||||
{ 'command': 'input-send-event',
|
||||
'data': { 'console':'int', 'events': [ 'InputEvent' ] } }
|
||||
|
||||
##
|
||||
# @NumaOptions
|
||||
#
|
||||
|
@@ -162,6 +162,31 @@ static void qapi_dealloc_type_enum(Visitor *v, int *obj, const char *strings[],
|
||||
{
|
||||
}
|
||||
|
||||
/* If there's no data present, the dealloc visitor has nothing to free.
|
||||
* Thus, indicate to visitor code that the subsequent union fields can
|
||||
* be skipped. This is not an error condition, since the cleanup of the
|
||||
* rest of an object can continue unhindered, so leave errp unset in
|
||||
* these cases.
|
||||
*
|
||||
* NOTE: In cases where we're attempting to deallocate an object that
|
||||
* may have missing fields, the field indicating the union type may
|
||||
* be missing. In such a case, it's possible we don't have enough
|
||||
* information to differentiate data_present == false from a case where
|
||||
* data *is* present but happens to be a scalar with a value of 0.
|
||||
* This is okay, since in the case of the dealloc visitor there's no
|
||||
* work that needs to done in either situation.
|
||||
*
|
||||
* The current inability in QAPI code to more thoroughly verify a union
|
||||
* type in such cases will likely need to be addressed if we wish to
|
||||
* implement this interface for other types of visitors in the future,
|
||||
* however.
|
||||
*/
|
||||
static bool qapi_dealloc_start_union(Visitor *v, bool data_present,
|
||||
Error **errp)
|
||||
{
|
||||
return data_present;
|
||||
}
|
||||
|
||||
Visitor *qapi_dealloc_get_visitor(QapiDeallocVisitor *v)
|
||||
{
|
||||
return &v->visitor;
|
||||
@@ -191,6 +216,7 @@ QapiDeallocVisitor *qapi_dealloc_visitor_new(void)
|
||||
v->visitor.type_str = qapi_dealloc_type_str;
|
||||
v->visitor.type_number = qapi_dealloc_type_number;
|
||||
v->visitor.type_size = qapi_dealloc_type_size;
|
||||
v->visitor.start_union = qapi_dealloc_start_union;
|
||||
|
||||
QTAILQ_INIT(&v->stack);
|
||||
|
||||
|
@@ -58,6 +58,21 @@ void visit_end_list(Visitor *v, Error **errp)
|
||||
v->end_list(v, errp);
|
||||
}
|
||||
|
||||
bool visit_start_union(Visitor *v, bool data_present, Error **errp)
|
||||
{
|
||||
if (v->start_union) {
|
||||
return v->start_union(v, data_present, errp);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
void visit_end_union(Visitor *v, bool data_present, Error **errp)
|
||||
{
|
||||
if (v->end_union) {
|
||||
v->end_union(v, data_present, errp);
|
||||
}
|
||||
}
|
||||
|
||||
void visit_optional(Visitor *v, bool *present, const char *name,
|
||||
Error **errp)
|
||||
{
|
||||
|
65
qapi/trace.json
Normal file
65
qapi/trace.json
Normal file
@@ -0,0 +1,65 @@
|
||||
# -*- mode: python -*-
|
||||
#
|
||||
# Copyright (C) 2011-2014 Lluís Vilanova <vilanova@ac.upc.edu>
|
||||
#
|
||||
# This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||
# See the COPYING file in the top-level directory.
|
||||
|
||||
|
||||
##
|
||||
# @TraceEventState:
|
||||
#
|
||||
# State of a tracing event.
|
||||
#
|
||||
# @unavailable: The event is statically disabled.
|
||||
#
|
||||
# @disabled: The event is dynamically disabled.
|
||||
#
|
||||
# @enabled: The event is dynamically enabled.
|
||||
#
|
||||
# Since 2.2
|
||||
##
|
||||
{ 'enum': 'TraceEventState',
|
||||
'data': ['unavailable', 'disabled', 'enabled'] }
|
||||
|
||||
##
|
||||
# @TraceEventInfo:
|
||||
#
|
||||
# Information of a tracing event.
|
||||
#
|
||||
# @name: Event name.
|
||||
# @state: Tracing state.
|
||||
#
|
||||
# Since 2.2
|
||||
##
|
||||
{ 'type': 'TraceEventInfo',
|
||||
'data': {'name': 'str', 'state': 'TraceEventState'} }
|
||||
|
||||
##
|
||||
# @trace-event-get-state:
|
||||
#
|
||||
# Query the state of events.
|
||||
#
|
||||
# @name: Event name pattern (case-sensitive glob).
|
||||
#
|
||||
# Returns: a list of @TraceEventInfo for the matching events
|
||||
#
|
||||
# Since 2.2
|
||||
##
|
||||
{ 'command': 'trace-event-get-state',
|
||||
'data': {'name': 'str'},
|
||||
'returns': ['TraceEventInfo'] }
|
||||
|
||||
##
|
||||
# @trace-event-set-state:
|
||||
#
|
||||
# Set the dynamic tracing state of events.
|
||||
#
|
||||
# @name: Event name pattern (case-sensitive glob).
|
||||
# @enable: Whether to enable tracing.
|
||||
# @ignore-unavailable: #optional Do not match unavailable events with @name.
|
||||
#
|
||||
# Since 2.2
|
||||
##
|
||||
{ 'command': 'trace-event-set-state',
|
||||
'data': {'name': 'str', 'enable': 'bool', '*ignore-unavailable': 'bool'} }
|
@@ -1011,14 +1011,14 @@ static int img_compare(int argc, char **argv)
|
||||
goto out3;
|
||||
}
|
||||
|
||||
bs1 = bdrv_new_open("image 1", filename1, fmt1, flags, true, quiet);
|
||||
bs1 = bdrv_new_open("image_1", filename1, fmt1, flags, true, quiet);
|
||||
if (!bs1) {
|
||||
error_report("Can't open file %s", filename1);
|
||||
ret = 2;
|
||||
goto out3;
|
||||
}
|
||||
|
||||
bs2 = bdrv_new_open("image 2", filename2, fmt2, flags, true, quiet);
|
||||
bs2 = bdrv_new_open("image_2", filename2, fmt2, flags, true, quiet);
|
||||
if (!bs2) {
|
||||
error_report("Can't open file %s", filename2);
|
||||
ret = 2;
|
||||
@@ -1359,7 +1359,7 @@ static int img_convert(int argc, char **argv)
|
||||
|
||||
total_sectors = 0;
|
||||
for (bs_i = 0; bs_i < bs_n; bs_i++) {
|
||||
char *id = bs_n > 1 ? g_strdup_printf("source %d", bs_i)
|
||||
char *id = bs_n > 1 ? g_strdup_printf("source_%d", bs_i)
|
||||
: g_strdup("source");
|
||||
bs[bs_i] = bdrv_new_open(id, argv[optind + bs_i], fmt, src_flags,
|
||||
true, quiet);
|
||||
|
@@ -773,7 +773,7 @@ int main(int argc, char **argv)
|
||||
}
|
||||
} while (state != TERMINATED);
|
||||
|
||||
bdrv_close(bs);
|
||||
bdrv_unref(bs);
|
||||
if (sockpath) {
|
||||
unlink(sockpath);
|
||||
}
|
||||
|
@@ -3752,5 +3752,103 @@ Example:
|
||||
|
||||
-> { "execute": "rtc-reset-reinjection" }
|
||||
<- { "return": {} }
|
||||
EQMP
|
||||
|
||||
{
|
||||
.name = "trace-event-get-state",
|
||||
.args_type = "name:s",
|
||||
.mhandler.cmd_new = qmp_marshal_input_trace_event_get_state,
|
||||
},
|
||||
|
||||
SQMP
|
||||
trace-event-get-state
|
||||
---------------------
|
||||
|
||||
Query the state of events.
|
||||
|
||||
Example:
|
||||
|
||||
-> { "execute": "trace-event-get-state", "arguments": { "name": "qemu_memalign" } }
|
||||
<- { "return": [ { "name": "qemu_memalign", "state": "disabled" } ] }
|
||||
EQMP
|
||||
|
||||
{
|
||||
.name = "trace-event-set-state",
|
||||
.args_type = "name:s,enable:b,ignore-unavailable:b?",
|
||||
.mhandler.cmd_new = qmp_marshal_input_trace_event_set_state,
|
||||
},
|
||||
|
||||
SQMP
|
||||
trace-event-set-state
|
||||
---------------------
|
||||
|
||||
Set the state of events.
|
||||
|
||||
Example:
|
||||
|
||||
-> { "execute": "trace-event-set-state", "arguments": { "name": "qemu_memalign", "enable": "true" } }
|
||||
<- { "return": {} }
|
||||
EQMP
|
||||
|
||||
{
|
||||
.name = "input-send-event",
|
||||
.args_type = "console:i,events:q",
|
||||
.mhandler.cmd_new = qmp_marshal_input_input_send_event,
|
||||
},
|
||||
|
||||
SQMP
|
||||
@input-send-event
|
||||
-----------------
|
||||
|
||||
Send input event to guest.
|
||||
|
||||
Arguments:
|
||||
|
||||
- "console": console index.
|
||||
- "events": list of input events.
|
||||
|
||||
The consoles are visible in the qom tree, under
|
||||
/backend/console[$index]. They have a device link and head property, so
|
||||
it is possible to map which console belongs to which device and display.
|
||||
|
||||
Example (1):
|
||||
|
||||
Press left mouse button.
|
||||
|
||||
-> { "execute": "input-send-event",
|
||||
"arguments": { "console": 0,
|
||||
"events": [ { "type": "btn",
|
||||
"data" : { "down": true, "button": "Left" } } } }
|
||||
<- { "return": {} }
|
||||
|
||||
-> { "execute": "input-send-event",
|
||||
"arguments": { "console": 0,
|
||||
"events": [ { "type": "btn",
|
||||
"data" : { "down": false, "button": "Left" } } } }
|
||||
<- { "return": {} }
|
||||
|
||||
Example (2):
|
||||
|
||||
Press ctrl-alt-del.
|
||||
|
||||
-> { "execute": "input-send-event",
|
||||
"arguments": { "console": 0, "events": [
|
||||
{ "type": "key", "data" : { "down": true,
|
||||
"key": {"type": "qcode", "data": "ctrl" } } },
|
||||
{ "type": "key", "data" : { "down": true,
|
||||
"key": {"type": "qcode", "data": "alt" } } },
|
||||
{ "type": "key", "data" : { "down": true,
|
||||
"key": {"type": "qcode", "data": "delete" } } } ] } }
|
||||
<- { "return": {} }
|
||||
|
||||
Example (3):
|
||||
|
||||
Move mouse pointer to absolute coordinates (20000, 400).
|
||||
|
||||
-> { "execute": "input-send-event" ,
|
||||
"arguments": { "console": 0, "events": [
|
||||
{ "type": "abs", "data" : { "axis": "X", "value" : 20000 } },
|
||||
{ "type": "abs", "data" : { "axis": "Y", "value" : 400 } } ] } }
|
||||
<- { "return": {} }
|
||||
|
||||
EQMP
|
||||
|
12
qom/cpu.c
12
qom/cpu.c
@@ -202,10 +202,15 @@ static bool cpu_common_virtio_is_big_endian(CPUState *cpu)
|
||||
return target_words_bigendian();
|
||||
}
|
||||
|
||||
static void cpu_common_debug_excp_handler(CPUState *cpu)
|
||||
static void cpu_common_noop(CPUState *cpu)
|
||||
{
|
||||
}
|
||||
|
||||
static bool cpu_common_exec_interrupt(CPUState *cpu, int int_req)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
void cpu_dump_state(CPUState *cpu, FILE *f, fprintf_function cpu_fprintf,
|
||||
int flags)
|
||||
{
|
||||
@@ -344,7 +349,10 @@ static void cpu_class_init(ObjectClass *klass, void *data)
|
||||
k->gdb_read_register = cpu_common_gdb_read_register;
|
||||
k->gdb_write_register = cpu_common_gdb_write_register;
|
||||
k->virtio_is_big_endian = cpu_common_virtio_is_big_endian;
|
||||
k->debug_excp_handler = cpu_common_debug_excp_handler;
|
||||
k->debug_excp_handler = cpu_common_noop;
|
||||
k->cpu_exec_enter = cpu_common_noop;
|
||||
k->cpu_exec_exit = cpu_common_noop;
|
||||
k->cpu_exec_interrupt = cpu_common_exec_interrupt;
|
||||
dc->realize = cpu_common_realizefn;
|
||||
/*
|
||||
* Reason: CPUs still need special care by board code: wiring up
|
||||
|
Submodule roms/openbios updated: d9e38ba2ff...5387e76474
@@ -25,7 +25,7 @@ sub out {
|
||||
|
||||
while (<>) {
|
||||
if (/^(disable )?([a-z_0-9]+)\(/) {
|
||||
open GREP, '-|', 'git', 'grep', '-l', "trace_$2"
|
||||
open GREP, '-|', 'git', 'grep', '-lw', "trace_$2"
|
||||
or die "run git grep: $!";
|
||||
my $fname;
|
||||
while ($fname = <GREP>) {
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user