Tim Peters <t...@python.org> added the comment:

I think it's time to change what address_in_range() tries to answer. It 
currently gives a precise answer to "is this byte address in a region obmalloc 
owns?". But that's stronger than what it needs to do its job: the real question 
is "is this an address that obmalloc could return from its malloc() or 
realloc() functions?". Because it's only used by free() and realloc(), and they 
only care where the address they're _passed_ came from.

The difference is when an arena is not pool-aligned. Oddball chunks outside of 
full pools, at both ends of the arena, are never returned by obmalloc then.

Instead the tree could be changed to record the starting addresses of the 
_full_ pools obmalloc controls.  Those contain the only addresses obmalloc will 
ever pass out (from malloc() or realloc()).

Like so, where p is the arena base address. This hasn't been tested, and may 
contain typos or logical errors. I'm far more interested in the latter for now 
;-)

ideal1 = p & ~ARENA_SIZE_MASK # round down to ideal
ideal2 = ideal1 + ARENA_SIZE
offset = p - ideal1
b1 = bottom_node(ideal1)
b2 = bottom_node(ideal2)
if not offset:
    # already ideal
    b1.hi = -1
    assert b2.lo == 0
elif offset & POOL_SIZE_MASK == 0:
    # arena is pool-aligned
    b1.hi = b2.lo = offset
else:
    # Not pool-aligned.
    # obmalloc won't pass out an address higher than in
    # the last full pool.
    # Round offset down to next lower pool address.
    offset &= ~POOL_SIZE_MASK
    b2.lo = offset
    # And won't pass out an address lower than in the
    # first full pool.
    offset += POOL_SIZE # same as rounding original offset up
    # That's almost right for b1.hi, but one exception: if
    # offset is ARENA_SIZE now, the first full pool is at the
    # start of ideal2, and no feasible address is in ideal1.
    assert offset <= ARENA_SIZE
    b1.hi = offset & ARENA_SIZE_MASK 

Note that, except for the oddball -1, everything stored in a bottom node is a 
pool address then, and so there's no need to store their all-zero lowest 
POOL_BITS bits. .lo and .hi can be shrunk to a signed 8-bit int with the 
current settings (20 arena bits and 14 pool bits).

And caching pool addresses wouldn't have any obscure failing end-cases either: 
address_in_range() could just as well be passed a pool address to begin with. 
It would only care about pool addresses, not byte addresses.

----------

_______________________________________
Python tracker <rep...@bugs.python.org>
<https://bugs.python.org/issue43593>
_______________________________________
_______________________________________________
Python-bugs-list mailing list
Unsubscribe: 
https://mail.python.org/mailman/options/python-bugs-list/archive%40mail-archive.com

Reply via email to