]>
Commit | Line | Data |
---|---|---|
306e196f PD |
1 | /* |
2 | * replay-snapshot.c | |
3 | * | |
4 | * Copyright (c) 2010-2016 Institute for System Programming | |
5 | * of the Russian Academy of Sciences. | |
6 | * | |
7 | * This work is licensed under the terms of the GNU GPL, version 2 or later. | |
8 | * See the COPYING file in the top-level directory. | |
9 | * | |
10 | */ | |
11 | ||
12 | #include "qemu/osdep.h" | |
13 | #include "qapi/error.h" | |
306e196f PD |
14 | #include "sysemu/replay.h" |
15 | #include "replay-internal.h" | |
306e196f PD |
16 | #include "monitor/monitor.h" |
17 | #include "qapi/qmp/qstring.h" | |
18 | #include "qemu/error-report.h" | |
19 | #include "migration/vmstate.h" | |
5e22479a | 20 | #include "migration/snapshot.h" |
306e196f | 21 | |
44b1ff31 | 22 | static int replay_pre_save(void *opaque) |
306e196f PD |
23 | { |
24 | ReplayState *state = opaque; | |
25 | state->file_offset = ftell(replay_file); | |
44b1ff31 DDAG |
26 | |
27 | return 0; | |
306e196f PD |
28 | } |
29 | ||
30 | static int replay_post_load(void *opaque, int version_id) | |
31 | { | |
32 | ReplayState *state = opaque; | |
bb3d7702 PD |
33 | if (replay_mode == REPLAY_MODE_PLAY) { |
34 | fseek(replay_file, state->file_offset, SEEK_SET); | |
bb3d7702 PD |
35 | /* If this was a vmstate, saved in recording mode, |
36 | we need to initialize replay data fields. */ | |
37 | replay_fetch_data_kind(); | |
38 | } else if (replay_mode == REPLAY_MODE_RECORD) { | |
39 | /* This is only useful for loading the initial state. | |
40 | Therefore reset all the counters. */ | |
13f26713 | 41 | state->instruction_count = 0; |
bb3d7702 PD |
42 | state->block_request_id = 0; |
43 | } | |
306e196f PD |
44 | |
45 | return 0; | |
46 | } | |
47 | ||
48 | static const VMStateDescription vmstate_replay = { | |
49 | .name = "replay", | |
a02fe2ca DDAG |
50 | .version_id = 2, |
51 | .minimum_version_id = 2, | |
306e196f PD |
52 | .pre_save = replay_pre_save, |
53 | .post_load = replay_post_load, | |
54 | .fields = (VMStateField[]) { | |
55 | VMSTATE_INT64_ARRAY(cached_clock, ReplayState, REPLAY_CLOCK_COUNT), | |
13f26713 PD |
56 | VMSTATE_UINT64(current_icount, ReplayState), |
57 | VMSTATE_INT32(instruction_count, ReplayState), | |
306e196f PD |
58 | VMSTATE_UINT32(data_kind, ReplayState), |
59 | VMSTATE_UINT32(has_unread_data, ReplayState), | |
60 | VMSTATE_UINT64(file_offset, ReplayState), | |
6d0ceb80 | 61 | VMSTATE_UINT64(block_request_id, ReplayState), |
0b30dc01 PD |
62 | VMSTATE_INT32(read_event_kind, ReplayState), |
63 | VMSTATE_UINT64(read_event_id, ReplayState), | |
64 | VMSTATE_INT32(read_event_checkpoint, ReplayState), | |
306e196f PD |
65 | VMSTATE_END_OF_LIST() |
66 | }, | |
67 | }; | |
68 | ||
69 | void replay_vmstate_register(void) | |
70 | { | |
71 | vmstate_register(NULL, 0, &vmstate_replay, &replay_state); | |
72 | } | |
9c2037d0 PD |
73 | |
74 | void replay_vmstate_init(void) | |
75 | { | |
927d6638 JQ |
76 | Error *err = NULL; |
77 | ||
9c2037d0 PD |
78 | if (replay_snapshot) { |
79 | if (replay_mode == REPLAY_MODE_RECORD) { | |
f1a9fcdd DB |
80 | if (!save_snapshot(replay_snapshot, |
81 | true, NULL, false, NULL, &err)) { | |
927d6638 | 82 | error_report_err(err); |
9c2037d0 PD |
83 | error_report("Could not create snapshot for icount record"); |
84 | exit(1); | |
85 | } | |
86 | } else if (replay_mode == REPLAY_MODE_PLAY) { | |
f1a9fcdd | 87 | if (!load_snapshot(replay_snapshot, NULL, false, NULL, &err)) { |
927d6638 | 88 | error_report_err(err); |
9c2037d0 PD |
89 | error_report("Could not load snapshot for icount replay"); |
90 | exit(1); | |
91 | } | |
92 | } | |
93 | } | |
94 | } | |
377b21cc PD |
95 | |
96 | bool replay_can_snapshot(void) | |
97 | { | |
98 | return replay_mode == REPLAY_MODE_NONE | |
99 | || !replay_has_events(); | |
100 | } |