Opened 15 years ago
Closed 15 years ago
#6312 closed enhancement (fixed)
slab: protection from wrong freed objects
Reported by: | lucian | Owned by: | axeld |
---|---|---|---|
Priority: | normal | Milestone: | R1 |
Component: | System/Kernel | Version: | R1/Development |
Keywords: | Cc: | ||
Blocked By: | Blocking: | ||
Platform: | All |
Description
A kernel add-on can easily corrupt a SLAB by calling free()
on a pointer that was not received from a corresponding malloc()
.
Attachments (2)
Change History (6)
by , 15 years ago
Attachment: | slab-ReturnObjectToSlab-protect-from-wrong-freed-object-v1.patch added |
---|
comment:1 by , 15 years ago
patch: | 0 → 1 |
---|
comment:2 by , 15 years ago
follow-up: 4 comment:3 by , 15 years ago
Are you sure the second check is not correct?
intptr_t objectOffset = data - source->offset - (uint8*) source->pages; if (objectOffset % object_size != 0) { panic... }
I do remove the slab::offset before checking: data - source->offset
.
I took as guide the way objects are created in InitSlab:
uint8* data = ((uint8*)pages) + slab->offset; for (size_t i = 0; i < slab->size; i++) { status = constructor(cookie, data); data += object_size; }
The lower bounds check can be tighter (should be data < source->pages + source->offset
I'll address that in a second version, but as far as I understand the code there's no off-by-one error (the check uses ">=", not ">").
I wasn't aware one could return after a panic()
.
I'm not as sure as you are about the KDEBUG >= 1 checks. Trashing a slab's ->free list can easily lead to data corruption or data loss. As a user (not only as a developer) I'd prefer to reboot than have data silently corrupted (think of a malloc() from a file system that receives a pointer to an object that's being written by someone else in parallel corrupting data in transit or before being sent to the disk). I'll put them in because you requested them, but I still disagree with you.
I've rewritten the patch.
by , 15 years ago
Attachment: | slab-ReturnObjectToSlab-protect-from-wrong-freed-object-v2.patch added |
---|
comment:4 by , 15 years ago
Resolution: | → fixed |
---|---|
Status: | new → closed |
Replying to lucian:
Are you sure the second check is not correct?
intptr_t objectOffset = data - source->offset - (uint8*) source->pages; if (objectOffset % object_size != 0) { panic... }I do remove the slab::offset before checking:
data - source->offset
.
Yeah, sorry, I overlooked that.
I took as guide the way objects are created in InitSlab:
uint8* data = ((uint8*)pages) + slab->offset; for (size_t i = 0; i < slab->size; i++) { status = constructor(cookie, data); data += object_size; }The lower bounds check can be tighter (should be
data < source->pages + source->offset
I'll address that in a second version, but as far as I understand the code there's no off-by-one error (the check uses ">=", not ">").
You're right. The check could be tighter, but just like for the lower bounds check, the alignment check catches those cases anyway. Never mind.
I wasn't aware one could return after a
panic()
.
Yeah, the continue
KDL command does that for you. Depending on the case that might not make much sense or will cause the same panic()
to occur again (e.g. on page fault), but some situations are well continuable (as this one). Although one might want to shut the system down orderly as soon as possible and fix the cause of the problem.
I'm not as sure as you are about the KDEBUG >= 1 checks. Trashing a slab's ->free list can easily lead to data corruption or data loss. As a user (not only as a developer) I'd prefer to reboot than have data silently corrupted (think of a malloc() from a file system that receives a pointer to an object that's being written by someone else in parallel corrupting data in transit or before being sent to the disk). I'll put them in because you requested them, but I still disagree with you.
There's always a trade-off when it comes to error checks. This is pretty hot code (on UP machines called for every free()), so only very cheap checks for programmer errors should be in the production code. I'd also like to add a check for double-frees, but since that is even more expensive, I would only enable it at a higher debug level or as an optional debug feature.
I've rewritten the patch.
Thanks. Committed in hrev37508 with small changes: I joined the two if
s and adjusted the panic()
text.
A few remarks:
source->size - 1
objects only.KDEBUG >= 1
.panic()
, i.e.return
in this case.