linuxppc-dev.lists.ozlabs.org archive mirror
 help / color / mirror / Atom feed
* [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs
@ 2019-11-27  1:11 Haren Myneni
  2019-11-27  8:46 ` Christoph Hellwig
  2019-12-06  9:51 ` Haren Myneni
  0 siblings, 2 replies; 4+ messages in thread
From: Haren Myneni @ 2019-11-27  1:11 UTC (permalink / raw)
  To: linuxppc-dev, devicetree, mpe, npiggin, mikey, herbert; +Cc: sukadev


For each fault CRB, update fault address in CRB (fault_storage_addr)
and translation error status in CSB so that user space touch the
fault address and resend the request. If the user space passed invalid
CSB address send signal to process with SIGSEGV.

Signed-off-by: Sukadev Bhattiprolu <sukadev@linux.vnet.ibm.com>
Signed-off-by: Haren Myneni <haren@us.ibm.com>
---
 arch/powerpc/platforms/powernv/vas-fault.c | 121 ++++++++++++++++++++++++++++-
 1 file changed, 120 insertions(+), 1 deletion(-)

diff --git a/arch/powerpc/platforms/powernv/vas-fault.c b/arch/powerpc/platforms/powernv/vas-fault.c
index 7a8b2b5..dd27649 100644
--- a/arch/powerpc/platforms/powernv/vas-fault.c
+++ b/arch/powerpc/platforms/powernv/vas-fault.c
@@ -36,6 +36,124 @@ void vas_wakeup_fault_handler(int virq, void *arg)
 	wake_up(&vinst->fault_wq);
 }
 
+static void notify_process(pid_t pid, u64 fault_addr)
+{
+	int rc;
+	struct kernel_siginfo info;
+
+	memset(&info, 0, sizeof(info));
+
+	info.si_signo = SIGSEGV;
+	info.si_errno = EFAULT;
+	info.si_code = SEGV_MAPERR;
+
+	info.si_addr = (void *)fault_addr;
+	rcu_read_lock();
+	rc = kill_pid_info(SIGSEGV, &info, find_vpid(pid));
+	rcu_read_unlock();
+
+	pr_devel("%s(): pid %d kill_proc_info() rc %d\n", __func__, pid, rc);
+}
+
+/*
+ * Update the CSB to indicate a translation error.
+ *
+ * If the fault is in the CSB address itself or if we are unable to
+ * update the CSB, send a signal to the process, because we have no
+ * other way of notifying the user process.
+ *
+ * Remaining settings in the CSB are based on wait_for_csb() of
+ * NX-GZIP.
+ */
+static void update_csb(struct vas_window *window,
+			struct coprocessor_request_block *crb)
+{
+	int rc;
+	pid_t pid;
+	int task_exit = 0;
+	void __user *csb_addr;
+	struct task_struct *tsk;
+	struct coprocessor_status_block csb;
+
+	/*
+	 * NX user space windows can not be opened for task->mm=NULL
+	 * and faults will not be generated for kernel requests.
+	 */
+	if (!window->mm || !window->user_win)
+		return;
+
+	csb_addr = (void *)__be64_to_cpu(crb->csb_addr);
+
+	csb.cc = CSB_CC_TRANSLATION;
+	csb.ce = CSB_CE_TERMINATION;
+	csb.cs = 0;
+	csb.count = 0;
+
+	/*
+	 * Returns the fault address in CPU format since it is passed with
+	 * signal. But if the user space expects BE format, need changes.
+	 * i.e either kernel (here) or user should convert to CPU format.
+	 * Not both!
+	 */
+	csb.address = crb_nx_fault_addr(crb);
+	csb.flags = 0;
+
+	use_mm(window->mm);
+	rc = copy_to_user(csb_addr, &csb, sizeof(csb));
+	/*
+	 * User space polls on csb.flags (first byte). So add barrier
+	 * then copy first byte with csb flags update.
+	 */
+	smp_mb();
+	if (!rc) {
+		csb.flags = CSB_V;
+		rc = copy_to_user(csb_addr, &csb, sizeof(u8));
+	}
+	unuse_mm(window->mm);
+
+	/* Success */
+	if (!rc)
+		return;
+
+	/*
+	 * User space passed invalid CSB address, Notify process with
+	 * SEGV signal.
+	 */
+	tsk = get_pid_task(window->pid, PIDTYPE_PID);
+	/*
+	 * Send window will be closed after processing all NX requests
+	 * and process exits after closing all windows. In multi-thread
+	 * applications, thread may not exists, but does not close FD
+	 * (means send window) upon exit. Parent thread (tgid) can use
+	 * and close the window later.
+	 */
+	if (tsk) {
+		if (tsk->flags & PF_EXITING)
+			task_exit = 1;
+		put_task_struct(tsk);
+		pid = vas_window_pid(window);
+	} else {
+		pid = vas_window_tgid(window);
+
+		rcu_read_lock();
+		tsk = find_task_by_vpid(pid);
+		if (!tsk) {
+			rcu_read_unlock();
+			return;
+		}
+		if (tsk->flags & PF_EXITING)
+			task_exit = 1;
+		rcu_read_unlock();
+	}
+
+	/* Do not notify if the task is exiting. */
+	if (!task_exit) {
+		pr_err("Invalid CSB address 0x%p signalling pid(%d)\n",
+				csb_addr, pid);
+		notify_process(pid, (u64)csb_addr);
+	}
+}
+
 /*
  * Process CRBs that we receive on the fault window.
  */
@@ -97,7 +215,7 @@ static void process_fault_crbs(struct vas_instance *vinst)
 
 		if (IS_ERR(window)) {
 			/*
-			 * What now? We got an interrupt about a specific send
+			 * We got an interrupt about a specific send
 			 * window but we can't find that window and we can't
 			 * even clean it up (return credit).
 			 * But we should not get here.
@@ -111,6 +229,7 @@ static void process_fault_crbs(struct vas_instance *vinst)
 			return;
 		}
 
+		update_csb(window, crb);
 	} while (true);
 }
 
-- 
1.8.3.1




^ permalink raw reply related	[flat|nested] 4+ messages in thread

* Re: [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs
  2019-11-27  1:11 [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs Haren Myneni
@ 2019-11-27  8:46 ` Christoph Hellwig
  2019-11-27 10:03   ` Haren Myneni
  2019-12-06  9:51 ` Haren Myneni
  1 sibling, 1 reply; 4+ messages in thread
From: Christoph Hellwig @ 2019-11-27  8:46 UTC (permalink / raw)
  To: Haren Myneni; +Cc: devicetree, mikey, herbert, npiggin, sukadev, linuxppc-dev

>  
> +static void notify_process(pid_t pid, u64 fault_addr)
> +{
> +	int rc;
> +	struct kernel_siginfo info;
> +
> +	memset(&info, 0, sizeof(info));
> +
> +	info.si_signo = SIGSEGV;
> +	info.si_errno = EFAULT;
> +	info.si_code = SEGV_MAPERR;
> +
> +	info.si_addr = (void *)fault_addr;
> +	rcu_read_lock();
> +	rc = kill_pid_info(SIGSEGV, &info, find_vpid(pid));
> +	rcu_read_unlock();
> +
> +	pr_devel("%s(): pid %d kill_proc_info() rc %d\n", __func__, pid, rc);
> +}

Shouldn't this use force_sig_fault_to_task instead?

> +	/*
> +	 * User space passed invalid CSB address, Notify process with
> +	 * SEGV signal.
> +	 */
> +	tsk = get_pid_task(window->pid, PIDTYPE_PID);
> +	/*
> +	 * Send window will be closed after processing all NX requests
> +	 * and process exits after closing all windows. In multi-thread
> +	 * applications, thread may not exists, but does not close FD
> +	 * (means send window) upon exit. Parent thread (tgid) can use
> +	 * and close the window later.
> +	 */
> +	if (tsk) {
> +		if (tsk->flags & PF_EXITING)
> +			task_exit = 1;
> +		put_task_struct(tsk);
> +		pid = vas_window_pid(window);

The pid is later used for sending the signal again, why not keep the
reference?

> +	} else {
> +		pid = vas_window_tgid(window);
> +
> +		rcu_read_lock();
> +		tsk = find_task_by_vpid(pid);
> +		if (!tsk) {
> +			rcu_read_unlock();
> +			return;
> +		}
> +		if (tsk->flags & PF_EXITING)
> +			task_exit = 1;
> +		rcu_read_unlock();

Why does this not need a reference to the task, but the other one does?

^ permalink raw reply	[flat|nested] 4+ messages in thread

* RE: [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs
  2019-11-27  8:46 ` Christoph Hellwig
@ 2019-11-27 10:03   ` Haren Myneni
  0 siblings, 0 replies; 4+ messages in thread
From: Haren Myneni @ 2019-11-27 10:03 UTC (permalink / raw)
  To: Christoph Hellwig
  Cc: mikey, herbert, npiggin, sukadev, linuxppc-dev, Haren Myneni

[-- Attachment #1: Type: text/plain, Size: 2769 bytes --]




"Linuxppc-dev" <linuxppc-dev-bounces+hbabu=us.ibm.com@lists.ozlabs.org>
wrote on 11/27/2019 12:46:09 AM:
> >
> > +static void notify_process(pid_t pid, u64 fault_addr)
> > +{
> > +   int rc;
> > +   struct kernel_siginfo info;
> > +
> > +   memset(&info, 0, sizeof(info));
> > +
> > +   info.si_signo = SIGSEGV;
> > +   info.si_errno = EFAULT;
> > +   info.si_code = SEGV_MAPERR;
> > +
> > +   info.si_addr = (void *)fault_addr;
> > +   rcu_read_lock();
> > +   rc = kill_pid_info(SIGSEGV, &info, find_vpid(pid));
> > +   rcu_read_unlock();
> > +
> > +   pr_devel("%s(): pid %d kill_proc_info() rc %d\n", __func__, pid,
rc);
> > +}
>
> Shouldn't this use force_sig_fault_to_task instead?
>
> > +   /*
> > +    * User space passed invalid CSB address, Notify process with
> > +    * SEGV signal.
> > +    */
> > +   tsk = get_pid_task(window->pid, PIDTYPE_PID);
> > +   /*
> > +    * Send window will be closed after processing all NX requests
> > +    * and process exits after closing all windows. In multi-thread
> > +    * applications, thread may not exists, but does not close FD
> > +    * (means send window) upon exit. Parent thread (tgid) can use
> > +    * and close the window later.
> > +    */
> > +   if (tsk) {
> > +      if (tsk->flags & PF_EXITING)
> > +         task_exit = 1;
> > +      put_task_struct(tsk);
> > +      pid = vas_window_pid(window);
>
> The pid is later used for sending the signal again, why not keep the
> reference?

Sorry, Not dropping the PID reference here, Happens only when window
closed. If the task for this PID is not available, looking for tgid in the
case of multi-thread process.
>
> > +   } else {
> > +      pid = vas_window_tgid(window);
> > +
> > +      rcu_read_lock();
> > +      tsk = find_task_by_vpid(pid);
> > +      if (!tsk) {
> > +         rcu_read_unlock();
> > +         return;
> > +      }
> > +      if (tsk->flags & PF_EXITING)
> > +         task_exit = 1;
> > +      rcu_read_unlock();
>
> Why does this not need a reference to the task, but the other one does?

Window is opened with open() and ioctl(fd), will be closed either by close
(fd) explicitly or release FD during process exit.

Process closes all open windows when it exits. So we do not need to keep
the reference for this case. In multi-thread case, child thread can open a
window, but it does not release FD when it exits. Parent thread (tgid) can
continue use this window and closes it upon its exit. So taking reference
to PID in case if this pid is assigned to child thread to make sure its pid
is not reused until window is closed.

We are taking pid reference during window open and releases it when closing
the window.

Thanks
Haren



>

[-- Attachment #2: Type: text/html, Size: 4030 bytes --]

^ permalink raw reply	[flat|nested] 4+ messages in thread

* RE: [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs
  2019-11-27  1:11 [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs Haren Myneni
  2019-11-27  8:46 ` Christoph Hellwig
@ 2019-12-06  9:51 ` Haren Myneni
  1 sibling, 0 replies; 4+ messages in thread
From: Haren Myneni @ 2019-12-06  9:51 UTC (permalink / raw)
  To: hch; +Cc: devicetree, mikey, herbert, npiggin, sukadev, linuxppc-dev, haren

[-- Attachment #1: Type: text/html, Size: 1714 bytes --]

^ permalink raw reply	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2019-12-06 22:49 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2019-11-27  1:11 [PATCH 09/14] powerpc/vas: Update CSB and notify process for fault CRBs Haren Myneni
2019-11-27  8:46 ` Christoph Hellwig
2019-11-27 10:03   ` Haren Myneni
2019-12-06  9:51 ` Haren Myneni

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).