Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

vsock: parse cid from vsock #267

Merged
merged 2 commits into from
Dec 28, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,8 @@ windows-sys = {version = "0.48", features = [ "Win32_Foundation", "Win32_Storage
tokio-vsock = { version = "0.4.0", optional = true }

[build-dependencies]
# lock home to avoid conflict with latest version
home = "=0.5.9"
protobuf-codegen = "3.1.0"

[features]
Expand Down
2 changes: 2 additions & 0 deletions compiler/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@ homepage = "https://github.com/containerd/ttrpc-rust/tree/master/compiler"
readme = "README.md"

[dependencies]
# lock home to avoid conflict with latest version
home = "=0.5.9"
protobuf = "2.27.1"
protobuf-codegen = "2.27.1"
prost = "0.8"
Expand Down
77 changes: 53 additions & 24 deletions src/common.rs
Original file line number Diff line number Diff line change
Expand Up @@ -101,8 +101,40 @@ fn make_addr(_domain: Domain, sockaddr: &str) -> Result<UnixAddr> {
UnixAddr::new(sockaddr).map_err(err_to_others_err!(e, ""))
}

fn make_socket(addr: (&str, u32)) -> Result<(RawFd, Domain, Box<dyn SockaddrLike>)> {
let (sockaddr, _) = addr;
// addr: cid:port
// return (cid, port)
#[cfg(any(target_os = "linux", target_os = "android"))]
fn parse_vscok(addr: &str) -> Result<(u32, u32)> {
// vsock://cid:port
let sockaddr_port_v: Vec<&str> = addr.split(':').collect();
if sockaddr_port_v.len() != 2 {
return Err(Error::Others(format!(
"sockaddr {addr} is not right for vsock"
)));
}

// for -1 need trace to libc::VMADDR_CID_ANY
let cid: u32 = if sockaddr_port_v[0].trim().eq("-1") {
libc::VMADDR_CID_ANY
} else {
sockaddr_port_v[0].parse().map_err(|e| {
Error::Others(format!(
"failed to parse cid from {:?} error: {:?}",
sockaddr_port_v[0], e
))
})?
};

let port: u32 = sockaddr_port_v[1].parse().map_err(|e| {
Error::Others(format!(
"failed to parse port from {:?} error: {:?}",
sockaddr_port_v[1], e
))
})?;
Ok((cid, port))
}

fn make_socket(sockaddr: &str) -> Result<(RawFd, Domain, Box<dyn SockaddrLike>)> {
let (domain, sockaddrv) = parse_sockaddr(sockaddr)?;

let get_sock_addr = |domain, sockaddr| -> Result<(RawFd, Box<dyn SockaddrLike>)> {
Expand All @@ -121,23 +153,14 @@ fn make_socket(addr: (&str, u32)) -> Result<(RawFd, Domain, Box<dyn SockaddrLike
Domain::Unix => get_sock_addr(domain, sockaddrv)?,
#[cfg(any(target_os = "linux", target_os = "android"))]
Domain::Vsock => {
let sockaddr_port_v: Vec<&str> = sockaddrv.split(':').collect();
if sockaddr_port_v.len() != 2 {
return Err(Error::Others(format!(
"sockaddr {sockaddr} is not right for vsock"
)));
}
let port: u32 = sockaddr_port_v[1]
.parse()
.expect("the vsock port is not an number");
let (cid, port) = parse_vscok(sockaddrv)?;
let fd = socket(
AddressFamily::Vsock,
SockType::Stream,
SockFlag::SOCK_CLOEXEC,
None,
)
.map_err(|e| Error::Socket(e.to_string()))?;
let cid = addr.1;
let sockaddr = VsockAddr::new(cid, port);
(fd, Box::new(sockaddr))
}
Expand All @@ -146,18 +169,8 @@ fn make_socket(addr: (&str, u32)) -> Result<(RawFd, Domain, Box<dyn SockaddrLike
Ok((fd, domain, sockaddr))
}

// Vsock is not supported on non Linux.
#[cfg(any(target_os = "linux", target_os = "android"))]
use libc::VMADDR_CID_ANY;
#[cfg(not(any(target_os = "linux", target_os = "android")))]
const VMADDR_CID_ANY: u32 = 0;
#[cfg(any(target_os = "linux", target_os = "android"))]
use libc::VMADDR_CID_HOST;
#[cfg(not(any(target_os = "linux", target_os = "android")))]
const VMADDR_CID_HOST: u32 = 0;

pub(crate) fn do_bind(sockaddr: &str) -> Result<(RawFd, Domain)> {
let (fd, domain, sockaddr) = make_socket((sockaddr, VMADDR_CID_ANY))?;
let (fd, domain, sockaddr) = make_socket(sockaddr)?;

setsockopt(fd, sockopt::ReusePort, &true)?;
bind(fd, sockaddr.as_ref()).map_err(err_to_others_err!(e, ""))?;
Expand All @@ -167,7 +180,7 @@ pub(crate) fn do_bind(sockaddr: &str) -> Result<(RawFd, Domain)> {

/// Creates a unix socket for client.
pub(crate) unsafe fn client_connect(sockaddr: &str) -> Result<RawFd> {
let (fd, _, sockaddr) = make_socket((sockaddr, VMADDR_CID_HOST))?;
let (fd, _, sockaddr) = make_socket(sockaddr)?;

connect(fd, sockaddr.as_ref())?;

Expand Down Expand Up @@ -236,4 +249,20 @@ mod tests {
}
}
}
#[cfg(any(target_os = "linux", target_os = "android"))]
#[test]
fn test_parse_vscok() {
for i in &[
("-1:1024", (libc::VMADDR_CID_ANY, 1024)),
("0:1", (0, 1)),
("1:2", (1, 2)),
("4294967294:3", (4294967294, 3)),
// 4294967295 = 0xFFFFFFFF
("4294967295:4", (libc::VMADDR_CID_ANY, 4)),
] {
let (input, (cid, port)) = (i.0, i.1);
let r = parse_vscok(input);
assert_eq!(r.unwrap(), (cid, port), "parse {:?} failed", i);
}
}
}
2 changes: 2 additions & 0 deletions ttrpc-codegen/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ readme = "README.md"


[dependencies]
# lock home to avoid conflict with latest version
home = "=0.5.9"
protobuf-support = "3.2.0"
protobuf = { version = "2.27.1" }
protobuf-codegen = "3.2.0"
Expand Down
Loading