From e149cde883352f2e6eccf9b41c1fe5cf55f4d4a0 Mon Sep 17 00:00:00 2001 From: WuZheng Date: Tue, 3 Dec 2024 00:44:53 +0800 Subject: [PATCH] fix bug for unexpected pagefault when nested fork. --- api/ruxos_posix_api/src/imp/pthread/mod.rs | 1 - crates/driver_net/src/loopback.rs | 8 +++----- modules/ruxnet/src/smoltcp_impl/mod.rs | 4 +--- modules/ruxtask/src/run_queue.rs | 1 + modules/ruxtask/src/task.rs | 21 +++++++++++++++------ 5 files changed, 20 insertions(+), 15 deletions(-) diff --git a/api/ruxos_posix_api/src/imp/pthread/mod.rs b/api/ruxos_posix_api/src/imp/pthread/mod.rs index da2694ff8..48a205c2e 100644 --- a/api/ruxos_posix_api/src/imp/pthread/mod.rs +++ b/api/ruxos_posix_api/src/imp/pthread/mod.rs @@ -310,7 +310,6 @@ pub unsafe fn sys_clone( } else if (flags as u32 & ctypes::SIGCHLD) != 0 { TID_TO_PTHREAD.read(); let pid = if let Some(task_ref) = ruxtask::fork_task() { - warn!("fork_task success, pid: {}", task_ref.id().as_u64()); task_ref.id().as_u64() } else { let children_ref = ruxtask::current(); diff --git a/crates/driver_net/src/loopback.rs b/crates/driver_net/src/loopback.rs index 3574cee75..072bd0d6f 100644 --- a/crates/driver_net/src/loopback.rs +++ b/crates/driver_net/src/loopback.rs @@ -57,8 +57,6 @@ impl BaseDriverOps for LoopbackDevice { } } -use log::info; - impl NetDriverOps for LoopbackDevice { #[inline] fn mac_address(&self) -> EthernetAddress { @@ -85,11 +83,11 @@ impl NetDriverOps for LoopbackDevice { self.queue.len() } - fn fill_rx_buffers(&mut self, buf_pool: &Arc) -> DevResult { + fn fill_rx_buffers(&mut self, _buf_pool: &Arc) -> DevResult { Ok(()) } - fn recycle_rx_buffer(&mut self, rx_buf: NetBufPtr) -> DevResult { + fn recycle_rx_buffer(&mut self, _rx_buf: NetBufPtr) -> DevResult { Ok(()) } @@ -97,7 +95,7 @@ impl NetDriverOps for LoopbackDevice { Ok(()) } - fn prepare_tx_buffer(&self, tx_buf: &mut NetBuf, pkt_len: usize) -> DevResult { + fn prepare_tx_buffer(&self, _tx_buf: &mut NetBuf, _pkt_len: usize) -> DevResult { Ok(()) } diff --git a/modules/ruxnet/src/smoltcp_impl/mod.rs b/modules/ruxnet/src/smoltcp_impl/mod.rs index 192fe2d93..50bfbfe9d 100644 --- a/modules/ruxnet/src/smoltcp_impl/mod.rs +++ b/modules/ruxnet/src/smoltcp_impl/mod.rs @@ -36,8 +36,6 @@ pub use self::dns::dns_query; pub use self::tcp::TcpSocket; pub use self::udp::UdpSocket; -pub use driver_net::loopback::LoopbackDevice; - macro_rules! env_or_default { ($key:literal) => { match option_env!($key) { @@ -347,7 +345,7 @@ pub fn bench_receive() { } pub(crate) fn init() { - let mut socketset = SocketSetWrapper::new(); + let socketset = SocketSetWrapper::new(); IFACE_LIST.init_by(Mutex::new(vec::Vec::new())); SOCKET_SET.init_by(socketset); diff --git a/modules/ruxtask/src/run_queue.rs b/modules/ruxtask/src/run_queue.rs index 82fa5ffd1..95de4926f 100644 --- a/modules/ruxtask/src/run_queue.rs +++ b/modules/ruxtask/src/run_queue.rs @@ -129,6 +129,7 @@ impl AxRunQueue { assert!(!curr.is_idle()); // we must not block current task with preemption disabled. + // only allow blocking current task with run_queue lock held. #[cfg(feature = "preempt")] assert!(curr.can_preempt(1)); diff --git a/modules/ruxtask/src/task.rs b/modules/ruxtask/src/task.rs index cc1a45d20..7311d53ad 100644 --- a/modules/ruxtask/src/task.rs +++ b/modules/ruxtask/src/task.rs @@ -81,6 +81,7 @@ pub struct TaskInner { exit_code: AtomicI32, wait_for_exit: WaitQueue, + stack_map_addr: SpinNoIrq, kstack: SpinNoIrq>>, ctx: UnsafeCell, @@ -235,6 +236,7 @@ impl TaskInner { preempt_disable_count: AtomicUsize::new(0), exit_code: AtomicI32::new(0), wait_for_exit: WaitQueue::new(), + stack_map_addr: SpinNoIrq::new(VirtAddr::from(0)), // should be set later kstack: SpinNoIrq::new(Arc::new(None)), ctx: UnsafeCell::new(TaskContext::new()), #[cfg(feature = "tls")] @@ -279,6 +281,7 @@ impl TaskInner { preempt_disable_count: AtomicUsize::new(0), exit_code: AtomicI32::new(0), wait_for_exit: WaitQueue::new(), + stack_map_addr: SpinNoIrq::new(VirtAddr::from(0)), kstack: SpinNoIrq::new(Arc::new(None)), ctx: UnsafeCell::new(TaskContext::new()), #[cfg(feature = "tls")] @@ -299,6 +302,7 @@ impl TaskInner { pub fn set_stack_top(&self, begin: usize, size: usize) { debug!("set_stack_top: begin={:#x}, size={:#x}", begin, size); + *self.stack_map_addr.lock() = VirtAddr::from(begin); *self.kstack.lock() = Arc::new(Some(TaskStack { ptr: NonNull::new(begin as *mut u8).unwrap(), layout: Layout::from_size_align(size, PAGE_SIZE_4K).unwrap(), @@ -406,14 +410,14 @@ impl TaskInner { // Note: the stack region is mapped to the same position as the parent process's stack, be careful when update the stack region for the forked process. let (_, prev_flag, _) = cloned_page_table - .query(current_stack.end()) + .query(*current().stack_map_addr.lock()) .expect("failed to query stack region when forking"); cloned_page_table - .unmap_region(current_stack.end(), align_up_4k(stack_size)) + .unmap_region(*current().stack_map_addr.lock(), align_up_4k(stack_size)) .expect("failed to unmap stack region when forking"); cloned_page_table .map_region( - current_stack.end(), + *current().stack_map_addr.lock(), stack_paddr, stack_size, prev_flag, @@ -477,10 +481,11 @@ impl TaskInner { need_resched: AtomicBool::new(current_task.need_resched.load(Ordering::Relaxed)), #[cfg(feature = "preempt")] preempt_disable_count: AtomicUsize::new( - current_task.preempt_disable_count.load(Ordering::Relaxed), + current_task.preempt_disable_count.load(Ordering::Acquire), ), exit_code: AtomicI32::new(0), wait_for_exit: WaitQueue::new(), + stack_map_addr: SpinNoIrq::new(*current().stack_map_addr.lock()), kstack: SpinNoIrq::new(Arc::new(Some(new_stack))), ctx: UnsafeCell::new(TaskContext::new()), #[cfg(feature = "tls")] @@ -515,6 +520,7 @@ impl TaskInner { .lock() .insert(new_pid.as_u64(), task_ref.clone()); + warn!("forked task: save_current_content {}", task_ref.id_name()); unsafe { // copy the stack content from current stack to new stack (*task_ref.ctx_mut_ptr()).save_current_content( @@ -554,6 +560,7 @@ impl TaskInner { preempt_disable_count: AtomicUsize::new(0), exit_code: AtomicI32::new(0), wait_for_exit: WaitQueue::new(), + stack_map_addr: SpinNoIrq::new(VirtAddr::from(0)), // set in set_stack_top kstack: SpinNoIrq::new(Arc::new(None)), ctx: UnsafeCell::new(TaskContext::new()), #[cfg(feature = "tls")] @@ -590,6 +597,7 @@ impl TaskInner { let bindings = PROCESS_MAP.lock(); let (&_parent_id, &ref task_ref) = bindings.first_key_value().unwrap(); let idle_kstack = TaskStack::alloc(align_up_4k(IDLE_STACK_SIZE)); + let idle_kstack_top = idle_kstack.top(); let mut t = Self { parent_process: Some(Arc::downgrade(task_ref)), @@ -609,7 +617,8 @@ impl TaskInner { preempt_disable_count: AtomicUsize::new(0), exit_code: AtomicI32::new(0), wait_for_exit: WaitQueue::new(), - kstack: SpinNoIrq::new(Arc::new(None)), + stack_map_addr: SpinNoIrq::new(idle_kstack.end()), + kstack: SpinNoIrq::new(Arc::new(Some(idle_kstack))), ctx: UnsafeCell::new(TaskContext::new()), #[cfg(feature = "tls")] tls: TlsArea::alloc(), @@ -633,7 +642,7 @@ impl TaskInner { debug!("new idle task: {}", t.id_name()); t.ctx .get_mut() - .init(task_entry as usize, idle_kstack.top(), tls); + .init(task_entry as usize, idle_kstack_top, tls); let task_ref = Arc::new(AxTask::new(t));