virtio_drivers/device/socket/
protocol.rs
1use super::error::{self, SocketError};
4use crate::volatile::ReadOnly;
5use bitflags::bitflags;
6use core::{
7 convert::{TryFrom, TryInto},
8 fmt,
9};
10use zerocopy::{
11 byteorder::{LittleEndian, U16, U32, U64},
12 AsBytes, FromBytes, FromZeroes,
13};
14
15pub const VMADDR_CID_HOST: u64 = 2;
17
18#[derive(Copy, Clone, Debug)]
20#[repr(u16)]
21pub enum SocketType {
22 Stream = 1,
24 SeqPacket = 2,
26}
27
28impl From<SocketType> for U16<LittleEndian> {
29 fn from(socket_type: SocketType) -> Self {
30 (socket_type as u16).into()
31 }
32}
33
34#[repr(C)]
36pub struct VirtioVsockConfig {
37 pub guest_cid_low: ReadOnly<u32>,
44 pub guest_cid_high: ReadOnly<u32>,
45}
46
47#[repr(C, packed)]
49#[derive(AsBytes, Clone, Copy, Debug, Eq, FromBytes, FromZeroes, PartialEq)]
50pub struct VirtioVsockHdr {
51 pub src_cid: U64<LittleEndian>,
52 pub dst_cid: U64<LittleEndian>,
53 pub src_port: U32<LittleEndian>,
54 pub dst_port: U32<LittleEndian>,
55 pub len: U32<LittleEndian>,
56 pub socket_type: U16<LittleEndian>,
57 pub op: U16<LittleEndian>,
58 pub flags: U32<LittleEndian>,
59 pub buf_alloc: U32<LittleEndian>,
61 pub fwd_cnt: U32<LittleEndian>,
63}
64
65impl Default for VirtioVsockHdr {
66 fn default() -> Self {
67 Self {
68 src_cid: 0.into(),
69 dst_cid: 0.into(),
70 src_port: 0.into(),
71 dst_port: 0.into(),
72 len: 0.into(),
73 socket_type: SocketType::Stream.into(),
74 op: 0.into(),
75 flags: 0.into(),
76 buf_alloc: 0.into(),
77 fwd_cnt: 0.into(),
78 }
79 }
80}
81
82impl VirtioVsockHdr {
83 pub fn len(&self) -> u32 {
85 u32::from(self.len)
86 }
87
88 pub fn op(&self) -> error::Result<VirtioVsockOp> {
89 self.op.try_into()
90 }
91
92 pub fn source(&self) -> VsockAddr {
93 VsockAddr {
94 cid: self.src_cid.get(),
95 port: self.src_port.get(),
96 }
97 }
98
99 pub fn destination(&self) -> VsockAddr {
100 VsockAddr {
101 cid: self.dst_cid.get(),
102 port: self.dst_port.get(),
103 }
104 }
105
106 pub fn check_data_is_empty(&self) -> error::Result<()> {
107 if self.len() == 0 {
108 Ok(())
109 } else {
110 Err(SocketError::UnexpectedDataInPacket)
111 }
112 }
113}
114
115#[derive(Copy, Clone, Debug, Default, PartialEq, Eq)]
117pub struct VsockAddr {
118 pub cid: u64,
120 pub port: u32,
122}
123
124#[derive(Copy, Clone, Debug, Default, AsBytes, FromBytes, FromZeroes)]
126#[repr(C)]
127pub struct VirtioVsockEvent {
128 pub id: U32<LittleEndian>,
130}
131
132#[derive(Copy, Clone, Eq, PartialEq)]
133#[repr(u16)]
134pub enum VirtioVsockOp {
135 Invalid = 0,
136
137 Request = 1,
139 Response = 2,
140 Rst = 3,
141 Shutdown = 4,
142
143 Rw = 5,
145
146 CreditUpdate = 6,
148 CreditRequest = 7,
150}
151
152impl From<VirtioVsockOp> for U16<LittleEndian> {
153 fn from(op: VirtioVsockOp) -> Self {
154 (op as u16).into()
155 }
156}
157
158impl TryFrom<U16<LittleEndian>> for VirtioVsockOp {
159 type Error = SocketError;
160
161 fn try_from(v: U16<LittleEndian>) -> Result<Self, Self::Error> {
162 let op = match u16::from(v) {
163 0 => Self::Invalid,
164 1 => Self::Request,
165 2 => Self::Response,
166 3 => Self::Rst,
167 4 => Self::Shutdown,
168 5 => Self::Rw,
169 6 => Self::CreditUpdate,
170 7 => Self::CreditRequest,
171 _ => return Err(SocketError::UnknownOperation(v.into())),
172 };
173 Ok(op)
174 }
175}
176
177impl fmt::Debug for VirtioVsockOp {
178 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
179 match self {
180 Self::Invalid => write!(f, "VIRTIO_VSOCK_OP_INVALID"),
181 Self::Request => write!(f, "VIRTIO_VSOCK_OP_REQUEST"),
182 Self::Response => write!(f, "VIRTIO_VSOCK_OP_RESPONSE"),
183 Self::Rst => write!(f, "VIRTIO_VSOCK_OP_RST"),
184 Self::Shutdown => write!(f, "VIRTIO_VSOCK_OP_SHUTDOWN"),
185 Self::Rw => write!(f, "VIRTIO_VSOCK_OP_RW"),
186 Self::CreditUpdate => write!(f, "VIRTIO_VSOCK_OP_CREDIT_UPDATE"),
187 Self::CreditRequest => write!(f, "VIRTIO_VSOCK_OP_CREDIT_REQUEST"),
188 }
189 }
190}
191
192bitflags! {
193 #[derive(Copy, Clone, Debug, Default, Eq, PartialEq)]
194 pub(crate) struct Feature: u64 {
195 const STREAM = 1 << 0;
197 const SEQ_PACKET = 1 << 1;
199
200 const NOTIFY_ON_EMPTY = 1 << 24; const ANY_LAYOUT = 1 << 27; const RING_INDIRECT_DESC = 1 << 28;
204 const RING_EVENT_IDX = 1 << 29;
205 const UNUSED = 1 << 30; const VERSION_1 = 1 << 32; const ACCESS_PLATFORM = 1 << 33;
210 const RING_PACKED = 1 << 34;
211 const IN_ORDER = 1 << 35;
212 const ORDER_PLATFORM = 1 << 36;
213 const SR_IOV = 1 << 37;
214 const NOTIFICATION_DATA = 1 << 38;
215 }
216}
217
218bitflags! {
219 #[derive(Copy, Clone, Debug, Default, Eq, PartialEq)]
221 pub struct StreamShutdown: u32 {
222 const RECEIVE = 1 << 0;
224 const SEND = 1 << 1;
226 }
227}
228
229impl From<StreamShutdown> for U32<LittleEndian> {
230 fn from(flags: StreamShutdown) -> Self {
231 flags.bits().into()
232 }
233}