Closed
Bug 506693
Opened 15 years ago
Closed 11 years ago
SELinux is preventing JIT from changing memory segment access
Categories
(Core :: JavaScript Engine, defect)
Core
JavaScript Engine
Tracking
()
People
(Reporter: dveditz, Unassigned)
References
Details
Attachments
(2 files, 21 obsolete files)
177.32 KB,
patch
|
Details | Diff | Splinter Review | |
265.76 KB,
patch
|
Details | Diff | Splinter Review |
Jan Lieskovsky from Red Hat Security Response Team reported the following to the Mozilla Security via mail:
while using Firefox-3.5 version if Fedora 11
(firefox-3.5.1-1.fc11), we noticed the following
security problem, related with the new Firefox's 3.5
JavaScript Just-In-Time (JIT) compiler (SELinux setroubleshoot message):
The problem:
===========
<cite>
Summary:
SELinux is preventing firefox from changing a writable memory segment
executable.
Detailed Description:
[SELinux is in permissive mode, the operation would have been denied but was
permitted due to permissive mode.]
The firefox application attempted to change the access protection of memory
(e.g., allocated using malloc). This is a potential security problem.
Applications should not be doing this. Applications are sometimes coded
incorrectly and request this permission. The SELinux Memory Protection Tests
(http://people.redhat.com/drepper/selinux-mem.html) web page explains how to
remove this requirement. If firefox does not work and you need it to work, you
can configure SELinux temporarily to allow this access until the application is
fixed. Please file a bug report
(http://bugzilla.redhat.com/bugzilla/enter_bug.cgi) against this package.
Allowing Access:
If you trust firefox to run correctly, you can change the context of the
executable to execmem_exec_t. "chcon -t execmem_exec_t
'/usr/lib/firefox-3.5/firefox'". You must also change the default file context
files on the system in order to preserve them even on a full relabel. "semanage
fcontext -a -t execmem_exec_t '/usr/lib/firefox-3.5/firefox'"
Fix Command:
chcon -t execmem_exec_t '/usr/lib/firefox-3.5/firefox'
</cite>
Analysis and argumentation:
===========================
After further investigation / review it implied, this is a potential security
issue related with the way, the new Just-In-Time compiler is designed.
Here is the argumentation from Ulrich Drepper what's related to this issue:
<snip>
It's simply not acceptable that the program which is most susceptible to the
problems the Internet exposes a machine is the one which doesn't follow the
rules for secure programming. If the new JIT requires writable and executable
memory it either has to be rewritten (have two mappings for the same data: one
writable, one executable) or it has to be disabled.
</snip>
Conclusion:
===========
This results to the following proposed solution -- if newest Mozilla's Firefox 3.5
JavaScript JIT compiler needs to map relevant memory area for both, writing and
execution, it should do that in two separate requests. Because as Ulrich
already pointed out, the current implementation is prone to security attacks,
as demonstrated in the wild by:
https://bugzilla.mozilla.org/show_bug.cgi?id=503286
(CVE-2009-2477 CVE-2009-2478 CVE-2009-2479)
Separation of 'write' and 'exec' memory map requests into two separate requests
could prevent occurrance of such exploits / flaws in the future.
Ulrich also recommends / provides sample code, how this problem can be
solved in a secure way:
http://people.redhat.com/drepper/selinux-mem.html
Could you have a further look into this issue and fix / reimplement relevant
parts of Mozilla Firefox's 3.5 JIT compiler to address this deficiency?
Reporter | ||
Updated•15 years ago
|
Whiteboard: [sg:investigate]
Comment 1•15 years ago
|
||
The basic SELinux problem is that you can't map writable and executable at the same time, correct?
I don't see any reason this needs to be private: it's well known.
Comment 2•15 years ago
|
||
Could I ask for adding stransky@redhat.com to the CC list so he can see this?
Comment 3•15 years ago
|
||
(In reply to comment #2)
> Could I ask for adding stransky@redhat.com to the CC list so he can see this?
Done.
Reporter | ||
Updated•15 years ago
|
blocking1.9.1: ? → ---
status1.9.1:
--- → ?
Comment 4•15 years ago
|
||
I'll note that I agree this should not be private.
Also, this bug will block Fedora 12 which will deny execmem by default, in SELinux enforcing mode. We had to revert that feature temporarily to be able to ship our alpha, but the intention is to ship it for final.
Comment 5•15 years ago
|
||
Anyone there who can help with a patch to the JIT to get the better behaviour?
Updated•15 years ago
|
Assignee: general → gal
Comment 6•15 years ago
|
||
I think graydon's merge work will help this, but I assume we have to fix this for 3.5.x so we are going to have to patch the old fragmento. I have a patch for r/o code fragments somewhere in bugzilla.
Comment 7•15 years ago
|
||
Yeah. I'm hesitant to accept Ulrich's reasoning on this. The theory is that someone who wants to attack us might be able to pull off said attack by finding a bug and exploiting it to write to executable memory (this much I agree with) but is then *not* going to be able to pull off the exact same attack when it's done via "dual mapping", writing their attack into the W mapping and then convincing the program to run it via the X mapping.
The argument rests on the belief that "twice random = more random". It's a nice thought, but it's also a lot of work to implement, and I don't see anything other than wishful thinking motivating it. Just this notion that "since it's more complicated, it'll be secure". If that were true, we could just throw in a few more randomized hashtables in the middle of our page-mapping data structures and declare ourselves secure.
Comment 8•15 years ago
|
||
I think we have a unique opportunity with traces that we can actually implement a read-only code cache, since traces are specialized at recording time and don't require on-the-fly patching like PICs do. The previous patch was pretty expensive because we used to have 4k pages and we had to set and revoke the execute bit for each page individually. With the new code allocator that should be less of an issue. What I still don't know how to solve is how to address concurrent code cache access. As long we keep one code cache per thread though, that should be fine.
Comment 9•15 years ago
|
||
Dan, I would like to unhide the bug to get more feedback. I am pretty sure its widely known that our code cache is write and executable, and I don't think this bug contains information that warrants to hide it.
Comment 10•15 years ago
|
||
For what it's worth, I agree completely with comment 7. Executing dynamically generated code is what can get you into trouble; you still have that with the proposed workaround.
I also agree completely with comment 9. Are there any other serious JIT engines which actually do the two-views-of-one-page hackaround? I'm not aware of any, which makes me think anyone malicious is going to expect precisely what we do without having looked, and it's not exactly hard to look and verify that assumption in any case.
Updated•15 years ago
|
Group: core-security
Comment 11•15 years ago
|
||
Let me say one thing first:
getting writable and executable memory in any form is nowadys a privilege, not a right.
I think even today's rules as implemented in SELinux are to loose and should be tightened even further.
In this light those who insist on having writable memory have the obligation to do everything possible to justify the trust. Reading Graydon's comments does exactly the opposite.
Of course using the double-mapping scheme doesn't solve the problem 100%. Only disallowing writable&executable memory completely does it. If this is what you prefer, fine, it's easy enough to implement in the OS.
But this is not in the best interest of everybody and therefore we have to mitigate the risk with all means. The double mapping does help, as simple math tells you. If an attacker can guess the address range for a mapping with, let's say, 5% (very high) then guessing both addresses at the same time is possible in only 0.25% of the time. That's a significant reduction.
If only on address is needed an attacker can write position-independent code which just has to be injected. On x86-64 we have an appropriate addressing mode (relative to the %rip). Using this makes it far too easy to write exploits. If the writable region is elsewhere this becomes much harder.
It is no extravagant request/demand to fix this. It is just common sense. Firefox is *by far* the most problematic application, security-wise. 5 out of 7 critical advisories for RHEL5.3 were for firefox (see Mark Cox's blog). *Anything* that can help here, reducing the severity level or whatever, must be done.
The changes also shouldn't be that problematic, unless the code is truly chaotic:
- change buffer management to keep two buffers aligned in size
- wrap all places where code/data is injected
- do some pointer arithmetic before the actual poking
If there are ways in which we (= Red Hat) can help let any of us know. We'll see what can be done.
Comment 12•15 years ago
|
||
Personally I am shocked security folks are floating the double mapped memory hack. It seems like false security through obscurity to me, with very little randomization, even if we accept your math above. In my little academic ivory tower, executable and writable bits shouldn't co-exist, and the kernel should enforce this. I would prefer implementing this strategy in ff, but unfortunately it seems to have a pretty significant performance cost (ballpark 5% last time I measured).
The root cause for this overhead is the fact that dynamically typed programming languages like JS require frequent code cache updates, either because code is rewritten on the fly (PICs, V8 says hello) or in case of TraceMonkey a lot of specialized read-only code fragments are assembled in very quick succession, interleaved with actual code execution. Flipping the read/write/exec bits every time we update the code cache seems impractical. The syscall takes a small eternity, not to mention the havoc that causes at the hardware level (TLBs and what not).
An additional complication I ran into is concurrent code generation. I am playing around with a background compiler thread, and revoking the execute bit becomes extremely tricky without tripping the foreground thread that runs code from the same code cache.
Both frequent code cache updates and concurrent updates would be overhead free with the proposed double mapped page approach, so I am willing to go that route as long there is buy-in from our security folks that this actually a valid fix for the writable code cache issue.
As for an actual implementation, its not as an easy 1. 2. 3. as in the above post. Our code cache is chunked, and native code fragments can span multiple regions in multiple chunks. Figuring out for each region what the local mapping to the sibling page(s) is, is not going to be trivial, especially if we want the offset to differ for each chunk (which we want for randomization, but also because address space fragmentation will likely not allow to always get additional pages with a constant offset).
Ed, what is your take on this?
Comment 13•15 years ago
|
||
(In reply to comment #12)
> The root cause for this overhead is the fact that dynamically typed programming
> languages like JS require frequent code cache updates, either because code is
> rewritten on the fly (PICs, V8 says hello) or in case of TraceMonkey a lot of
> specialized read-only code fragments are assembled in very quick succession,
> interleaved with actual code execution. Flipping the read/write/exec bits every
> time we update the code cache seems impractical.
Nobody suggests this. With two mappings you can always have a writable segment. It seems you haven't even tried to understand what I described in the page linked in the original report.
Comment 14•15 years ago
|
||
(In reply to comment #13)
> Nobody suggests this. With two mappings you can always have a writable
> segment. It seems you haven't even tried to understand what I described in the
> page linked in the original report.
*Andreas* suggests this, since as you say the current different-mapping limitation is a mitigation, not a solution to the problem. It would actually solve the problem if we didn't have any memory that was concurrently mapped executable and writable (via one or multiple address regions, the attacker probably doesn't care), but unfortunately the performance characteristics of making such changes frequently aren't really viable.
It seems you haven't tried to understand the comment to which you replied?
Thanks for your offer of help -- I think that Graydon and Andreas can point you to the appropriate code, and I know they'd be willing to review a patch.
Comment 15•15 years ago
|
||
For case this would be helpful for you (to identify particular code parts).
This was reported in relevant Fedora bug BZ#509945:
<cite>
This is where xulrunner traces back when executable/writable mappings are
disallowed:
Program received signal SIGSEGV, Segmentation fault.
nanojit::LirBufWriter::ins0 (op=<value optimized out>, this=<value optimized
out>, this=<value optimized out>,
op=<value optimized out>) at nanojit/LIR.cpp:315
315 l->initOpcode(op);
Missing separate debuginfos, use: debuginfo-install firefox-3.5-2.fc12.i586
(gdb) bt
#0 0x00a3890d in nanojit::LirBufWriter::ins0 (op=<value optimized out>,
this=<value optimized out>,
this=<value optimized out>, op=<value optimized out>) from
/usr/lib/xulrunner-1.9.1/libmozjs.so
#1 0x00a0202f in RegExpNativeCompiler::compile (this=0xbfffa26c,
cx=0xb1ba2c00) at jsregexp.cpp:2411
#2 0x009f94aa in CompileRegExpToNative (re=<value optimized out>, cx=<value
optimized out>, fragment=<value optimized out>)
at jsregexp.cpp:2475
#3 GetNativeRegExp (re=<value optimized out>, cx=<value optimized out>,
fragment=<value optimized out>)
at jsregexp.cpp:2510
#4 MatchRegExp (re=<value optimized out>, cx=<value optimized out>,
fragment=<value optimized out>) at jsregexp.cpp:3922
#5 js_ExecuteRegExp (re=<value optimized out>, cx=<value optimized out>,
fragment=<value optimized out>)
at jsregexp.cpp:4090
#6 0x009fff92 in regexp_exec_sub (cx=0xb1ba2c00, obj=<value optimized out>,
argc=1, argv=0xb15cd0c0, test=1,
rval=0xb15cd0b8) at jsregexp.cpp:4889
#7 0x00a00062 in regexp_test (cx=0xb1ba2c00, argc=1, vp=0xb15cd0b8) at
jsregexp.cpp:4911
#8 0x009c8d99 in js_Interpret (cx=0xb1ba2c00) at jsinterp.cpp:5147
Just debuginfo-install xulrunner and you'll see several instances of such
mappings:
/usr/src/debug/xulrunner-1.9.1/mozilla-1.9.1/js/src/nanojit/Assembler.cpp
/usr/src/debug/xulrunner-1.9.1/mozilla-1.9.1/js/src/nanojit/Assembler.h
/usr/src/debug/xulrunner-1.9.1/mozilla-1.9.1/js/src/nanojit/Nativei386.cpp
/usr/src/debug/xulrunner-1.9.1/mozilla-1.9.1/js/src/nanojit/avmplus.h
And a rather funny comment as well:
#elif defined AVMPLUS_UNIX
/**
* Don't use normal heap with mprotect+PROT_EXEC for executable
code.
* SELinux and friends don't allow this.
*/
return mmap(NULL,
pages * kNativePageSize,
PROT_READ | PROT_WRITE | PROT_EXEC,
MAP_PRIVATE | MAP_ANON,
-1,
0);
#else
</cite>
Cc-ed also original Fedora reporter Lubomir Rintel to this bug.
Comment 16•15 years ago
|
||
Thanks, we know where the regions are coming from, it's all quite centralized.
I wish I could say Ulrich's response "shocks" me, but it's what I knew we'd get here. I still don't think this will make us one iota safer: heapspray though the writable mapping (of the sort we just were exploited by) will still work, and %rip-relative addressing is a red herring (you can build PIC code if you can find a single GOT-derived address in a register or on the stack). But that's beside the point.
The plain fact is that one can't win an argument like this with Ulrich, so we really only have two options: do what he says, or close the bug and/or maintain a perpetual disagreement with him. I don't have the energy for the latter. It'll be less work to just accept his preference and implement it. Let's put this on the "post merge" list of feature work on the code allocator and be done with it.
Comment 17•15 years ago
|
||
I'm working on a patch based on the double-mapped pages. The page allocating/altering seems to work but I expect the code is generated as position dependent and generated relatively to the write-pages. Can you point me where I should look when I want to generate the code relatively to the execution pages?
Comment 18•15 years ago
|
||
(In reply to comment #17)
> I'm working on a patch based on the double-mapped pages. The page
> allocating/altering seems to work but I expect the code is generated as
> position dependent and generated relatively to the write-pages. Can you point
> me where I should look when I want to generate the code relatively to the
> execution pages?
This is going to be a long effort. Look in the backends, such as nanojit/Nativei386.cpp and its associated header nanojit/Nativei386.h. See, for example, the macro JMP, which composes a pc-relative jump instruction. This is based on the distance to its target, but its target may not be in the same contiguous chunk of memory / page. So if you're going to have each writable page correspond to an executable page, all such arithmetic is going to have to change to go indirectly through a mapping table that transforms "addresses in a writable page" to "addresses in an executable page".
This is why I said "it'll be a lot of work". The code might be position-independent (or it might be possible to *make* it position-independent) but it's certainly not the case that all code referencing into or out of page X is contained within page X. Or even "multi-page chunk X". Instructions are written repeatedly into tiny bits of memory scattered all over the CodeAlloc's pages. You're going to be doing a lot of book-keeping to map between each writable bit and its executable partner.
Comment 19•15 years ago
|
||
First work in progress version, applies to 1.9.1.12 and passes trace-test.js. Should work fine (just need to change home dir at GCHeap::Alloc().
A short description&discussion:
- Each Page has attached its executable mirror (exec pointer at PageHeader)
- write_to_exec() translates any address at write page to corresponding executable page. Page verification is managed by PAGE_SIGNATURE hack (should we traverse the page list here? disable the check? or something else?)
- calcOffset() macro calculates offset in executable space. Note that the source address has to be decremented some code is generated from end of the page so effective jump offset is actually calculated relative to next (but unrelated) page. So we have to differentiate source & target address.
- CALL (c->_address) is not translated, supposed to be in code segment only. Do we even use call into generated code?
- CALLr(c,r) is ignored for now.
- GCHeap::Alloc() has to map some sane tmp file.
Updated•15 years ago
|
Assignee: gal → stransky
Comment 20•15 years ago
|
||
Why is the PAGE_SIGNATURE hack necessary? Is that intended for debugging only? Why would we ever touch a page we don't have to translate?
Comment 21•15 years ago
|
||
(In reply to comment #20)
> Why is the PAGE_SIGNATURE hack necessary? Is that intended for debugging only?
> Why would we ever touch a page we don't have to translate?
I used it for debugging and because I don't know how the code is composed and if all translated address/jump targets are from our page space (e.g. destinations of CALL()). It can be removed if we don't translate any foreign address.
From my current experience (run trace-test.js) the PAGE_SIGNATURE check is not necessary.
Comment 22•15 years ago
|
||
(In reply to comment #20)
> Why is the PAGE_SIGNATURE hack necessary? Is that intended for debugging only?
> Why would we ever touch a page we don't have to translate?
btw. The PAGE_SIGNATURE check could be simplified to check whether the executable page points to itself:
assert(pageExec->exec == pageExec);
Another options is to add a reference to write page to the page header so we can provide full exec <=> write translation (and checks) like so:
struct PageHeader
{
struct Page *next;
struct Page *write;
struct Page *exec;
};
assert(pageWrite->write == pageWrite);
pageExec = pageWrite->exec;
assert(pageExec->exec == pageExec);
Anyway I'd like to hear if you agree with the overall conception. If so I'll provide some polished version.
Comment 23•15 years ago
|
||
(In reply to comment #19)
> Created an attachment (id=399047) [details]
> WIP
>
> First work in progress version, applies to 1.9.1.12 and passes trace-test.js.
> Should work fine (just need to change home dir at GCHeap::Alloc().
You're working on old code (perhaps 3.5 release series?)
We replaced the entire allocation system after that release, and it is no longer based on single-page allocations or a Fragmento. Please base your work on trunk, unless you're just hoping to backport to 3.5.
Comment 24•15 years ago
|
||
Oops, I just learned that the new allocator is not even on 1.9.2 / 3.6 branch yet. So we're ... still in transition. But shifting 1.9.2 to the new allocator is proposed. May or may not happen. Trunk definitely has switched, as has Adobe's code, and it's the interface we're hoping to use for the future.
Comment 25•15 years ago
|
||
Martin, I am looking through your patch. I will give you some feedback based on it. As Graydon pointed out we reworked the code allocator path, so we would have to port to that. 3.6 will be based off the old code afaik.
Comment 26•15 years ago
|
||
Looks like the new code allocator _will_ land for 3.6, so this makes reworking the patch for the new CodeAlloc even more important.
Comment 27•15 years ago
|
||
(In reply to comment #23)
> We replaced the entire allocation system after that release, and it is no
> longer based on single-page allocations or a Fragmento. Please base your work
> on trunk, unless you're just hoping to backport to 3.5.
Yes, we (Red Hat) really want it for Fedora 12/ff3.5. But I will definitely work on a patch for the trunk too.
Comment 28•15 years ago
|
||
Here is the updated trunk version.
CodeAlloc::freePage(), CodeAlloc::allocPage() manage the physical memory allocation and CodeAlloc::getPage(NIns* p) returns appropriate page (CodeList *) for p. Address translation is realised by CodeList::toExec().
(btw. If pagesPerAlloc = 1 we can drop the page traversal at CodeAlloc::getPage() and just round up the address like in the first patch)
Comment 29•15 years ago
|
||
Ping, any feedback here?
Comment 30•15 years ago
|
||
Sorry for the delay. I will review the patch.
Comment 31•15 years ago
|
||
Comment on attachment 402318 [details] [diff] [review]
WIP v2 (for trunk)
>
> #ifdef NANOJIT_IA32
> void Assembler::patch(SideExit* exit, SwitchInfo* si)
> {
> for (GuardRecord* lr = exit->guards; lr; lr = lr->next) {
> Fragment *frag = lr->exit->target;
> NanoAssert(frag->fragEntry != 0);
>- si->table[si->index] = frag->fragEntry;
>+ si->table[si->index] = TO_EXEC(frag->fragEntry);
We are trying to get rid of macro magic in NJ. If there is no strong reason to use a macro here, please don't.
> underrunProtect(si->count * sizeof(NIns*) + 20);
> _nIns = reinterpret_cast<NIns*>(uintptr_t(_nIns) & ~(sizeof(NIns*) - 1));
> for (uint32_t i = 0; i < si->count; ++i) {
> _nIns = (NIns*) (((uint8*) _nIns) - sizeof(NIns*));
>- *(NIns**) _nIns = target;
>+ *(NIns**) _nIns = TO_EXEC(target);
> }
Why not adjust this outside the loop?
>+nanojit::CodeAlloc::allocCodeChunk(size_t nbytes, void *&exec) {
>+ char tmpfname[] = "/home/komat/.execmemXXXXXX";
Whats the final plan for this? /tmp or env[HOME]? Are there any security implications here? Reliability issues? What if the file exists at startup? Make sure you understand all aspects of this (because I am not sure I do).
> #elif defined AVMPLUS_UNIX
> // fixme: __clear_cache is a libgcc feature, test for libgcc or gcc
> void CodeAlloc::flushICache(CodeList* &blocks) {
> for (CodeList *b = blocks; b != 0; b = b->next) {
>+ // TODO -> translate to exec space?
Yeah, we should translate here for sure.
> __clear_cache((char*)b->start(), (char*)b->start()+b->size());
> }
> }
> #endif // AVMPLUS_MAC && NANOJIT_PPC
>
> void CodeAlloc::addBlock(CodeList* &blocks, CodeList* b) {
> b->next = blocks;
> blocks = b;
> }
>
>+ /** return true if just this block contains p */
>+ bool contains(NIns* p, uintptr_t blockSize) { return uintptr_t(p) >= uintptr_t(start()) &&
>+ uintptr_t(p) < uintptr_t(start()) + blockSize; }
Style. Use
foo
{
}
>+
>+ bool hasExecPage() { return write != exec; }
>+
>+ /** converts address to exec page */
>+ NIns* toExec(NIns *p)
>+ {
>+ if(!p || !hasExecPage())
>+ return(p);
Can these cases really happen? Might be better to assert here and only hand in useful conversion requests.
>+
>+ assert(p >= write->start());
>+ return (NIns*)(uintptr_t(exec) + (uintptr_t(p) - uintptr_t(write)));
>+ }
>+
>+ /** converts address to write page */
>+ NIns* toWrite(NIns *p)
>+ {
>+ if(!p || !hasExecPage())
>+ return(p);
Dito here.
>+
>+ assert(p >= exec->start());
>+ return (NIns*)(uintptr_t(write) + (uintptr_t(p) - uintptr_t(exec)));
>+ }
>+ };
>
Looks promising. Thanks for working on this. The next round we should ask someone from Adobe to review the patch.
Comment 32•15 years ago
|
||
> > underrunProtect(si->count * sizeof(NIns*) + 20);
> > _nIns = reinterpret_cast<NIns*>(uintptr_t(_nIns) & ~(sizeof(NIns*) - 1));
> > for (uint32_t i = 0; i < si->count; ++i) {
> > _nIns = (NIns*) (((uint8*) _nIns) - sizeof(NIns*));
> >- *(NIns**) _nIns = target;
> >+ *(NIns**) _nIns = TO_EXEC(target);
> > }
>
> Why not adjust this outside the loop?
Oh, sure, it's just overlooked.
> >+nanojit::CodeAlloc::allocCodeChunk(size_t nbytes, void *&exec) {
> >+ char tmpfname[] = "/home/komat/.execmemXXXXXX";
>
> Whats the final plan for this? /tmp or env[HOME]? Are there any security
> implications here? Reliability issues? What if the file exists at startup? Make
> sure you understand all aspects of this (because I am not sure I do).
Me neither but http://gcc.gnu.org/viewcvs/trunk/libffi/src/closures.c?revision=150042&view=markup contains some ideas how to manage it. And I believe Ulrich can help here too.
> >+
> >+ bool hasExecPage() { return write != exec; }
> >+
> >+ /** converts address to exec page */
> >+ NIns* toExec(NIns *p)
> >+ {
> >+ if(!p || !hasExecPage())
> >+ return(p);
>
> Can these cases really happen? Might be better to assert here and only hand in
> useful conversion requests.
Sometimes offsets are NULL. And !hasExecPage() happens when the douple-page design is not enabled (win and other arches).
> Looks promising. Thanks for working on this. The next round we should ask
> someone from Adobe to review the patch.
Thanks. If you agree with the overall conception I'll move toward x86_64 implementation and the correct tmpfname[].
Comment 33•15 years ago
|
||
(In reply to comment #32)
> Me neither but
> http://gcc.gnu.org/viewcvs/trunk/libffi/src/closures.c?revision=150042&view=markup
> contains some ideas how to manage it. And I believe Ulrich can help here too.
The libffi code is the best possible solution. It covers all the bases. There are systems where home cannot be used for mmap and others where /tmp is mounted with noexec. There must be one place where mmap(PROT_EXEC) is possible and this code should find it. I suggest you implement the same for moz.
Comment 34•15 years ago
|
||
We should make sure the patch does not cause any overhead for systems where this won't be enabled.
Comment 35•15 years ago
|
||
Third version, includes the dynamic file allocation, i386 only (for now).
Attachment #402318 -
Attachment is obsolete: true
Attachment #405024 -
Flags: review?(gal)
Comment 36•15 years ago
|
||
Comment on attachment 405024 [details] [diff] [review]
v3
>- {
>+ {
Bogus tab at the end.
>+ // TODO -> Do we need to adjust something here for the separated read/write pages?
> for (CodeList *b = blocks; b != 0; b = b->next)
> VALGRIND_DISCARD_TRANSLATIONS(b->start(), b->size());
Lets get some feedback from Nick or Julian on this.
>+ /** convert address to exec page */
>+ NIns* toExec(NIns *p)
>+ {
>+ if(!p || !hasExecPage())
>+ return p;
Why is this not an assert? How can an already executable page end up here?
>+ {
>+ if(!p || !hasExecPage())
>+ return p;
>+
Dito.
Lets get Ed to take a look too. I am pretty sure Adobe will want 0 overhead for this on Win32 if we don't use it there, so we would have to provide empty toWrite toExec wrappers for all platforms where this isn't going to be used.
Attachment #405024 -
Flags: review?(edwsmith)
Comment 37•15 years ago
|
||
(In reply to comment #36)
>
> >+ // TODO -> Do we need to adjust something here for the separated read/write pages?
> > for (CodeList *b = blocks; b != 0; b = b->next)
> > VALGRIND_DISCARD_TRANSLATIONS(b->start(), b->size());
>
> Lets get some feedback from Nick or Julian on this.
I don't know this code, but it looks reasonable. In general, if you generate code over the top of previously executed code, Valgrind needs to be told about the affected memory region via VALGRIND_DISCARD_TRANSLATIONS. And that's what the above code seems to do.
Comment 38•15 years ago
|
||
Should address the comments. Added different toExec/calcOffset methods for unsupported platforms and fixed the valgrind notification.
Attachment #405024 -
Attachment is obsolete: true
Attachment #406206 -
Flags: review?(gal)
Attachment #405024 -
Flags: review?(gal)
Attachment #405024 -
Flags: review?(edwsmith)
Comment 39•15 years ago
|
||
Ah, hg diff ignores new files so the v4 is missing the MemMap class. Attaching a correct patch now.
Attachment #406206 -
Attachment is obsolete: true
Attachment #406207 -
Flags: review?(gal)
Attachment #406206 -
Flags: review?(gal)
Updated•15 years ago
|
Attachment #406207 -
Flags: review?(rreitmai)
Attachment #406207 -
Flags: review?(edwsmith)
Comment 40•15 years ago
|
||
You should add this to your .hgrc:
# Use git diffs for binary diffing, emulate -p
[diff]
git = 1
showfunc = true
Comment 41•15 years ago
|
||
Ping - any update here?
Comment 42•15 years ago
|
||
Preliminary comments
* I can't find anything obviously wrong with the patch. it looks like the invariant throughout Assembler is that code pointers point to writeable memory, and every point we need the executable address, we use toExec(). cool... a final review will be needed to just make sure we dont miss a spot.
one way to mitigate that risk is to use a different pointer type for pointers to executable memory vs pointers to writeable memory.
* its also important that the machenery for double-mapping memory be encapsulated (all the code with pathnames in it is clearly machine specific. I think that's been done but the patch attached doesn't preserve filenames.
* the working version of toExec has lots of code in it, which worries me. it could slow down the assembler by a lot. I think the CodeList data structure should only track writeable memory (no twins) and have just one field, the offset to the executable memory. The Assembler tracks the current pages being used, so the fast path through Assembler::toExec should be: { return p + exec_offset } where exec_offset is computed once and only updated when we allocate new pages.
(its value also needs to be swapped in the swapCodeChunks() function when we swap codeStart/exitStart, etc).
is this feasible? do we ever have to do anything with CodeList structures when starting solely from an executable address (ie: i think all the calculations start with write addresses).
If the fast path is { p + offset } then in non-double-mapped builds, offset can be a constant 0 and we dont need as many ifdef's.
Updated•15 years ago
|
Attachment #406207 -
Flags: review?(edwsmith) → review-
Comment 43•15 years ago
|
||
Thanks for the comments!
> one way to mitigate that risk is to use a different pointer type for pointers
> to executable memory vs pointers to writeable memory.
Do you mean something like NIns* and NInsExec* ?
> * its also important that the machenery for double-mapping memory be
> encapsulated (all the code with pathnames in it is clearly machine specific. > I think that's been done but the patch attached doesn't preserve filenames.
I'm not sure what do you mean with "pathnames" here...
> * the working version of toExec has lots of code in it, which worries me. it
> could slow down the assembler by a lot. I think the CodeList data structure
> should only track writeable memory (no twins) and have just one field, the
> offset to the executable memory. The Assembler tracks the current pages being
> used, so the fast path through Assembler::toExec should be: { return p +
> exec_offset } where exec_offset is computed once and only updated when we
> allocate new pages.
> [....]
Okay, will try move the address translation to higher level.
Comment 44•15 years ago
|
||
Let's divide the patch to some smaller pieces. This one contains secure memory allocation for linux systems.
Attachment #417089 -
Flags: review?(gal)
Comment 45•15 years ago
|
||
There is a patch with requested changes here:
- offset to current executable pages has been moved to Assembler::_execOffset, Assembler::_exitExecOffset, they are allocated by Assembler::codeAlloc().
- Assembler::toExec() simplification.
- CodeList::toExec has been removed completely, page twins has been
replaced by execOffset.
- Assembler::swapCodeChunks() update (i386 only for now)
I haven't done the NIns->NInsExec change for now.
Attachment #406207 -
Attachment is obsolete: true
Attachment #417098 -
Flags: review?(edwsmith)
Attachment #406207 -
Flags: review?(rreitmai)
Attachment #406207 -
Flags: review?(gal)
Comment 46•15 years ago
|
||
(In reply to comment #43)
> Thanks for the comments!
>
> > one way to mitigate that risk is to use a different pointer type for pointers
> > to executable memory vs pointers to writeable memory.
>
> Do you mean something like NIns* and NInsExec* ?
yes
> > * its also important that the machenery for double-mapping memory be
> > encapsulated (all the code with pathnames in it is clearly machine specific. > I think that's been done but the patch attached doesn't preserve filenames.
>
> I'm not sure what do you mean with "pathnames" here...
I was referring to the code that creates the temporary file, unlinks it, then maps it twice. You've already factored it out of CodeAlloc, which addressed my concern.
> > * the working version of toExec has lots of code in it, which worries me. it
> > could slow down the assembler by a lot. I think the CodeList data structure
> > should only track writeable memory (no twins) and have just one field, the
> > offset to the executable memory. The Assembler tracks the current pages being
> > used, so the fast path through Assembler::toExec should be: { return p +
> > exec_offset } where exec_offset is computed once and only updated when we
> > allocate new pages.
> > [....]
>
> Okay, will try move the address translation to higher level.
Comment 47•15 years ago
|
||
Does freeCodeChunk() need to take the exec address as an argument as well? Seems like if allocCodeChunk returns two addresses then freeCodeChunk should free (munmap) them both at the same time.
In the patch for bug 460993, I'm moving protect calls out of CodeAlloc and into a host api; that way the host controls how/when page protections are set. If we extend the api to handle addr and exec in each call, would it clean things up?
my last concern is still that we could spend too much compile time traversing data structures in calcOffset(). Some benchmark results would alleviate the concern.
Comment 48•15 years ago
|
||
(In reply to comment #47)
> my last concern is still that we could spend too much compile time traversing
> data structures in calcOffset(). Some benchmark results would alleviate the
> concern.
Which benchmark/CPU combination would you like to see? I did some sunspider runs
(from http://www2.webkit.org/perf/sunspider-0.9/sunspider-driver.html) on Intel Core 2 box (2GHz) and there was no difference between patched/non patched version.
Comment 49•15 years ago
|
||
There is a sun spider sample here, run on Core2/trunk/debug build/optimalizations disabled.
Patched:
http://www2.webkit.org/perf/sunspider-0.9/sunspider-results.html?%7B%223d-cube%22:%5B516,513,514,516,509%5D,%223d-morph%22:%5B87,86,86,88,85%5D,%223d-raytrace%22:%5B349,349,352,354,348%5D,%22access-binary-trees%22:%5B156,157,157,157,155%5D,%22access-fannkuch%22:%5B272,255,256,256,263%5D,%22access-nbody%22:%5B130,138,131,130,134%5D,%22access-nsieve%22:%5B48,49,47,48,48%5D,%22bitops-3bit-bits-in-byte%22:%5B2,2,2,2,3%5D,%22bitops-bits-in-byte%22:%5B12,12,12,12,13%5D,%22bitops-bitwise-and%22:%5B4,4,4,4,4%5D,%22bitops-nsieve-bits%22:%5B79,82,78,79,79%5D,%22controlflow-recursive%22:%5B19,19,19,18,19%5D,%22crypto-aes%22:%5B219,217,215,219,264%5D,%22crypto-md5%22:%5B99,99,98,102,99%5D,%22crypto-sha1%22:%5B37,38,37,37,37%5D,%22date-format-tofte%22:%5B674,678,671,674,672%5D,%22date-format-xparb%22:%5B315,318,315,317,316%5D,%22math-cordic%22:%5B31,31,31,31,31%5D,%22math-partial-sums%22:%5B37,36,38,37,36%5D,%22math-spectral-norm%22:%5B25,25,25,25,25%5D,%22regexp-dna%22:%5B122,119,120,119,123%5D,%22string-base64%22:%5B106,106,106,105,107%5D,%22string-fasta%22:%5B476,479,473,473,497%5D,%22string-tagcloud%22:%5B514,512,515,518,517%5D,%22string-unpack-code%22:%5B650,642,661,645,648%5D,%22string-validate-input%22:%5B343,361,360,351,355%5D%7D
Not patched:
http://www2.webkit.org/perf/sunspider-0.9/sunspider-results.html?%7B%223d-cube%22:%5B511,511,516,505,517%5D,%223d-morph%22:%5B85,85,84,82,85%5D,%223d-raytrace%22:%5B350,345,345,345,345%5D,%22access-binary-trees%22:%5B160,158,157,158,161%5D,%22access-fannkuch%22:%5B250,251,264,264,250%5D,%22access-nbody%22:%5B127,125,126,122,129%5D,%22access-nsieve%22:%5B47,47,48,50,47%5D,%22bitops-3bit-bits-in-byte%22:%5B2,2,2,2,2%5D,%22bitops-bits-in-byte%22:%5B13,13,13,12,13%5D,%22bitops-bitwise-and%22:%5B3,3,3,3,3%5D,%22bitops-nsieve-bits%22:%5B79,80,79,79,78%5D,%22controlflow-recursive%22:%5B19,19,18,18,19%5D,%22crypto-aes%22:%5B214,214,217,214,212%5D,%22crypto-md5%22:%5B98,98,97,96,97%5D,%22crypto-sha1%22:%5B36,36,36,37,36%5D,%22date-format-tofte%22:%5B664,665,663,665,662%5D,%22date-format-xparb%22:%5B310,309,309,311,307%5D,%22math-cordic%22:%5B31,33,31,31,31%5D,%22math-partial-sums%22:%5B36,36,36,36,37%5D,%22math-spectral-norm%22:%5B25,25,25,25,25%5D,%22regexp-dna%22:%5B119,123,121,121,118%5D,%22string-base64%22:%5B106,105,105,108,108%5D,%22string-fasta%22:%5B499,500,498,497,497%5D,%22string-tagcloud%22:%5B506,517,515,511,512%5D,%22string-unpack-code%22:%5B651,634,637,636,633%5D,%22string-validate-input%22:%5B353,354,350,359,354%5D%7D
Total results:
--------------
Patched: 5335.2ms +/- 0.7%
Not patched: 5288.4ms +/- 0.2%
As for the calcOffset() and traversing of allocated pages - there are usually only 2-3 physical pages allocated so the page search is pretty fast. I can implement some page cache but it looks pointless for now.
Unfortunately I can't test a patch from bug 460993, the protect-impl.patch does not apply to mozilla tree.
Comment 50•15 years ago
|
||
I did a small optimalization to CodeAlloc::toExec()/CodeAlloc::calcOffset(),
+ NIns* CodeAlloc::toExec(NIns* p, CodeList* page) {
+ if(!p) {
+ return NULL;
+ }
+
+ if (!page)
+ page = getPage(p);
+
+ assert(page != NULL);
+
+ NIns *out = (NIns*)(uintptr_t(p) + page->execOffset);
+
+ if (verbose)
+ avmplus::AvmLog("toExec write %p -> exec %p page = %d\n", p, out, page);
+
+ return out;
+ }
+
+ uintptr_t CodeAlloc::calcOffset(NIns* target, NIns* source) {
+ CodeList* target_page = getPage(target);
+ CodeList* source_page = getPage(source);
+
+ uintptr_t offset;
+
+ if(target_page == source_page) {
+ // both adresses are from one continous memory block
+ offset = uintptr_t(target) - (uintptr_t(source) + sizeof(NIns));
+ }
+ else {
+ // different pages - run the translation machinery
+ uintptr_t t = uintptr_t(toExec(target, target_page));
+ uintptr_t s = uintptr_t(toExec(source, source_page));
+ assert(t != 0 || s != 0);
+
+ offset = t - (s + sizeof(NIns));
+ }
+
+ if (verbose)
+ avmplus::AvmLog("calcOffset target %p, source %p -> offset %p\n", target, source+1, offset);
+
+ return offset;
+ }
Comment 51•15 years ago
|
||
BTW. With this small optimalization in CodeAlloc::toExec()/CodeAlloc::calcOffset(), CodeAlloc::toExec() can take a page argument so if it's called from calcOffset(), getPage() is calculated only once.
Comment 52•15 years ago
|
||
Ping. Is there anything else you need?
Comment 53•15 years ago
|
||
Comment on attachment 417098 [details] [diff] [review]
write/exec code with requested changes
I don't see anything else glaringly wrong. In Tamarin we decided to go with a more heavy handed page protection scheme (flip between RX/RW with no dual mapping), but if TM wants this scheme to facilitate faster patching, it should be doable. again, the key is not impacting performance or maintainability when single-mapped code pages are wanted by the host vm.
Attachment #417098 -
Flags: review?(gal)
Attachment #417098 -
Flags: review?(edwsmith)
Attachment #417098 -
Flags: review+
Comment 54•15 years ago
|
||
(In reply to comment #53)
> (From update of attachment 417098 [details] [diff] [review])
> In Tamarin we decided to go with a
> more heavy handed page protection scheme (flip between RX/RW with no dual
> mapping), but if TM wants this scheme to facilitate faster patching, it should
> be doable.
Ehm, switching between RW and RX is almost equally bad and also isn't allowed by SELinux. If the program can do this then the bad guy can craft some code to do this as well (return-to-libc sequences to call into the mprotect wrappers in libc).
Once a page is marked writable it cannot be allowed to become executable again. That's the only safe policy.
Comment 55•15 years ago
|
||
Comment on attachment 417098 [details] [diff] [review]
write/exec code with requested changes
>+// TODO -> when write/exec pages are enabled, c->_address has to be in exec code segment
What's the follow-up for this?
I recommend renaming CodeAlloc::toExec to something else. There is already a toExec function that does a very different mapping (without walking all heap blocks).
I am still not a big fan of the approach, but Ulrich seems to make a good point about RW<>RX mapping changes using return-to-libc, which somewhat sinks my favorite approach (disable X while emitting code).
I wonder what the next steps are. Windows support? Should we try to settle on one mechanism together with Adobe?
Attachment #417098 -
Flags: review?(gal) → review+
Comment 56•15 years ago
|
||
(In reply to comment #55)
> (From update of attachment 417098 [details] [diff] [review])
>
> >+// TODO -> when write/exec pages are enabled, c->_address has to be in exec code segment
>
> What's the follow-up for this?
I'm going to put some assert here. From what I see the c->_address is always in external executable code space (libs and so) but we should catch if the design changes.
> I recommend renaming CodeAlloc::toExec to something else. There is already a
> toExec function that does a very different mapping (without walking all heap
> blocks).
Will do.
> I am still not a big fan of the approach, but Ulrich seems to make a good point
> about RW<>RX mapping changes using return-to-libc, which somewhat sinks my
> favorite approach (disable X while emitting code).
>
> I wonder what the next steps are. Windows support? Should we try to settle on
> one mechanism together with Adobe?
I've been doing a merge with latest trunk but it seems to contain the RW<>RX mapping change already. So what is the proposal here? Shall I integrate them together? Or do you suppose to remove it?
Comment 57•15 years ago
|
||
An updated patch with:
- new NInsExec on some places
- freeCodeChunk() has two arguments now (write&exec)
- CodeAlloc::toExec()/CodeAlloc::calcOffset() optimalization
- c->_address is checked by isExec()
- CodeAlloc::toExec() changed to CodeAlloc::getExec()
I hope it address all the remarks. It supports i386 for now and I'd like to have some decision about the general approach and then I can add support for other arches (x86_64/PPC).
Attachment #417098 -
Attachment is obsolete: true
Attachment #426675 -
Flags: review?(gal)
Attachment #426675 -
Flags: review?(edwsmith)
Comment 58•15 years ago
|
||
> I've been doing a merge with latest trunk but it seems to contain the RW<>RX
> mapping change already. So what is the proposal here? Shall I integrate them
> together? Or do you suppose to remove it?
The intent with the RX<->RW implementation was to make it easy for it to do nothing, based on VM implementations of allocCodeChunk, etc. For example in tamarin, it's controlled by an ifdef and in Tracemonkey the protection calls are no-ops. (memory is mmapped as RWX and later munmapped() with no intermediate permission changes).
One general question about dual-mapping: Once we're done writing to the RW region, and assuming no patching, we could unmap it, right? any experience with this, or recommendations?
Comment 59•15 years ago
|
||
(In reply to comment #58)
> One general question about dual-mapping: Once we're done writing to the RW
> region, and assuming no patching, we could unmap it, right? any experience
> with this, or recommendations?
If you never intend to write to the region then the region can indeed be unmap. But is this the case? It is terribly expensive to unmap everything just to create a new pair of mappings for later. But if this is how it is normally done then by all means, unmap the writable mapping.
Comment 60•15 years ago
|
||
Patch with markBlockWrite() routines, it's compatible with the latest trunk.
Attachment #426675 -
Attachment is obsolete: true
Attachment #426949 -
Flags: review?(gal)
Attachment #426949 -
Flags: review?(edwsmith)
Attachment #426675 -
Flags: review?(gal)
Attachment #426675 -
Flags: review?(edwsmith)
Comment 61•15 years ago
|
||
This patch adds x86_64 support. It's without any extra hack in the 64-bit part so sometimes the offset is calculated twice.
Plus this patch has a safe version of calcOffset routine - you can pass any kind of pointer here (from exec/write/external space) and the pointer is translated only if relevant page is found. It's useful for the CALL() instructions and so.
Attachment #426949 -
Attachment is obsolete: true
Attachment #426949 -
Flags: review?(gal)
Attachment #426949 -
Flags: review?(edwsmith)
Updated•15 years ago
|
Attachment #427321 -
Flags: review?(gal)
Attachment #427321 -
Flags: review?(edwsmith)
Comment 62•15 years ago
|
||
(In reply to comment #55)
> I wonder what the next steps are. Windows support? Should we try to settle on
> one mechanism together with Adobe?
So what is the recent status here? Is there any agreement between Adobe and mozilla about an united approach?
Updated•15 years ago
|
Whiteboard: [sg:investigate]
Comment 63•15 years ago
|
||
Ping?
Updated•15 years ago
|
Attachment #427321 -
Flags: review?(gal) → review?(nnethercote)
Updated•15 years ago
|
Attachment #417089 -
Flags: review?(gal) → review?(nnethercote)
Comment 64•15 years ago
|
||
Comment on attachment 417089 [details] [diff] [review]
separated patch for secure memory allocation
Big nit: the whole file uses two space indents. NJ uses four spaces.
>+ if (statfs ("/selinux", &sfs) >= 0 && (unsigned int)sfs.f_type == 0xf97cff8cU)
>+ return true;
What is 0xf97cff8cU? Is there a name for that constant?
>+ f = fopen ("/proc/mounts", "r");
>+ if (f == NULL)
>+ return false;
>+ while (getline(&buf, &len, f) >= 0) {
>+ char *p = strchr(buf, ' ');
>+ if (p == NULL)
>+ break;
>+ p = strchr(p + 1, ' ');
>+ if (p == NULL)
>+ break;
>+ if (strncmp(p + 1, "selinuxfs ", 10) != 0) {
>+ free(buf);
>+ fclose(f);
>+ return true;
>+ }
A brief comment explaining the format of /proc/mounts would help here.
>+ writable and exexutable filesystem. */
Nit: s/exexutable/executable/
>+ assert(!execFileReady());
We use NanoAssert() in Nanojit. And nanojit/VMPI.h has local versions of various other functions. We might need to use them, although if this is Linux-only code I'm not sure if it's necessary. Ed?
>+ /* Map the file for writting */
Nit: s/writting/writing/
>+ /* Open a map file */
>+ if (!execFileReady()) {
>+ pthread_mutex_lock(&fileMutex);
>+ bool ret = openExecFile();
>+ pthread_mutex_unlock(&fileMutex);
>+ if(!ret) {
>+ return false;
>+ }
>+ }
>+
>+ /* Map memory */
>+ return mapExec(length, write, exec);
>+ }
mapExec() doesn't look thread-safe to me. Why isn't there a lock protecting it?
>+ /** Our curent position inside loader table */
Nit: s/curent/current/
Comment 65•15 years ago
|
||
(In reply to comment #64)
> We use NanoAssert() in Nanojit. And nanojit/VMPI.h has local versions of
> various other functions. We might need to use them, although if this is
> Linux-only code I'm not sure if it's necessary. Ed?
not strictly necessary... In platform specific code it's okay to use api's that are known to be what you want on that platform. but unrelated calls should use portable api's when possible. (e.g. use NanoAssert anyway).
> >+ /* Map the file for writting */
>
> Nit: s/writting/writing/
>
>
> >+ /* Open a map file */
> >+ if (!execFileReady()) {
> >+ pthread_mutex_lock(&fileMutex);
> >+ bool ret = openExecFile();
> >+ pthread_mutex_unlock(&fileMutex);
> >+ if(!ret) {
> >+ return false;
> >+ }
> >+ }
> >+
> >+ /* Map memory */
> >+ return mapExec(length, write, exec);
> >+ }
>
> mapExec() doesn't look thread-safe to me. Why isn't there a lock protecting
> it?
>
>
> >+ /** Our curent position inside loader table */
>
> Nit: s/curent/current/
Comment 66•15 years ago
|
||
Shaver and I discussed this and we would like to try this across all platforms (no need to block the patch on that), so lets land this and file bugs on windows and mac support.
I will hold off with the alternative approach in bug 555033.
Updated•15 years ago
|
Comment 67•15 years ago
|
||
(In reply to comment #66)
> Shaver and I discussed this and we would like to try this across all platforms
In that case all the "is this SELinux?" stuff can be removed.
Comment 68•15 years ago
|
||
Comment on attachment 427321 [details] [diff] [review]
exec patch v.4 - i386 & x86_64
In general, AFAICT the patch works as intended, ie. implements double
mapping. As for whether this is what we want, I'll leave that up to others
to decide.
Martin, can you update the patch? It's bit-rotted a bit and I'd like to run
some tests and timings.
Detailed comments below.
----
First of all, 8 lines of context would help with this patch -- put
"diff=-p -U 8 " in the "[defaults]" section of your .hgrc file.
We now have a MIPS backend that will need to be modified.
>+ uintptr_t _execOffset; // offset to executable code page for current normal code chunk
>+ uintptr_t _exitExecOffset; // offset to executable code page for current exit code chunk
There is no big comment explaining the whole separate-write-and-exec-pages
idea. I'd like to see one, IMO it's quite non-obvious. This would be a
good place for it. A reference to this bug would be good.
>+ // write to exec page translations
>+ NInsExec* toExec(NIns *p) { return (NInsExec*)(uintptr_t(p) + _execOffset); }
>+
>+#ifdef AVMPLUS_UNIX
>+ uintptr_t calcOffset(NIns *target, NIns *source) { return _codeAlloc.calcOffset(target, source); }
>+#else
>+ uintptr_t calcOffset(NIns *target, NIns *source) { return uintptr_t(target) - uintptr_t(source); }
>+#endif
Nit: standard whitespace usage, please.
Non-nit: I'd like CodeAlloc::calcOffset() to exist in the non-AVMPLUS_UNIX
code, ie. have the #ifdef inside CodeAlloc::calcOffset() rather than here.
>+ NInsExec* CodeAlloc::getExec(NIns* p, CodeList* page) {
>+ if (!p || !page) {
>+ return p;
>+ }
This should instead assert (!p && !page).
>+ NInsExec* CodeAlloc::getExec(NIns* p) {
>+ return(getExec(p,getPage(p)));
>+ }
This function appears to be dead, please remove it.
>+ uintptr_t CodeAlloc::calcOffset(NIns* target, NIns* source) {
>+
>+ // We don't expect that
>+ assert(source != NULL);
Is that comment truncated?
>+
>+ // Source address can lay on unmapped/forein page
Nit: s/forein/foreign/
>+ // (for instance exit code with JMP at the end)
>+ // Because offsets are calculated for jumps,
>+ // calculate it relatively to the jmp instructions (source-1)
>+ // and then shift it back.
>+ source--;
>+
>+ CodeList* target_page = getPage(target);
>+ CodeList* source_page = getPage(source);
>+
>+ uintptr_t offset;
>+
>+ if(target_page == source_page) {
>+ // both adresses are from one continous memory block
Nit: s/adresses/addresses/
>+ offset = uintptr_t(target) - (uintptr_t(source) + sizeof(NIns));
>+ }
>+ else {
>+ // different pages - run the translation machinery
>+ uintptr_t t = uintptr_t(getExec(target, target_page));
>+ uintptr_t s = uintptr_t(getExec(source, source_page));
>+ assert(t != 0 || s != 0);
>+
>+ offset = t - (s + sizeof(NIns));
>+ }
>+
>+ if (verbose)
>+ avmplus::AvmLog("calcOffset target %p, source %p -> offset %p\n", target, source+1, offset);
>+
>+ return offset;
>+ }
>+
>+#endif // AVMPLUS_UNIX
That looks like it could be slow. What used to be just a pointer difference
now involves two calls to getPage(), each of which scans the CodeList.
Maybe the two calls could be combined into one that looks for both pages in
tandem? Or maybe it's not that important.
Comment 69•15 years ago
|
||
Fixed indentation, typos, mutex, removed selinux check.
Attachment #417089 -
Attachment is obsolete: true
Attachment #435577 -
Flags: review?(nnethercote)
Attachment #417089 -
Flags: review?(nnethercote)
Comment 70•15 years ago
|
||
Thanks for the review, there's an updated one, with fixed typos/comments and getPages routine which translates two pointers on one loop.
Attachment #427321 -
Attachment is obsolete: true
Attachment #435614 -
Flags: review?(nnethercote)
Attachment #427321 -
Flags: review?(nnethercote)
Attachment #427321 -
Flags: review?(edwsmith)
Updated•15 years ago
|
Attachment #399047 -
Attachment is obsolete: true
Comment 71•15 years ago
|
||
> >+ NInsExec* CodeAlloc::getExec(NIns* p, CodeList* page) {
> >+ if (!p || !page) {
> >+ return p;
> >+ }
>
> This should instead assert (!p && !page).
No, we can't assert here because it's a correct situation. p is NULL if target in calcOffset is NULL and NULL page means that p comes from outside address space (libraries and so) and we're not going to translate it.
Comment 72•15 years ago
|
||
Comment on attachment 435577 [details] [diff] [review]
linux allocation v2
>+ /* Map in a writable and executable chunk of memory if possible.
>+ Failing that, fall back to mapExec. */
>+ bool MemMap::map(size_t length, void *&write, void *&exec)
>+ {
>+ write = NULL;
>+ exec = NULL;
>+
>+ pthread_mutex_lock(&fileMutex);
>+
>+ /* Open a map file */
>+ if (!execFileReady()) {
>+ if (!openExecFile()) {
>+ return false;
>+ }
>+ }
>+ /* Map memory */
>+ bool ret = mapExec(length, write, exec);
>+
>+ pthread_mutex_unlock(&fileMutex);
>+
>+ return ret;
>+ }
You haven't unlocked the mutex on the 'return false' exit path.
>+ /* Return a file descriptor of a temporary zero-sized file in a
>+ writable and exexutable filesystem. */
Nit: s/executable/exexutable/
r=me with those fixed.
Attachment #435577 -
Flags: review?(nnethercote) → review+
Comment 73•15 years ago
|
||
Hmm, on memmap.cpp I get this warning from GCC:
../nanojit/memmap.cpp: In constructor ‘nanojit::MemMap::MemMap()’:
../nanojit/memmap.cpp:319: warning: extended initializer lists only available with -std=c++0x or -std=gnu++0x
../nanojit/memmap.cpp:319: warning: extended initializer lists only available with -std=c++0x or -std=gnu++0x
The relevant line is this:
fileMutex = PTHREAD_MUTEX_INITIALIZER;
I don't know how to fix that.
Also:
../nanojit/memmap.cpp: In member function ‘bool nanojit::MemMap::openExecFile()’:
../nanojit/memmap.cpp:247: warning: ignoring return value of ‘int ftruncate(int, __off_t)’, declared with attribute warn_unused_result
../nanojit/memmap.cpp: In member function ‘bool nanojit::MemMap::mapExec(size_t, void*&, void*&)’:
../nanojit/memmap.cpp:274: warning: ignoring return value of ‘int ftruncate(int, __off_t)’, declared with attribute warn_unused_result
../nanojit/memmap.cpp:284: warning: ignoring return value of ‘int ftruncate(int, __off_t)’, declared with attribute warn_unused_result
That can be fixed with:
int dummy = ftruncate(...)
(void)dummy;
The second line is needed to avoid a warning about 'dummy' being unused.
Comment 74•15 years ago
|
||
Also, it doesn't compile on Mac:
In file included from ../nanojit/avmplus.cpp:50:
../nanojit/memmap.h:65: error: ‘pthread_mutex_t’ does not name a type
I tried adding pthread.h to memmap.h but that results in this:
Undefined symbols:
"nanojit::MemMap::MemMap()", referenced from:
__static_initialization_and_destruction_0(int, int)in avmplus.o
"nanojit::MemMap::~MemMap()", referenced from:
___tcf_0 in avmplus.o
"nanojit::MemMap::map(unsigned long, void*&, void*&)", referenced from:
nanojit::CodeAlloc::allocCodeChunk(unsigned long, void*&)in avmplus.o
ld: symbol(s) not found
And I haven't tried Windows.
Comment 75•15 years ago
|
||
Comment on attachment 435614 [details] [diff] [review]
exec patch v5
Still no NativeMIPS.cpp changes, it's not crucial but would be nice to have before landing (even if untested).
>+ NInsExec* CodeAlloc::getExec(NIns* p, CodeList* page) {
>+ // There are who cases when we don't translate given pointer:
>+ //
>+ // * p == NULL - calcOffset() has been called with target = NULL.
>+ //
>+ // * page == NULL - calcOffset() has been called with target or source
>+ // from outside of our dual-mapped address space.
>+ if (!p || !page) {
>+ return p;
>+ }
>+
> [...]
>+
>+ // different pages - run the translation machinery
>+ uintptr_t t = uintptr_t(getExec(target, targetPage));
>+ uintptr_t s = uintptr_t(getExec(source, sourcePage));
>+ assert(t != 0 || s != 0);
We never call getExec() with p==NULL; if we did this assertion would fail.
So I'd like the !p assertion within getExec() where it's more obvious.
I ran the TM trace-tests on i386/Linux and X64/Linux and they all passed.
Comment 76•15 years ago
|
||
I ran timings on i386/Linux on SS and V8 and saw no noticeable change.
Comment 77•15 years ago
|
||
(In reply to comment #73)
> ../nanojit/memmap.cpp: In constructor ‘nanojit::MemMap::MemMap()’:
> ../nanojit/memmap.cpp:319: warning: extended initializer lists only available
> with -std=c++0x or -std=gnu++0x
> ../nanojit/memmap.cpp:319: warning: extended initializer lists only available
> with -std=c++0x or -std=gnu++0x
>
> The relevant line is this:
>
> fileMutex = PTHREAD_MUTEX_INITIALIZER;
Use
pthread_mutex_init(&fileMutex, NULL);
PTHREAD_MUTEX_INITIALIZER and the other initializer macros are only meant to
initialize global variables statically.
Comment 78•15 years ago
|
||
Fixed return from mutex, pthread initialization, ftruncate return values (I hope). Which compile arguments do you use? -std=c++0x or -std=gnu++0x don't throw me the warning messages about ftruncate (gcc version 4.4.3 20100127/Fedora 12). As for Mac, I don't have such platform available but I should be able to find some box with Windows.
Attachment #435577 -
Attachment is obsolete: true
Attachment #435844 -
Flags: review?(nnethercote)
Comment 79•15 years ago
|
||
(In reply to comment #75)
> >+ NInsExec* CodeAlloc::getExec(NIns* p, CodeList* page) {
> >+ // There are who cases when we don't translate given pointer:
> >+ //
> >+ // * p == NULL - calcOffset() has been called with target = NULL.
> >+ //
> >+ // * page == NULL - calcOffset() has been called with target or source
> >+ // from outside of our dual-mapped address space.
> >+ if (!p || !page) {
> >+ return p;
> >+ }
> >+
> > [...]
> >+
> >+ // different pages - run the translation machinery
> >+ uintptr_t t = uintptr_t(getExec(target, targetPage));
> >+ uintptr_t s = uintptr_t(getExec(source, sourcePage));
> >+ assert(t != 0 || s != 0);
>
> We never call getExec() with p==NULL; if we did this assertion would fail.
> So I'd like the !p assertion within getExec() where it's more obvious.
Yes, we do. Here is the backtrace if I put NanoAssert(p); to CodeAlloc::getExec():
#0 NanoAssertFail () at ./nanojit/avmplus.cpp:75
#1 0x081b7f6b in nanojit::CodeAlloc::getExec (this=0x826cb48, p=0x0, page=0x0) at ./nanojit/CodeAlloc.cpp:167
#2 0x081b8045 in nanojit::CodeAlloc::calcOffset (this=0x826cb48, target=0x0, source=0xb72ccfe2 "")
at ./nanojit/CodeAlloc.cpp:207
#3 0x081b6bcc in nanojit::Assembler::calcOffset(unsigned char*, unsigned char*) ()
#4 0x081b314b in nanojit::Assembler::gen (this=0x827dc04, reader=0x82d4f0c) at ./nanojit/Assembler.cpp:1548
#5 0x081b225e in nanojit::Assembler::assemble (this=0x827dc04, frag=0x8281eac, reader=0x82d4f0c)
at ./nanojit/Assembler.cpp:1039
#6 0x081b1b66 in nanojit::Assembler::compile (this=0x827dc04, frag=0x8281eac, alloc=..., optimize=true, labels=0x8375bdc)
at ./nanojit/Assembler.cpp:921
#7 0x081766c4 in js::TraceRecorder::compile (this=0x8288530) at jstracer.cpp:4255
It's because:
./nanojit/Assembler.cpp:1548, i386
1538 else {
1539 // backwards jump
1540 handleLoopCarriedExprs(pending_lives);
1541 if (!label) {
1542 // save empty register state at loop header
1543 _labels.add(to, 0, _allocator);
1544 }
1545 else {
1546 intersectRegisterState(label->regs);
1547 }
(gdb)
1548 JMP(0); <<<< here we come
1549 _patches.put(_nIns, to);
1550 }
1551 break;
Comment 80•15 years ago
|
||
With an attempt of basic MIPS support, untested.
Attachment #435614 -
Attachment is obsolete: true
Attachment #435867 -
Flags: review?(nnethercote)
Attachment #435614 -
Flags: review?(nnethercote)
Comment 81•15 years ago
|
||
Can you update to the latest TraceMonkey tip? Also, you've updated configure.in but you need to update js/src/configure.in. Thanks.
Once I have a cleanly applying patch I'll try to work out the problem on Mac.
Comment 82•15 years ago
|
||
What do you mean with "TraceMonkey tip"? This patch cleanly apply to latest trunk.
Comment 83•15 years ago
|
||
#82: SpiderMonkey development usually happens against the hg.mozilla.org/tracemonkey tree and is then merged into mozilla-central every few days.
Comment 84•15 years ago
|
||
Attachment #436161 -
Flags: review?(nnethercote)
Updated•15 years ago
|
Attachment #435844 -
Attachment is obsolete: true
Attachment #435844 -
Flags: review?(nnethercote)
Comment 85•15 years ago
|
||
Attachment #435867 -
Attachment is obsolete: true
Attachment #436162 -
Flags: review?(nnethercote)
Attachment #435867 -
Flags: review?(nnethercote)
Comment 86•15 years ago
|
||
I worked out the Mac build problem. In the configure.in file you were only compiling memmap.cpp if it was a Linux machine, but in the C++ code the use of memmap.cpp was decided by AVMPLUS_UNIX, which is true on both Linux and Mac. Rather than conditionally including memmap.cpp I changed things to always include memmap.cpp, but it's entire contents are guarded by "#ifdef AVMPLUS_UNIX".
I did some measurements on Mac, there was a noticeable slowdown, something like 2--3%. It's not yet clear to me how much of that is due to extra syscalls and how much is due to the extra work looking for pages and offsets.
And that got me thinking. I can see two ways to clearly improve the patch, and I've attached an in-progress patch that does these things:
- Properly distinguish writable instructions from executable instructions. We have the types NIns and NInsExec but they are equivalent so there's no meaningful checking. My attached patch renames NInsExec as NInsX and changes it to a struct containing a single field of the same size. This makes NIns and NInsX equivalent in terms of layout, but different types. (It also fixes some problems with the debug output, where the address shown were a mix of writable and executable instructions.)
- Record NInsX values instead of NIns value, where appropriate, rather than reconstituting them from the NIns values later on. This allows all the calcOffset() and getPages() stuff to be completely avoided.
The patch compiles on i386 but I'm currently getting a segfault, I screwed something up. I really think this is the right way to go, and I'm happy to take it from here, Martin, but I won't have a chance to work on it again until the Tuesday after Easter.
Assignee: stransky → nnethercote
Status: NEW → ASSIGNED
Comment 87•15 years ago
|
||
I've found only one bug in the patch so far, missing "execOffset = b->execOffset;" in CodeAlloc::alloc() when availblocks == false. But it still crashes right after start, the exec address seems to be broken.
Comment 88•15 years ago
|
||
This patch is similar in spirit to my previous one but it actually works.
Changes:
- Only working for i386 and X64 so far. Getting i386 working (along with
the arch-neutral changes) was difficult but getting X64 working after that
was very easy -- it worked first got after I fixed all the compile errors.
So hopefully the other backends won't be too hard.
- NIns is split into NInsW and NInsX. NInsW is the same as the old NIns,
just a uint8_t on i386. NInsX is structurally equivalent, but uses a
union, which means that you cannot mix NInsW and NInsX. This is a very
good thing. Lots of the new NInsW/NInsX variables now have 'W' or 'X'
suffixes which clarifies what they are. Some places need one or the
other, some need both.
- I split Assembler::_patches into two, one part for normal branches, and
one for jtbl branches. Much clearer. BTW, LIR_jtbl doesn't appear to be
generated by TR or TM. Why do we have it? It's probably broken (even
before my changes).
- I overhauled memmap.{h,cpp}:
- There are now fewer methods and fewer data members
- The control flow is greatly simplified, esp. for mount points -- no need
for the 'repeat' field in the LoaderTable
- For the first mapping, it used to do a test mapping of one page, and if
that succeeded, threw that away and then did a second mapping of the
right size. Now the test mapping is of the right size and if it
succeeds its used directly. This avoids one mapping.
Martin, I'd appreciate it if you could look over these changes and see
that I preserved the behaviour.
- There are some tiny TM-only changes, I didn't bother separating them.
Some questions:
- In flushICache() for 64-bit PPC Mac, should the sys_dcache_flush() call
use 'startW' instead of 'startX'?
- I'm seeing very slight changes to the code generated by TM for SunSpider
-- in two or three of them some chunk-linking jumps are occurring
slightly earlier than before. This is because CodeList has an extra
field (execOffset), so the space available for code in each chunk is
reduced by one word.
- allocCodeChunk() is supposed to always succeed, but several (all?) of
the implementations of it can fail! This situation predates my patch.
- I think memmap/MemMap is a pretty bad name for the new file and class.
I'd prefer something like "MapWX", indicating the dual-mapping.
I've tested it on {i386,X64}/{Linux,Mac}. Further testing will involve
porting it to Tamarin and testing with the Mozilla try servers. I'm seeing
maybe a 1% slowdown on SunSpider, and a negligible slowdown on V8.
Attachment #436430 -
Attachment is obsolete: true
Attachment #437801 -
Flags: feedback?(edwsmith)
Comment 89•15 years ago
|
||
Comment on attachment 437801 [details] [diff] [review]
a different approach, v2, works
BTW, I've got a little more polishing to do, but I thought I'd put it up for feedback now.
Attachment #437801 -
Flags: feedback?(stransky)
Comment 90•15 years ago
|
||
The memmap change looks fine, I wonder why I messed with the one-page test mapping. But IMO if mapExec() fails to map executable segment on a given file, you should close the file and try to find another one...not just quit. Some file systems can be mounted as r/w only. So something like:
int i = 0;
for(i = 0; i < (sizeof(loaders) / sizeof(loaders[0])); i++) {
NanoAssert(execFile == FILE_ERROR);
execFile = (this->*loaders[i].loader)(loaders[i].arg);
if (execFile != FILE_ERROR) {
if (mapExec(length, tmpW, tmpX)) {
memW = tmpW;
memX = tmpX;
return true;
} else {
close(execFile);
// don't pretend we have a valid execFile
execFile = FILE_ERROR;
}
}
}
Comment 91•15 years ago
|
||
(In reply to comment #90)
> But IMO if mapExec() fails to map executable segment on a given file,
> you should close the file and try to find another one...not just quit. Some
> file systems can be mounted as r/w only. So something like:
Indeed. A common, good recommendation is to mount /tmp with noexec (in Linux-speak). If /tmp would be the only place to look for such a mapping to exist it wouldn't work. The code should look in other places such as the user's home dir etc. The list should be configurable.
Comment 92•15 years ago
|
||
Does Linux (glibc, mayhap) not have a way to set up this sort of split mapping that can follow whatever the system's preferences are? Given that it is described as such a critical security improvement, it seems strange to me that there's no support beyond "try a lot of filesystem places and do your own mmap trickery". Is that really what Linux ISVs are all supposed to do if they generate code (increasingly common for scripting languages, which are themselves increasingly common)? Surely one of the JVMs that is shipped has already solved this problem as well?
njn: how about TwinMapping as the class name, or TwinMappingWX? MapWX I would expect to be w+x, and therefore AN ABOMINATION.
Comment 93•15 years ago
|
||
www.semantiscope.com/research/BHDC2010/BHDC-2010-Paper.pdf
I'm skeptical this patch matters in a world where "Leon" has a "typewriter" (see Dion's slides).
/be
Comment 94•15 years ago
|
||
> - I split Assembler::_patches into two, one part for normal branches, and
> one for jtbl branches. Much clearer. BTW, LIR_jtbl doesn't appear to be
> generated by TR or TM. Why do we have it? It's probably broken (even
> before my changes).
TR generates LIR_jtbl when compiling the switch construct, and it works, but i just
noticed a stack overflow bug somewhere near printer->formatIns().
I'll investigate and post a fix.
Comment 95•15 years ago
|
||
(In reply to comment #93)
> www.semantiscope.com/research/BHDC2010/BHDC-2010-Paper.pdf
>
> I'm skeptical this patch matters in a world where "Leon" has a "typewriter"
> (see Dion's slides).
at least, nanojit does constant folding of xor. (the paper references flash's old jit, which nanojit replaces). still, thats not necessarily enough to weaken jit spraying of PIC code.
Comment 96•15 years ago
|
||
(In reply to comment #90)
> The memmap change looks fine, I wonder why I messed with the one-page test
> mapping. But IMO if mapExec() fails to map executable segment on a given file,
> you should close the file and try to find another one...not just quit. Some
> file systems can be mounted as r/w only. So something like:
>
> int i = 0;
> for(i = 0; i < (sizeof(loaders) / sizeof(loaders[0])); i++) {
> NanoAssert(execFile == FILE_ERROR);
>
> execFile = (this->*loaders[i].loader)(loaders[i].arg);
> if (execFile != FILE_ERROR) {
> if (mapExec(length, tmpW, tmpX)) {
> memW = tmpW;
> memX = tmpX;
> return true;
> } else {
> close(execFile);
> // don't pretend we have a valid execFile
> execFile = FILE_ERROR;
> }
> }
> }
That code is better and I'll use it, but I don't think it addresses your main point. IIUC you're talking about the mount case... in my new code if we can successfully open a file on one of the mounts but then fail to map from it, it won't try another mount. I assume your old code did but the control flow is too complicated for me to tell for sure.
Making this work again will require some fiddling.
Comment 97•15 years ago
|
||
(In reply to comment #91)
> (In reply to comment #90)
> > But IMO if mapExec() fails to map executable segment on a given file,
> > you should close the file and try to find another one...not just quit. Some
> > file systems can be mounted as r/w only. So something like:
>
> Indeed. A common, good recommendation is to mount /tmp with noexec (in
> Linux-speak). If /tmp would be the only place to look for such a mapping to
> exist it wouldn't work. The code should look in other places such as the
> user's home dir etc. The list should be configurable.
Yes. If you read the patch you'll see this:
LoaderTable MemMap::loaders[] = {
{(FileLoader) (&MemMap::openFileEnv), "TMPDIR"},
{(FileLoader) (&MemMap::openFileDir), "/tmp"},
{(FileLoader) (&MemMap::openFileDir), "/var/tmp"},
{(FileLoader) (&MemMap::openFileDir), "/dev/shm"},
{(FileLoader) (&MemMap::openFileEnv), "HOME"},
#ifdef HAVE_MNTENT_H
{(FileLoader) (&MemMap::openFileMnt), "/etc/mtab"},
{(FileLoader) (&MemMap::openFileMnt), "/proc/mounts"},
#endif // HAVE_MNTENT_H
};
which should give you a good idea of what's being tried.
(In reply to comment #92)
> Does Linux (glibc, mayhap) not have a way to set up this sort of split mapping
> that can follow whatever the system's preferences are? Given that it is
> described as such a critical security improvement, it seems strange to me that
> there's no support beyond "try a lot of filesystem places and do your own mmap
> trickery". Is that really what Linux ISVs are all supposed to do if they
> generate code (increasingly common for scripting languages, which are
> themselves increasingly common)?
It's a good point. The example at http://people.redhat.com/drepper/selinux-mem.html just tries the home directory, then adds this not-very-useful comment:
"The only requirement for this code to work is that the filesystem on which the file is created must allow execution. Some sites might (wisely) decide to mount filesystems like /tmp and /var/tmp with the noexec option. One can either detect this ... or actively look for directory with appropriate permissions."
Having said that, if we want to do this on Mac and Windows we can't rely on any glibc magic anyway.
I wonder how useful the mount point attempts are -- how often will the first five attempts fail? It makes me nervous because the mount code will be used very rarely, which increases it's likelihood of being broken.
Another question: is it ok to use pthread_mutex_lock() et al inside Mozilla code? Or do we have to indirect it to some NSPR thing?
Comment 98•15 years ago
|
||
Martin, seems like changing this:
if (!(hasmntopt(&mnt, "ro") ||
hasmntopt(&mnt, "noexec") || access(mnt.mnt_dir, W_OK)))
to this:
if (!(hasmntopt(&mnt, "ro") ||
hasmntopt(&mnt, "noexec") || access(mnt.mnt_dir, R_OK|W_OK|X_OK)))
might be better? Or maybe the the R_OK|X_OK check is subsumed by the hasmntopt() checks?
Comment 99•15 years ago
|
||
(In reply to comment #95)
> (In reply to comment #93)
> > www.semantiscope.com/research/BHDC2010/BHDC-2010-Paper.pdf
> >
> > I'm skeptical this patch matters in a world where "Leon" has a "typewriter"
> > (see Dion's slides).
>
> at least, nanojit does constant folding of xor. (the paper references flash's
> old jit, which nanojit replaces). still, thats not necessarily enough to
> weaken jit spraying of PIC code.
Before we take this patch, I want us to be clear on the security benefits we're going to get. Please, no sermonizing on "privileges" and "rights".
What attacks does this patch prevent, and which ones does it leave open? Let's make the trade-offs extremely clear, so we have a record of our decision, if nothing else.
Comment 100•15 years ago
|
||
This version fixes the problems with giving up too early in the mount cases.
Todo list:
- Test it in the browser (I've only done shell testing so far)
- Test it on Windows (I've only done Linux and Mac so far)
- Decide if pthreads can be used as-is
- Decide if __builtin_alloca can be used
- Port it to Tamarin, check it works there
Attachment #436161 -
Attachment is obsolete: true
Attachment #436162 -
Attachment is obsolete: true
Attachment #437801 -
Attachment is obsolete: true
Attachment #438016 -
Flags: feedback?(stransky)
Attachment #437801 -
Flags: feedback?(stransky)
Attachment #437801 -
Flags: feedback?(edwsmith)
Attachment #436161 -
Flags: review?(nnethercote)
Attachment #436162 -
Flags: review?(nnethercote)
Comment 101•15 years ago
|
||
(In reply to comment #98)
> Martin, seems like changing this:
>
> if (!(hasmntopt(&mnt, "ro") ||
> hasmntopt(&mnt, "noexec") || access(mnt.mnt_dir, W_OK)))
>
> to this:
>
> if (!(hasmntopt(&mnt, "ro") ||
> hasmntopt(&mnt, "noexec") || access(mnt.mnt_dir,
> R_OK|W_OK|X_OK)))
>
> might be better? Or maybe the the R_OK|X_OK check is subsumed by the
> hasmntopt() checks?
man access(2) says:
"access() may not work correctly on NFS file systems with UID mapping enabled, because UID mapping is done on the server and hidden from the client, which checks permissions."
so I think the idea is to check general accessibility by access() and the execute flags by hasmntopt().
Comment 102•15 years ago
|
||
(In reply to comment #100)
> - Decide if __builtin_alloca can be used
MemMap::openDirFileAndMapExec()/__builtin_alloca() is called only when a mapfile is created and the file creation takes some time so IMO malloc() would serve as well here.
Comment 103•15 years ago
|
||
I just saw this on the Linux mmap man page:
The use of MAP_ANONYMOUS in conjunction with MAP_SHARED
is only supported on Linux since kernel 2.4.
If we are able to rely on MAP_ANONYMOUS|MAP_SHARED working (on all AVMPLUS_UNIX platforms, which includes Linux and Mac) then we could greatly simplify memmap.cpp -- no need to look through $TMPDIR, /tmp, /var/tmp, etc, to find a file that can be marked executable. Also, it might be faster because we wouldn't need the ftruncate() syscalls.
Comment 104•15 years ago
|
||
(In reply to comment #103)
>
> If we are able to rely on MAP_ANONYMOUS|MAP_SHARED working (on all AVMPLUS_UNIX
> platforms, which includes Linux and Mac) then we could greatly simplify
> memmap.cpp -- no need to look through $TMPDIR, /tmp, /var/tmp, etc, to find a
> file that can be marked executable. Also, it might be faster because we
> wouldn't need the ftruncate() syscalls.
Is this capability available on SELinux?
Comment 105•15 years ago
|
||
(In reply to comment #104)
>
> Is this capability available on SELinux?
Looking more closely, I think not. The full excerpt from the man page:
MAP_ANONYMOUS
The mapping is not backed by any file; its contents are initialized to
zero. The fd and offset arguments are ignored; however, some
implementations require fd to be -1 if MAP_ANONYMOUS (or MAP_ANON) is
specified, and portable applications should ensure this. The use of
MAP_ANONYMOUS in conjunction with MAP_SHARED is only sup‐ ported on
Linux since kernel 2.4.
Also, it doesn't make much sense logically -- without the underlying file,
how do you tie two mmapped regions together?
Comment 106•15 years ago
|
||
(In reply to comment #105)
The MAP_ANONYMOUS|MAP_SHARED mapping works fine with SELinux (at least on Fedora 12) but the pages are not linked together, so a change in write page is not propagated to a executable one. IMHO MAP_SHARED here just means that the memory can be shared.
Comment 107•15 years ago
|
||
There's the v3 version synced with latest tracemonkey trunk. I've updated all arches although can check i386&x86_64 only. Is there anything else I can help with to move it forward?
Attachment #453033 -
Flags: review?(nnethercote)
Updated•15 years ago
|
Attachment #438016 -
Flags: feedback?(stransky)
Comment 108•15 years ago
|
||
Thanks for the update.
(In reply to comment #107)
> Is there anything else I can help
> with to move it forward?
Answers to the questions in comment 99 would help. Thanks.
Comment 109•15 years ago
|
||
(In reply to comment #99)
> What attacks does this patch prevent, and which ones does it leave open? Let's
> make the trade-offs extremely clear, so we have a record of our decision, if
> nothing else.
When SELinux is enabled flash plug-in can run in its own sandbox (as OOP) so we don't have to care about it.
The issue is the browser itself, with this patch an attacker can write (somehow) its code to write-only page and execute it then, even with enabled SELinux.
Let's assume an attacker somehow delivers its code to the JIT R/W page and tries to execute it. He may:
- write some specific constant to R/W page and search for it in all executable pages. But we can mmap the executable pages as executable only, w/o the read permissions.
- read /proc/PID/maps and look for mapped twins here. For instance I have there:
7ffff7fda000-7ffff7fea000 rw-s 00020000 08:01 3358809 /tmp/jitU8GOD2 (deleted)
7ffff7fea000-7ffff7ffa000 --xs 00020000 08:01 3358809 /tmp/jitU8GOD2 (deleted)
where "/tmp/jitU8GOD2 (deleted)" is the temporary file used for mmap so it gives clear 7ffff7fda000 => 7ffff7fea000 mapping. But again, access to the process map file can be disabled by SELinux.
So with this small tweak (map the executable memory as executable only without the read permission) I don't see any way how an attacker can find any executable page.
Updated•15 years ago
|
Comment 110•15 years ago
|
||
Hey nick, whats the status of this?
Comment 111•15 years ago
|
||
(In reply to comment #110)
> Hey nick, whats the status of this?
The code works, someone needs to decide if it's a good idea and should be landed. Eg. see comment 99.
Comment 112•15 years ago
|
||
(In reply to comment #111)
> The code works, someone needs to decide if it's a good idea and should be
> landed. Eg. see comment 99.
Well, who can make such decision?
Comment 113•14 years ago
|
||
(In reply to comment #109)
> Let's assume an attacker somehow delivers its code to the JIT R/W page and
> tries to execute it. He may:
>
> [...]
Dumb question: what stops the attacker from reading the same execOffset field that TraceMonkey itself uses to find the other mapping?
Comment 114•14 years ago
|
||
(In reply to comment #113)
> Dumb question: what stops the attacker from reading the same execOffset field
> that TraceMonkey itself uses to find the other mapping?
Nothing. But the attacker has to search for page header and if the executable page is executable only, he can't confirm the right executable address.
Comment 115•14 years ago
|
||
(In reply to comment #114)
> (In reply to comment #113)
> > Dumb question: what stops the attacker from reading the same execOffset field
> > that TraceMonkey itself uses to find the other mapping?
>
> Nothing. But the attacker has to search for page header and if the executable
> page is executable only, he can't confirm the right executable address.
See comment 93 and please read the paper. I think this "attacker has to search" point is meaningless.
/be
Comment 116•14 years ago
|
||
I read it tree times.
> When this executable page is entered at a known offset (0x6A, for the
> given example code below) will execute stage-0 shellcode that marks the
> rest of the page RWX and copies the next stage of shellcode
> from an ActionScript string that can be modified
> before compilation.
- SELinux does not allow to change page permissions. So once you allocate it as X, you can't change it to RWX.
- How do you manage to launch compiled script it with such offset?
> In the end, it's really just a pointer into the heap.
> To try and find a use for this pointer, we must understand the Flash heap and
> some of the details of Windows memory architecture.
How is this related to Linux/SELinux? And if there is any relevance (leaving unused executable segments on heap for instance) we can easily fix them, can't we?
I'm not a security expert but I didn't see anything relative in the paper. If you think I'm wrong please provide the details.
Comment 117•14 years ago
|
||
An interesting reading:
http://www.zdnet.com/blog/security/questions-for-pwn2own-hacker-charlie-miller/2941
"It’s really simple. Safari on the Mac is easier to exploit. The things that Windows do to make it harder (for an exploit to work), Macs don’t do. Hacking into Macs is so much easier. You don’t have to jump through hoops and deal with all the anti-exploit mitigations you’d find in Windows.
It’s more about the operating system than the (target) program. Firefox on Mac is pretty easy too. The underlying OS doesn’t have anti-exploit stuff built into it.
With my Safari exploit, I put the code into a process and I know exactly where it’s going to be. There’s no randomization. I know when I jump there, the code is there and I can execute it there. On Windows, the code might show up but I don’t know where it is. Even if I get to the code, it’s not executable. Those are two hurdles that Macs don’t have."
... do we really want to weak whole browser just because of one small part which fails to follow modern security standards?
Comment 118•14 years ago
|
||
Martin, Charlie was talking about lack of ASLR. This bug does nothing of the kind.
Re comment 116 -- return-oriented programming means you don't need to write code, you let the JIT write code and then find interesting "instructions" (not the ones generated) that can be hooked together by tail jumps and returns. If this is (as I contend) the threat to defend against, I don't see how this bug's patch helps.
/be
Comment 119•14 years ago
|
||
(In reply to comment #118)
>Re comment 116 -- return-oriented programming means you don't need
> to write code, you let the JIT write code and then find
> interesting "instructions" (not the ones generated) that
> can be hooked together by tail jumps and returns. If this
> is (as I contend) the threat to defend against, I don't see
> how this bug's patch helps.
With this patch and SELinux enabled, you can't switch the pages to RWX so you have to generate the initial code from those crafted/switched instruction.
If you want to load any code from any external source, it has to be smaller than the executable page and you have to calculate proper address in RW page. (You have to load it somewhere, right? And not into the executable page where is your code executed from).
And if you want to generate interpage tail jumps and returns that you're referencing, the exploit has to translate the address from write to executable space in the same way as the JIT does it.
I'm not convicted that the exploit, generated from malformed constants can find the page header, read execOffset, recalculate target, find and load the exploit, execute it and manage to calculate jumps/returns.
For recapitulation, the exploiting scenario with this patch could be:
1) create the crafted JIT code
2) execute it with given offset (how?)
3) malformed code find execOffset of its page (I don't know how but let's assume it's possible) or call toExec() or calcOffset() and calculates writtable twin (how?)
4) loads prepared code from an external source, (it has to fit ton the executable page size, actually it can't rewrite its current code so it may be even smaller)
5) fixes return point (by execOffset/toExec()/calcOffset()) and launch main exploit.
btw. You can see why I'm talking about "attacker has to search for page header" now, the execOffset is located there.
And we still can make it harder. Those steps would have an impact to performance (often memory allocation/deallocation, frequent interpage jumps) so they could be considered as optional:
- divide code to more independent segments. An attacker will be more restricted by maximal exploit size and will have less space for write -> executable page recalculation.
- aggressively free unused memory segments and allocate only what we need. Attacker would not find another space where he can load and execute the code.
Comment 120•14 years ago
|
||
Ping. Any update here?
Comment 121•14 years ago
|
||
I really don't think this blocks... it didn't block the last release either, so it's not a regression.
Comment 122•14 years ago
|
||
First version of double page patch for JM. It still contains a crash when it's run outside of gdb (it passes all js trace tests inside gdb on i386).
I tried another approach - to make it as small/simple as possible and translate the addresses on the lowest level. A performance could be improved by a page cache (I'll do speed tests when I sort out the crash outside of gdb).
I think the same approach would be applied to tracemonkey so the both engines would share the same ExecutableTranslator service.
Attachment #488986 -
Flags: feedback?(nnethercote)
Attachment #488986 -
Flags: feedback?(brendan)
Comment 123•14 years ago
|
||
Comment on attachment 488986 [details] [diff] [review]
JM WIP/i386
Ask a JM lead hacker for review.
/be
Attachment #488986 -
Flags: feedback?(brendan) → feedback?(dvander)
Comment 124•14 years ago
|
||
If the performance effects are not indistinguishable from noise, a minimally #ifdef'ed patch to let SELinux turn this on would be better -- provided that side of the ifdefs will see testing and use.
/be
Comment 125•14 years ago
|
||
Fixed offset calculation, added memory management for allocated pages, added page cache, range check, page translation statistics.
It's missing fixes in trap handler, (it's not redirected to correct return address) and fails the trap-* jit_tests.
Attachment #488986 -
Attachment is obsolete: true
Attachment #490903 -
Flags: feedback?(nnethercote)
Attachment #490903 -
Flags: feedback?(dvander)
Attachment #488986 -
Flags: feedback?(nnethercote)
Attachment #488986 -
Flags: feedback?(dvander)
Updated•14 years ago
|
Attachment #490903 -
Flags: feedback?(nnethercote)
Comment 126•14 years ago
|
||
This one combines patches for nanojit and methodjit, uses shared memory allocator (MemoryManager) plus the dual mapping can be controlled by --enable-dual-mapping configure switch.
From my measures the difference in performance between dual/single page builds is ~ 10ms for SpiderMonkey test and for optimized build. (~290ms single, ~300ms double on i5 box).
Attachment #453033 -
Attachment is obsolete: true
Attachment #490903 -
Attachment is obsolete: true
Attachment #497240 -
Flags: review?
Attachment #490903 -
Flags: feedback?(dvander)
Attachment #453033 -
Flags: review?(nnethercote)
Updated•14 years ago
|
Attachment #497240 -
Flags: review? → review?(dvander)
Updated•13 years ago
|
Assignee: nnethercote → general
Updated•13 years ago
|
Status: ASSIGNED → NEW
Comment 127•12 years ago
|
||
Comment on attachment 497240 [details] [diff] [review]
nanojit & methodjit patch
Canceling old r?s.
Attachment #497240 -
Flags: review?(dvander)
Comment 128•11 years ago
|
||
Was the security patch applied? If not: why not?
Comment 129•11 years ago
|
||
I do not see why this issue depends on bug 555111 and bug 555112.
But those issues seem to soehow depend on this one. Was there no progress regarding this issue due to circular issue dependencies? (an issue deadlock?)
The patches in this bug were very large and it wasn't clear whether there was actually any security benefit (see comment #7, comment #16) for the maintenance effort. By now these patches would not be applicable anyway since we have entirely new JITs.
The best path forward for JIT security is sandboxed content processes, which is in progress for both Desktop Firefox and Firefox OS. Sandboxing will greatly reduce the threat of JIT exploits, and it's unlikely any other solution will be as effective. For Desktop, it's part of the Electrolysis effort:
[1] https://wiki.mozilla.org/FoxInABox
[2] https://wiki.mozilla.org/Electrolysis
Status: NEW → RESOLVED
Closed: 11 years ago
Resolution: --- → WONTFIX
You need to log in
before you can comment on or make changes to this bug.
Description
•