From owner-freebsd-hackers@FreeBSD.ORG Mon Dec 12 19:13:50 2011 Return-Path: Delivered-To: hackers@freebsd.org Received: from mx1.freebsd.org (mx1.freebsd.org [IPv6:2001:4f8:fff6::34]) by hub.freebsd.org (Postfix) with ESMTP id 16C54106564A; Mon, 12 Dec 2011 19:13:50 +0000 (UTC) (envelope-from mdf356@gmail.com) Received: from mail-pz0-f54.google.com (mail-pz0-f54.google.com [209.85.210.54]) by mx1.freebsd.org (Postfix) with ESMTP id C55998FC0A; Mon, 12 Dec 2011 19:13:49 +0000 (UTC) Received: by dakp5 with SMTP id p5so7351381dak.13 for ; Mon, 12 Dec 2011 11:13:49 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=gamma; h=mime-version:sender:in-reply-to:references:date :x-google-sender-auth:message-id:subject:from:to:cc:content-type :content-transfer-encoding; bh=hZ8jPOrL9/CZdLmXWzfhaF0zRX46QB3ln3lyS4C1Nj4=; b=XsIk2CzLNkx75Vz5BI0J9APskBH6JnlnkLfF2lnpVB9y5R+ekfHu0Ygyp5stVQ2Idy ZASq6qD+/uIZE+EXuhkXN5URg1TEHwRRFffJH2fAXYk+zkiQlD1sQHs58OXfOOKgM+9c hl6q7h2Ih+ObuO8vYkiBnE9B4ye2RYennGADM= MIME-Version: 1.0 Received: by 10.68.73.197 with SMTP id n5mr36556308pbv.102.1323717229215; Mon, 12 Dec 2011 11:13:49 -0800 (PST) Sender: mdf356@gmail.com Received: by 10.68.197.198 with HTTP; Mon, 12 Dec 2011 11:13:49 -0800 (PST) In-Reply-To: <201112121405.38322.jhb@freebsd.org> References: <4EE51CB5.1060505@FreeBSD.org> <201112121658.22864.hselasky@c2i.net> <201112121405.38322.jhb@freebsd.org> Date: Mon, 12 Dec 2011 11:13:49 -0800 X-Google-Sender-Auth: tA1WgJS5KaNFUvDe4pesRX4T_5A Message-ID: From: mdf@FreeBSD.org To: John Baldwin Content-Type: text/plain; charset=ISO-8859-1 Content-Transfer-Encoding: quoted-printable Cc: usb@freebsd.org, freebsd-hackers@freebsd.org, hackers@freebsd.org, Andriy Gapon , Hans Petter Selasky Subject: Re: kern_yield vs ukbd_yield X-BeenThere: freebsd-hackers@freebsd.org X-Mailman-Version: 2.1.5 Precedence: list List-Id: Technical Discussions relating to FreeBSD List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Mon, 12 Dec 2011 19:13:50 -0000 On Mon, Dec 12, 2011 at 11:05 AM, John Baldwin wrote: > On Monday, December 12, 2011 10:58:22 am Hans Petter Selasky wrote: >> On Monday 12 December 2011 16:55:38 mdf@freebsd.org wrote: >> > On Mon, Dec 12, 2011 at 12:03 AM, Andriy Gapon wrote= : >> > > on 11/12/2011 23:48 mdf@FreeBSD.org said the following: >> > >> On Sun, Dec 11, 2011 at 1:12 PM, Andriy Gapon wro= te: >> > >>> Does the following change do what I think that it does? >> > >>> Thank you! >> > >>> >> > >>> Author: Andriy Gapon >> > >>> Date: =A0 Thu Sep 1 16:50:13 2011 +0300 >> > >>> >> > >>> =A0 =A0ukbd: drop local duplicate of kern_yield and use that inste= ad >> > >>> >> > >>> diff --git a/sys/dev/usb/input/ukbd.c b/sys/dev/usb/input/ukbd.c >> > >>> index 086c178..8078cbb 100644 >> > >>> --- a/sys/dev/usb/input/ukbd.c >> > >>> +++ b/sys/dev/usb/input/ukbd.c >> > >>> @@ -399,33 +399,6 @@ ukbd_put_key(struct ukbd_softc *sc, uint32_t = key) >> > >>> =A0} >> > >>> >> > >>> =A0static void >> > >>> -ukbd_yield(void) >> > >>> -{ >> > >>> - =A0 =A0 =A0 struct thread *td =3D curthread; >> > >>> - =A0 =A0 =A0 uint32_t old_prio; >> > >>> - >> > >>> - =A0 =A0 =A0 DROP_GIANT(); >> > >>> - >> > >>> - =A0 =A0 =A0 thread_lock(td); >> > >>> - >> > >>> - =A0 =A0 =A0 /* get current priority */ >> > >>> - =A0 =A0 =A0 old_prio =3D td->td_base_pri; >> > >>> - >> > >>> - =A0 =A0 =A0 /* set new priority */ >> > >>> - =A0 =A0 =A0 sched_prio(td, td->td_user_pri); >> > >>> - >> > >>> - =A0 =A0 =A0 /* cause a task switch */ >> > >>> - =A0 =A0 =A0 mi_switch(SW_INVOL | SWT_RELINQUISH, NULL); >> > >>> - >> > >>> - =A0 =A0 =A0 /* restore priority */ >> > >>> - =A0 =A0 =A0 sched_prio(td, old_prio); >> > >>> - >> > >>> - =A0 =A0 =A0 thread_unlock(td); >> > >>> - >> > >>> - =A0 =A0 =A0 PICKUP_GIANT(); >> > >>> -} >> > >>> - >> > >>> -static void >> > >>> =A0ukbd_do_poll(struct ukbd_softc *sc, uint8_t wait) >> > >>> =A0{ >> > >>> >> > >>> @@ -439,7 +412,7 @@ ukbd_do_poll(struct ukbd_softc *sc, uint8_t wa= it) >> > >>> =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0while (sc->sc_inputs =3D=3D 0) { >> > >>> >> > >>> =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0/* give USB threads= a chance to run */ >> > >>> - =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 ukbd_yield(); >> > >>> + =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 kern_yield(-1); >> > >> >> > >> Not quite. >> > >> >> > >> 1) -1 should be spelled PRI_UNCHANGED, except ukbd_yield() uses >> > >> td_user_pri, but then puts it back again, so I think UNCHANGED is w= hat >> > >> is meant. >> > >> 2) kern_yield() calls it a SW_VOL rather than SW_INVOL, which seems >> > >> the desired behaviour here anyways, since this is an explicit (i.e. >> > >> voluntary) yield. >> > > >> > > Thank you for the explanation. =A0So would you say that the patch is= OK? >> > >> > As far as I know, yes. =A0There may be a difference in behaviour, >> > though, while yielding, if the priority of the thread remains high (as >> > this change would make it) -- I'm not completely sure how the >> > scheduler chooses threads, because I'm pretty sure I've seen it take >> > threads with lower (higher numbered) priorities even when there's >> > runnable threads with a higher (lower numbered) priority available. > > The scheduler generally should not do this with the following exceptions: > > =A01) for timesharing threads, priorities are in bands, so effectively th= e > =A0 =A0pri / 4 is what is used for comparison and if two threads have the= same > =A0 =A0pri / 4 the scheduler will round-robin among htem. > > =A02) ULE might be a bit different because of the way it assigns threads = to > =A0 =A0CPUs, so if a CPU has two high priority threads and another CPU on= ly > =A0 =A0has a low priority thread, the second CPU will not run the second = high > =A0 =A0priority thread. =A04BSD handles this case more correctly. > >> > It has always seemed weird to me that the priorities in the kernel are >> > strictly higher than user-space -- but only after a prio change like >> > that done implicitly by many of the calls to sleep(9). =A0So it may be >> > that the better patch is to use PRI_USER, not PRI_UNCHANGED, which >> > would revert any potentially changed thread prio (e.g. due to a >> > sleep(9)) back to its user-space level, so that it contended as >> > expected with other threads. > > Realtime priorities (for rtprio user threads) are higher than the kernel > "sleep" priorities. =A0Also, keep in mind that a thread does not get an > automatic priority boost when it enters the kernel. =A0It only gets a boo= st > either temporarily from priority propagation, or a slightly longer (but > still temporary) boost from a sleep(9) call. I still don't understand why threads are boosting their priority while sleeping; if it's so they are woken preferentially, that makes some sense, but then they shouldn't be keeping the boosted priority after the thread is woken. If a thread really needs higher priority to do its work, that should be an explicit call to sched_prio(9), rather than implicitly only when/if it first sleeps. Thanks, matthew >> > hselasky@ or someone else familiar with the various usb threads would >> > have to answer that. >> >> The problem is only during init() where the init thread has highest prio= rity >> and that doesn't allow other threads to run even if the scheduler is > running! > > Hmm, that should be fixed by lowering the relevant thread's priority. > Do you mean thread0 (the one doing all the SYSINIT's or thread we create = for > init (pid 1) before it executes init? > > -- > John Baldwin