2020-10-28 17:36:17 +00:00
|
|
|
// Copyright © 2020 Intel Corporation
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
//
|
|
|
|
|
2021-09-27 12:04:15 +00:00
|
|
|
use crate::{MigratableError, VersionMapped};
|
2022-05-17 21:04:38 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2021-09-27 12:04:15 +00:00
|
|
|
use std::io::{Read, Write};
|
|
|
|
use versionize::{VersionMap, Versionize, VersionizeResult};
|
|
|
|
use versionize_derive::Versionize;
|
2021-02-11 15:55:00 +00:00
|
|
|
use vm_memory::ByteValued;
|
2020-10-28 17:36:17 +00:00
|
|
|
|
|
|
|
// Migration protocol
|
|
|
|
// 1: Source establishes communication with destination (file socket or TCP connection.)
|
|
|
|
// (The establishment is out of scope.)
|
|
|
|
// 2: Source -> Dest : send "start command"
|
|
|
|
// 3: Dest -> Source : sends "ok response" when read to accept state data
|
2020-11-12 16:55:55 +00:00
|
|
|
// 4: Source -> Dest : sends "config command" followed by config data, length
|
|
|
|
// in command is length of config data
|
2020-10-28 17:36:17 +00:00
|
|
|
// 5: Dest -> Source : sends "ok response" when ready to accept memory data
|
|
|
|
// 6: Source -> Dest : send "memory command" followed by table of u64 pairs (GPA, size)
|
|
|
|
// followed by the memory described in those pairs.
|
|
|
|
// !! length is size of table i.e. 16 * number of ranges !!
|
|
|
|
// 7: Dest -> Source : sends "ok response" when ready to accept more memory data
|
2020-11-12 16:55:55 +00:00
|
|
|
// 8..(n-4): Repeat steps 6 and 7 until source has no more memory to send
|
|
|
|
// (n-3): Source -> Dest : sends "state command" followed by state data, length
|
|
|
|
// in command is length of config data
|
|
|
|
// (n-2): Dest -> Source : sends "ok response"
|
2020-10-28 17:36:17 +00:00
|
|
|
// (n-1): Source -> Dest : send "complete command"
|
|
|
|
// n: Dest -> Source: sends "ok response"
|
2022-01-17 16:40:01 +00:00
|
|
|
//
|
|
|
|
// "Local version": (Handing FDs across socket for memory)
|
|
|
|
// 1: Source establishes communication with destination (file socket or TCP connection.)
|
|
|
|
// (The establishment is out of scope.)
|
|
|
|
// 2: Source -> Dest : send "start command"
|
|
|
|
// 3: Dest -> Source : sends "ok response" when read to accept state data
|
|
|
|
// 4: Source -> Dest : sends "config command" followed by config data, length
|
|
|
|
// in command is length of config data
|
|
|
|
// 5: Dest -> Source : sends "ok response" when ready to accept memory data
|
|
|
|
// 6: Source -> Dest : send "memory fd command" followed by u16 slot ID and FD for memory
|
|
|
|
// 7: Dest -> Source : sends "ok response" when received
|
|
|
|
// 8..(n-4): Repeat steps 6 and 7 until source has no more memory to send
|
|
|
|
// (n-3): Source -> Dest : sends "state command" followed by state data, length
|
|
|
|
// in command is length of config data
|
|
|
|
// (n-2): Dest -> Source : sends "ok response"
|
|
|
|
// (n-1): Source -> Dest : send "complete command"
|
|
|
|
// n: Dest -> Source: sends "ok response"
|
|
|
|
//
|
2020-10-28 17:36:17 +00:00
|
|
|
// The destination can at any time send an "error response" to cancel
|
|
|
|
// The source can at any time send an "abandon request" to cancel
|
|
|
|
|
|
|
|
#[repr(u16)]
|
|
|
|
#[derive(Copy, Clone)]
|
|
|
|
pub enum Command {
|
|
|
|
Invalid,
|
|
|
|
Start,
|
2020-11-12 16:55:55 +00:00
|
|
|
Config,
|
2020-10-28 17:36:17 +00:00
|
|
|
State,
|
|
|
|
Memory,
|
|
|
|
Complete,
|
|
|
|
Abandon,
|
2022-01-14 08:07:17 +00:00
|
|
|
MemoryFd,
|
2020-10-28 17:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Command {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::Invalid
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
2021-02-11 15:55:00 +00:00
|
|
|
#[derive(Default, Copy, Clone)]
|
2020-10-28 17:36:17 +00:00
|
|
|
pub struct Request {
|
|
|
|
command: Command,
|
|
|
|
padding: [u8; 6],
|
|
|
|
length: u64, // Length of payload for command excluding the Request struct
|
|
|
|
}
|
|
|
|
|
2021-11-17 13:44:53 +00:00
|
|
|
// SAFETY: Request contains a series of integers with no implicit padding
|
|
|
|
unsafe impl ByteValued for Request {}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
impl Request {
|
|
|
|
pub fn new(command: Command, length: u64) -> Self {
|
|
|
|
Self {
|
|
|
|
command,
|
|
|
|
length,
|
|
|
|
..Default::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn start() -> Self {
|
|
|
|
Self::new(Command::Start, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn state(length: u64) -> Self {
|
|
|
|
Self::new(Command::State, length)
|
|
|
|
}
|
|
|
|
|
2020-11-12 16:55:55 +00:00
|
|
|
pub fn config(length: u64) -> Self {
|
|
|
|
Self::new(Command::Config, length)
|
|
|
|
}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
pub fn memory(length: u64) -> Self {
|
|
|
|
Self::new(Command::Memory, length)
|
|
|
|
}
|
|
|
|
|
2022-01-14 08:07:17 +00:00
|
|
|
pub fn memory_fd(length: u64) -> Self {
|
|
|
|
Self::new(Command::MemoryFd, length)
|
|
|
|
}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
pub fn complete() -> Self {
|
|
|
|
Self::new(Command::Complete, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn abandon() -> Self {
|
|
|
|
Self::new(Command::Abandon, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn command(&self) -> Command {
|
|
|
|
self.command
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn length(&self) -> u64 {
|
|
|
|
self.length
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read_from(fd: &mut dyn Read) -> Result<Request, MigratableError> {
|
|
|
|
let mut request = Request::default();
|
2021-02-11 15:55:00 +00:00
|
|
|
fd.read_exact(Self::as_mut_slice(&mut request))
|
2020-10-28 17:36:17 +00:00
|
|
|
.map_err(MigratableError::MigrateSocket)?;
|
|
|
|
|
|
|
|
Ok(request)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_to(&self, fd: &mut dyn Write) -> Result<(), MigratableError> {
|
2021-02-11 15:55:00 +00:00
|
|
|
fd.write_all(Self::as_slice(self))
|
2020-10-28 17:36:17 +00:00
|
|
|
.map_err(MigratableError::MigrateSocket)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(u16)]
|
2022-06-30 15:41:46 +00:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
2020-10-28 17:36:17 +00:00
|
|
|
pub enum Status {
|
|
|
|
Invalid,
|
|
|
|
Ok,
|
|
|
|
Error,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Status {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::Invalid
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
2021-02-11 15:55:00 +00:00
|
|
|
#[derive(Default, Copy, Clone)]
|
2020-10-28 17:36:17 +00:00
|
|
|
pub struct Response {
|
|
|
|
status: Status,
|
|
|
|
padding: [u8; 6],
|
|
|
|
length: u64, // Length of payload for command excluding the Response struct
|
|
|
|
}
|
|
|
|
|
2021-11-17 13:44:53 +00:00
|
|
|
// SAFETY: Response contains a series of integers with no implicit padding
|
|
|
|
unsafe impl ByteValued for Response {}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
impl Response {
|
|
|
|
pub fn new(status: Status, length: u64) -> Self {
|
|
|
|
Self {
|
|
|
|
status,
|
|
|
|
length,
|
|
|
|
..Default::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn ok() -> Self {
|
|
|
|
Self::new(Status::Ok, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn error() -> Self {
|
|
|
|
Self::new(Status::Error, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn status(&self) -> Status {
|
|
|
|
self.status
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read_from(fd: &mut dyn Read) -> Result<Response, MigratableError> {
|
|
|
|
let mut response = Response::default();
|
2021-02-11 15:55:00 +00:00
|
|
|
fd.read_exact(Self::as_mut_slice(&mut response))
|
2020-10-28 17:36:17 +00:00
|
|
|
.map_err(MigratableError::MigrateSocket)?;
|
|
|
|
|
|
|
|
Ok(response)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_to(&self, fd: &mut dyn Write) -> Result<(), MigratableError> {
|
2021-02-11 15:55:00 +00:00
|
|
|
fd.write_all(Self::as_slice(self))
|
2020-10-28 17:36:17 +00:00
|
|
|
.map_err(MigratableError::MigrateSocket)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
2021-11-10 08:46:28 +00:00
|
|
|
#[derive(Clone, Default, Serialize, Deserialize, Versionize)]
|
2020-10-28 17:36:17 +00:00
|
|
|
pub struct MemoryRange {
|
|
|
|
pub gpa: u64,
|
|
|
|
pub length: u64,
|
|
|
|
}
|
|
|
|
|
2021-10-05 15:53:08 +00:00
|
|
|
#[derive(Clone, Default, Serialize, Deserialize, Versionize)]
|
2020-10-28 17:36:17 +00:00
|
|
|
pub struct MemoryRangeTable {
|
|
|
|
data: Vec<MemoryRange>,
|
|
|
|
}
|
|
|
|
|
2021-09-27 12:04:15 +00:00
|
|
|
impl VersionMapped for MemoryRangeTable {}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
impl MemoryRangeTable {
|
2021-09-23 07:54:40 +00:00
|
|
|
pub fn from_bitmap(bitmap: Vec<u64>, start_addr: u64, page_size: u64) -> Self {
|
2021-08-04 13:30:28 +00:00
|
|
|
let mut table = MemoryRangeTable::default();
|
|
|
|
let mut entry: Option<MemoryRange> = None;
|
|
|
|
for (i, block) in bitmap.iter().enumerate() {
|
|
|
|
for j in 0..64 {
|
|
|
|
let is_page_dirty = ((block >> j) & 1u64) != 0u64;
|
|
|
|
let page_offset = ((i * 64) + j) as u64 * page_size;
|
|
|
|
if is_page_dirty {
|
|
|
|
if let Some(entry) = &mut entry {
|
|
|
|
entry.length += page_size;
|
|
|
|
} else {
|
|
|
|
entry = Some(MemoryRange {
|
|
|
|
gpa: start_addr + page_offset,
|
|
|
|
length: page_size,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else if let Some(entry) = entry.take() {
|
|
|
|
table.push(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if let Some(entry) = entry.take() {
|
|
|
|
table.push(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
table
|
|
|
|
}
|
|
|
|
|
2020-10-28 17:36:17 +00:00
|
|
|
pub fn regions(&self) -> &[MemoryRange] {
|
|
|
|
&self.data
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn push(&mut self, range: MemoryRange) {
|
|
|
|
self.data.push(range)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read_from(fd: &mut dyn Read, length: u64) -> Result<MemoryRangeTable, MigratableError> {
|
|
|
|
assert!(length as usize % std::mem::size_of::<MemoryRange>() == 0);
|
|
|
|
|
2021-11-10 08:46:28 +00:00
|
|
|
let mut data: Vec<MemoryRange> = Vec::new();
|
|
|
|
data.resize_with(
|
|
|
|
length as usize / (std::mem::size_of::<MemoryRange>()),
|
|
|
|
Default::default,
|
|
|
|
);
|
2020-10-28 17:36:17 +00:00
|
|
|
fd.read_exact(unsafe {
|
|
|
|
std::slice::from_raw_parts_mut(
|
|
|
|
data.as_ptr() as *mut MemoryRange as *mut u8,
|
|
|
|
length as usize,
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.map_err(MigratableError::MigrateSocket)?;
|
|
|
|
|
|
|
|
Ok(Self { data })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn length(&self) -> u64 {
|
|
|
|
(std::mem::size_of::<MemoryRange>() * self.data.len()) as u64
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn write_to(&self, fd: &mut dyn Write) -> Result<(), MigratableError> {
|
|
|
|
fd.write_all(unsafe {
|
|
|
|
std::slice::from_raw_parts(
|
|
|
|
self.data.as_ptr() as *const MemoryRange as *const u8,
|
|
|
|
self.length() as usize,
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.map_err(MigratableError::MigrateSocket)
|
|
|
|
}
|
2021-08-04 13:30:28 +00:00
|
|
|
|
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.data.is_empty()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn extend(&mut self, table: Self) {
|
|
|
|
self.data.extend(table.data)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn new_from_tables(tables: Vec<Self>) -> Self {
|
|
|
|
let mut data = Vec::new();
|
|
|
|
for table in tables {
|
|
|
|
data.extend(table.data);
|
|
|
|
}
|
|
|
|
Self { data }
|
|
|
|
}
|
2020-10-28 17:36:17 +00:00
|
|
|
}
|