[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20100924165716.GR19804@ZenIV.linux.org.uk>
Date: Fri, 24 Sep 2010 17:57:16 +0100
From: Al Viro <viro@...IV.linux.org.uk>
To: Linus Torvalds <torvalds@...ux-foundation.org>
Cc: tglx@...utronix.de, mingo@...hat.com, linux-kernel@...r.kernel.org
Subject: Re: what's papered over by set_fs(USER_DS) in amd64 signal delivery?
On Fri, Sep 24, 2010 at 09:07:33AM -0700, Linus Torvalds wrote:
> I _think_ it is historical, and probably relates to just restoring all
> the user mode state at signal delivery to a known state. IOW, I think
> it really does go hand-in-hand with the whole "clear bits in the
> eflags register" thing.
>
> x86-64 has historically had some left-over crap that we already
> cleaned up in 32-bit mode, for the simple reason that the original
> x86-64 code was forked from an earlier base, and then hacked up
> somewhat. So I think this "#ifdef CONFIG_X86_64" is just a case of
> that.
Nope - look at the commit that merged them; i386 used to have it as well
right until the merge. And yes, I agree that it's most likely a leftover
from ancient times.
> But maybe we should have a WARN_ON_ONCE() to verify it, rather than
> just kill it outright.
Um... If we do that, I'd suggest taking it in front of setup_...frame()
and doing do_exit() with loud warning along the lines of "someone has
come that -><- close to rooting the box". Anybody who can trigger that
(assuming it can be triggered at all) can overwrite arbitrary parts of
kernel memory.
I'd still like to hear from the x86 maintainers, just in case there's
something more non-trivial to that than "it has always been that way"...
FWIW, looking at the history it seems that the thing has happened almost
by accident; the critical part is
- regs->cs = USER_CS; regs->ss = USER_DS;
- regs->ds = USER_DS; regs->es = USER_DS;
- regs->gs = USER_DS; regs->fs = USER_DS;
+ {
+ unsigned long seg = USER_DS;
+ __asm__("mov %w0,%%fs ; mov %w0,%%gs":"=r" (seg) :"0" (seg));
+ set_fs(seg);
+ regs->xds = seg;
+ regs->xes = seg;
+ regs->xss = seg;
+ regs->xcs = USER_CS;
in 2.1.2. And that's when we had
* fs and gs evicted from pt_regs
* fs and gs not saved restored on kernel entry/exit
* just introduced set_fs() to start with (that went in 2.1.0)
A bit before my time, so I'm not sure what's been going on there...
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists