disk_backend_resources/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! Disk resources, for use with [`vm_resource`].

#![forbid(unsafe_code)]

pub mod layer;

use mesh::MeshPayload;
use vm_resource::IntoResource;
use vm_resource::Resource;
use vm_resource::ResourceId;
use vm_resource::kind::DiskHandleKind;
use vm_resource::kind::DiskLayerHandleKind;

// Define config types here so that you don't have to pull in the individual
// crates just to describe the configuration.

/// File-backed disk handle.
#[derive(MeshPayload)]
pub struct FileDiskHandle(pub std::fs::File);

impl ResourceId<DiskHandleKind> for FileDiskHandle {
    const ID: &'static str = "file";
}

/// Disk handle for a disk that emulates persistent reservation support.
#[derive(MeshPayload)]
pub struct DiskWithReservationsHandle(pub Resource<DiskHandleKind>);

impl ResourceId<DiskHandleKind> for DiskWithReservationsHandle {
    const ID: &'static str = "prwrap";
}

/// Disk handle for a fixed VHD1 disk.
#[derive(MeshPayload)]
pub struct FixedVhd1DiskHandle(pub std::fs::File);

impl ResourceId<DiskHandleKind> for FixedVhd1DiskHandle {
    const ID: &'static str = "fixed_vhd1";
}

/// Disk configuration for a striped disk.
#[derive(MeshPayload)]
pub struct StripedDiskHandle {
    /// The underlying disks for the stripes.
    pub devices: Vec<Resource<DiskHandleKind>>,
    /// The size of each stripe.
    pub chunk_size_in_bytes: Option<u32>,
    /// The number of sectors to show for the disk.
    pub logic_sector_count: Option<u64>,
}

impl ResourceId<DiskHandleKind> for StripedDiskHandle {
    const ID: &'static str = "striped";
}

/// Configuration for a disk that is automatically formatted (if it is not
/// already formatted) while being resolved.
// DEVNOTE: this disk type supports a Azure-specific feature in Microsoft's
// closed-source OpenHCL. Due to the NTFS formatting library being used, the
// backing disk type and resolver are currently not able to be open-sourced.
//
// Unfortunately, this feature needs to "leak" into the open-source OpenVMM
// codebase, due to tight coupling in the code of Vtl2Settings.
#[derive(MeshPayload)]
pub struct AutoFormattedDiskHandle {
    /// The disk resource.
    pub disk: Resource<DiskHandleKind>,
    /// The GUID to check for.
    pub guid: [u8; 16],
}

impl ResourceId<DiskHandleKind> for AutoFormattedDiskHandle {
    const ID: &'static str = "ntfsfmt";
}

// blob

/// Handle for a read-only disk backed by a blob served over HTTP.
#[derive(MeshPayload)]
pub struct BlobDiskHandle {
    /// The URL to the disk.
    pub url: String,
    /// The format of the blob.
    pub format: BlobDiskFormat,
}

impl ResourceId<DiskHandleKind> for BlobDiskHandle {
    const ID: &'static str = "blob";
}

/// The format of a disk blob.
#[derive(MeshPayload)]
pub enum BlobDiskFormat {
    /// A flat blob, with no additional metadata.
    Flat,
    /// A fixed VHD1, with a VHD footer specifying disk metadata.
    FixedVhd1,
}

/// Handle for a disk that is backed by one or more layers.
#[derive(MeshPayload)]
pub struct LayeredDiskHandle {
    /// The layers that make up the disk. The first layer is the top-most layer.
    pub layers: Vec<DiskLayerDescription>,
}

impl LayeredDiskHandle {
    /// Create a new layered disk handle with a single layer.
    pub fn single_layer(layer: impl IntoResource<DiskLayerHandleKind>) -> Self {
        Self {
            layers: vec![layer.into_resource().into()],
        }
    }
}

impl ResourceId<DiskHandleKind> for LayeredDiskHandle {
    const ID: &'static str = "layered";
}

/// Description of a disk layer.
#[derive(MeshPayload)]
pub struct DiskLayerDescription {
    /// The layer resource.
    pub layer: Resource<DiskLayerHandleKind>,
    /// If true, reads that miss this layer are written back to this layer.
    pub read_cache: bool,
    /// If true, writes are written both to this layer and the next one.
    pub write_through: bool,
}

impl From<Resource<DiskLayerHandleKind>> for DiskLayerDescription {
    fn from(layer: Resource<DiskLayerHandleKind>) -> Self {
        Self {
            layer,
            read_cache: false,
            write_through: false,
        }
    }
}