|
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] [PATCH] x86/vPIT: check/bound values loaded from state save record
In particular pit_latch_status() and speaker_ioport_read() perform
calculations which assume in-bounds values. Several of the state save
record fields can hold wider ranges, though.
Note that ->gate should only be possible to be zero for channel 2;
enforce that as well.
Adjust pit_reset()'s writing of ->mode as well, to not unduly affect
the value pit_latch_status() may calculate. The chosen mode of 7 is
still one which cannot be established by writing the control word.
Signed-off-by: Jan Beulich <jbeulich@xxxxxxxx>
---
Of course an alternative would be to simply reject state save records
with out of bounds values.
--- a/xen/arch/x86/emul-i8254.c
+++ b/xen/arch/x86/emul-i8254.c
@@ -47,6 +47,7 @@
#define RW_STATE_MSB 2
#define RW_STATE_WORD0 3
#define RW_STATE_WORD1 4
+#define RW_STATE_NUM 5
static int cf_check handle_pit_io(
int dir, unsigned int port, unsigned int bytes, uint32_t *val);
@@ -426,6 +427,33 @@ static int cf_check pit_load(struct doma
}
/*
+ * Convert loaded values to be within valid range, for them to represent
+ * actually reachable state. Uses of some of the values elsewhere assume
+ * this is the case.
+ */
+ for ( i = 0; i < ARRAY_SIZE(pit->hw.channels); ++i )
+ {
+ struct hvm_hw_pit_channel *ch = &pit->hw.channels[i];
+
+ /* pit_load_count() will convert 0 suitably back to 0x10000. */
+ ch->count &= 0xffff;
+ if ( ch->count_latched >= RW_STATE_NUM )
+ ch->count_latched = 0;
+ if ( ch->read_state >= RW_STATE_NUM )
+ ch->read_state = 0;
+ if ( ch->read_state >= RW_STATE_NUM )
+ ch->write_state = 0;
+ if ( ch->rw_mode > RW_STATE_WORD0 )
+ ch->rw_mode = 0;
+ if ( (ch->mode &= 7) > 5 )
+ ch->mode -= 4;
+ ch->bcd = !!ch->bcd;
+ ch->gate = i != 2 || ch->gate;
+ }
+
+ pit->hw.speaker_data_on = !!pit->hw.speaker_data_on;
+
+ /*
* Recreate platform timers from hardware state. There will be some
* time jitter here, but the wall-clock will have jumped massively, so
* we hope the guest can handle it.
@@ -464,7 +492,7 @@ void pit_reset(struct domain *d)
for ( i = 0; i < 3; i++ )
{
s = &pit->hw.channels[i];
- s->mode = 0xff; /* the init mode */
+ s->mode = 7; /* the init mode */
s->gate = (i != 2);
pit_load_count(pit, i, 0);
}
|
![]() |
Lists.xenproject.org is hosted with RackSpace, monitoring our |