mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-05 05:47:40 +03:00
Close FD in various case for poll driver (#530)
This commit is contained in:
parent
f15c3203b1
commit
1f71b200ad
7 changed files with 50 additions and 41 deletions
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [2.5.4] - 2025-03-15
|
||||||
|
|
||||||
|
* Close FD in various case for poll driver
|
||||||
|
|
||||||
## [2.5.3] - 2025-03-14
|
## [2.5.3] - 2025-03-14
|
||||||
|
|
||||||
* Fix operation cancelation handling for poll driver
|
* Fix operation cancelation handling for poll driver
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex-net"
|
name = "ntex-net"
|
||||||
version = "2.5.3"
|
version = "2.5.4"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "ntexwork utils for ntex framework"
|
description = "ntexwork utils for ntex framework"
|
||||||
keywords = ["network", "framework", "async", "futures"]
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
@ -40,7 +40,7 @@ ntex-util = "2.5"
|
||||||
|
|
||||||
ntex-tokio = { version = "0.5.3", optional = true }
|
ntex-tokio = { version = "0.5.3", optional = true }
|
||||||
ntex-compio = { version = "0.2.4", optional = true }
|
ntex-compio = { version = "0.2.4", optional = true }
|
||||||
ntex-neon = { version = "0.1.3", optional = true }
|
ntex-neon = { version = "0.1.4", optional = true }
|
||||||
|
|
||||||
bitflags = { workspace = true }
|
bitflags = { workspace = true }
|
||||||
cfg-if = { workspace = true }
|
cfg-if = { workspace = true }
|
||||||
|
|
|
@ -68,7 +68,6 @@ impl ConnectOps {
|
||||||
let id = self.0.connects.borrow_mut().insert(item);
|
let id = self.0.connects.borrow_mut().insert(item);
|
||||||
|
|
||||||
self.0.api.register(fd, id, Interest::Writable);
|
self.0.api.register(fd, id, Interest::Writable);
|
||||||
|
|
||||||
Ok(id)
|
Ok(id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use std::os::fd::{AsRawFd, RawFd};
|
use std::os::fd::{AsRawFd, RawFd};
|
||||||
use std::{cell::Cell, collections::VecDeque, io, rc::Rc, task, task::Poll};
|
use std::{cell::Cell, collections::VecDeque, future::Future, io, rc::Rc, task};
|
||||||
|
|
||||||
use ntex_neon::driver::{DriverApi, Handler, Interest};
|
use ntex_neon::driver::{DriverApi, Handler, Interest};
|
||||||
use ntex_neon::{syscall, Runtime};
|
use ntex_neon::{syscall, Runtime};
|
||||||
|
@ -125,7 +125,7 @@ impl<T> Handler for StreamOpsHandler<T> {
|
||||||
let result = item.context.with_read_buf(|buf| {
|
let result = item.context.with_read_buf(|buf| {
|
||||||
let chunk = buf.chunk_mut();
|
let chunk = buf.chunk_mut();
|
||||||
let b = chunk.as_mut_ptr();
|
let b = chunk.as_mut_ptr();
|
||||||
Poll::Ready(
|
task::Poll::Ready(
|
||||||
task::ready!(syscall!(
|
task::ready!(syscall!(
|
||||||
break libc::read(item.fd, b as _, chunk.len())
|
break libc::read(item.fd, b as _, chunk.len())
|
||||||
))
|
))
|
||||||
|
@ -142,27 +142,37 @@ impl<T> Handler for StreamOpsHandler<T> {
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
|
||||||
if result.is_pending() {
|
if item.io.is_some() && result.is_pending() {
|
||||||
self.inner.api.register(item.fd, id, Interest::Readable);
|
self.inner.api.register(item.fd, id, Interest::Readable);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Change::Writable => {
|
Change::Writable => {
|
||||||
let item = &mut streams[id];
|
let item = &mut streams[id];
|
||||||
let result = item.context.with_write_buf(|buf| {
|
let result = item.context.with_write_buf(|buf| {
|
||||||
|
log::debug!(
|
||||||
|
"{}: writing {:?} SIZE: {:?}, BUF: {:?}",
|
||||||
|
item.context.tag(),
|
||||||
|
item.fd,
|
||||||
|
buf.len(),
|
||||||
|
buf,
|
||||||
|
);
|
||||||
let slice = &buf[..];
|
let slice = &buf[..];
|
||||||
syscall!(
|
syscall!(
|
||||||
break libc::write(item.fd, slice.as_ptr() as _, slice.len())
|
break libc::write(item.fd, slice.as_ptr() as _, slice.len())
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
|
||||||
if result.is_pending() {
|
if item.io.is_some() && result.is_pending() {
|
||||||
|
log::debug!("{}: want write {:?}", item.context.tag(), item.fd,);
|
||||||
self.inner.api.register(item.fd, id, Interest::Writable);
|
self.inner.api.register(item.fd, id, Interest::Writable);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Change::Error(err) => {
|
Change::Error(err) => {
|
||||||
if let Some(item) = streams.get_mut(id) {
|
if let Some(item) = streams.get_mut(id) {
|
||||||
item.context.stopped(Some(err));
|
item.context.stopped(Some(err));
|
||||||
self.inner.api.unregister_all(item.fd);
|
if let Some(_) = item.io.take() {
|
||||||
|
close(id, item.fd, &self.inner.api);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -183,7 +193,7 @@ impl<T> Handler for StreamOpsHandler<T> {
|
||||||
item.io.is_some()
|
item.io.is_some()
|
||||||
);
|
);
|
||||||
if item.io.is_some() {
|
if item.io.is_some() {
|
||||||
self.inner.api.unregister_all(item.fd);
|
close(id, item.fd, &self.inner.api);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -193,19 +203,33 @@ impl<T> Handler for StreamOpsHandler<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn close(id: usize, fd: RawFd, api: &DriverApi) -> ntex_rt::JoinHandle<io::Result<i32>> {
|
||||||
|
api.unregister_all(fd);
|
||||||
|
ntex_rt::spawn_blocking(move || {
|
||||||
|
syscall!(libc::shutdown(fd, libc::SHUT_RDWR))?;
|
||||||
|
syscall!(libc::close(fd))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
impl<T> StreamCtl<T> {
|
impl<T> StreamCtl<T> {
|
||||||
pub(crate) async fn close(self) -> io::Result<()> {
|
pub(crate) fn close(self) -> impl Future<Output = io::Result<()>> {
|
||||||
let (io, fd) =
|
let (io, fd) =
|
||||||
self.with(|streams| (streams[self.id].io.take(), streams[self.id].fd));
|
self.with(|streams| (streams[self.id].io.take(), streams[self.id].fd));
|
||||||
if let Some(io) = io {
|
let fut = if let Some(io) = io {
|
||||||
|
log::debug!("Closing ({}), {:?}", self.id, fd);
|
||||||
std::mem::forget(io);
|
std::mem::forget(io);
|
||||||
|
Some(close(self.id, fd, &self.inner.api))
|
||||||
ntex_rt::spawn_blocking(move || syscall!(libc::close(fd)))
|
} else {
|
||||||
.await
|
None
|
||||||
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))
|
};
|
||||||
.and_then(crate::helpers::pool_io_err)?;
|
async move {
|
||||||
|
if let Some(fut) = fut {
|
||||||
|
fut.await
|
||||||
|
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))
|
||||||
|
.and_then(crate::helpers::pool_io_err)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn with_io<F, R>(&self, f: F) -> R
|
pub(crate) fn with_io<F, R>(&self, f: F) -> R
|
||||||
|
@ -257,7 +281,7 @@ impl<T> StreamCtl<T> {
|
||||||
let result = item.context.with_read_buf(|buf| {
|
let result = item.context.with_read_buf(|buf| {
|
||||||
let chunk = buf.chunk_mut();
|
let chunk = buf.chunk_mut();
|
||||||
let b = chunk.as_mut_ptr();
|
let b = chunk.as_mut_ptr();
|
||||||
Poll::Ready(
|
task::Poll::Ready(
|
||||||
task::ready!(syscall!(break libc::read(item.fd, b as _, chunk.len())))
|
task::ready!(syscall!(break libc::read(item.fd, b as _, chunk.len())))
|
||||||
.inspect(|size| {
|
.inspect(|size| {
|
||||||
unsafe { buf.advance_mut(*size) };
|
unsafe { buf.advance_mut(*size) };
|
||||||
|
@ -272,7 +296,7 @@ impl<T> StreamCtl<T> {
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
|
||||||
if result.is_pending() {
|
if item.io.is_some() && result.is_pending() {
|
||||||
self.inner
|
self.inner
|
||||||
.api
|
.api
|
||||||
.register(item.fd, self.id, Interest::Readable);
|
.register(item.fd, self.id, Interest::Readable);
|
||||||
|
@ -284,12 +308,6 @@ impl<T> StreamCtl<T> {
|
||||||
self.with(|streams| {
|
self.with(|streams| {
|
||||||
let item = &mut streams[self.id];
|
let item = &mut streams[self.id];
|
||||||
|
|
||||||
log::debug!(
|
|
||||||
"{}: Resume io write ({}), {:?}",
|
|
||||||
item.context.tag(),
|
|
||||||
self.id,
|
|
||||||
item.fd
|
|
||||||
);
|
|
||||||
let result = item.context.with_write_buf(|buf| {
|
let result = item.context.with_write_buf(|buf| {
|
||||||
log::debug!(
|
log::debug!(
|
||||||
"{}: Writing io ({}), buf: {:?}",
|
"{}: Writing io ({}), buf: {:?}",
|
||||||
|
@ -302,7 +320,7 @@ impl<T> StreamCtl<T> {
|
||||||
syscall!(break libc::write(item.fd, slice.as_ptr() as _, slice.len()))
|
syscall!(break libc::write(item.fd, slice.as_ptr() as _, slice.len()))
|
||||||
});
|
});
|
||||||
|
|
||||||
if result.is_pending() {
|
if item.io.is_some() && result.is_pending() {
|
||||||
log::debug!(
|
log::debug!(
|
||||||
"{}: Write is pending ({}), {:?}",
|
"{}: Write is pending ({}), {:?}",
|
||||||
item.context.tag(),
|
item.context.tag(),
|
||||||
|
@ -347,14 +365,14 @@ impl<T> Drop for StreamCtl<T> {
|
||||||
if streams[self.id].ref_count == 0 {
|
if streams[self.id].ref_count == 0 {
|
||||||
let item = streams.remove(self.id);
|
let item = streams.remove(self.id);
|
||||||
log::debug!(
|
log::debug!(
|
||||||
"{}: Drop io ({}), {:?}, has-io: {}",
|
"{}: Drop io ({}), {:?}, has-io: {}",
|
||||||
item.context.tag(),
|
item.context.tag(),
|
||||||
self.id,
|
self.id,
|
||||||
item.fd,
|
item.fd,
|
||||||
item.io.is_some()
|
item.io.is_some()
|
||||||
);
|
);
|
||||||
if item.io.is_some() {
|
if item.io.is_some() {
|
||||||
self.inner.api.unregister_all(item.fd);
|
close(self.id, item.fd, &self.inner.api);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.inner.streams.set(Some(streams));
|
self.inner.streams.set(Some(streams));
|
||||||
|
|
|
@ -88,9 +88,5 @@ async fn run<T>(ctl: StreamCtl<T>, context: IoContext) {
|
||||||
|
|
||||||
ctl.resume_write();
|
ctl.resume_write();
|
||||||
context.shutdown(st == Status::Shutdown).await;
|
context.shutdown(st == Status::Shutdown).await;
|
||||||
|
context.stopped(ctl.close().await.err());
|
||||||
ctl.pause_all();
|
|
||||||
let result = ctl.close().await;
|
|
||||||
|
|
||||||
context.stopped(result.err());
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -248,10 +248,6 @@ where
|
||||||
Ok(())
|
Ok(())
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
// wait for server
|
|
||||||
if std::env::var("GITHUB_ACTIONS") == Ok("true".to_string()) {
|
|
||||||
thread::sleep(std::time::Duration::from_millis(150));
|
|
||||||
}
|
|
||||||
|
|
||||||
let (system, server, addr) = rx.recv().unwrap();
|
let (system, server, addr) = rx.recv().unwrap();
|
||||||
|
|
||||||
|
|
|
@ -701,10 +701,6 @@ where
|
||||||
Ok(())
|
Ok(())
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
// wait for server
|
|
||||||
if std::env::var("GITHUB_ACTIONS") == Ok("true".to_string()) {
|
|
||||||
thread::sleep(std::time::Duration::from_millis(150));
|
|
||||||
}
|
|
||||||
|
|
||||||
let (system, server, addr) = rx.recv().unwrap();
|
let (system, server, addr) = rx.recv().unwrap();
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue