Skip to content

Commit

Permalink
py/objarray: Raise error on out-of-bound memoryview slice start.
Browse files Browse the repository at this point in the history
32-bit platforms only support a slice offset start of 24 bit max due to the
limited size of the mp_obj_array_t.free member.  Similarly on 64-bit
platforms the limit is 56 bits.

This commit adds an OverflowError if the user attempts to slice a
memoryview beyond this limit.

Signed-off-by: Damien George <[email protected]>
  • Loading branch information
pi-anl authored and dpgeorge committed Jan 20, 2023
1 parent d6bc34a commit 5c4153e
Show file tree
Hide file tree
Showing 4 changed files with 37 additions and 1 deletion.
4 changes: 4 additions & 0 deletions py/objarray.c
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@
#if MICROPY_PY_BUILTINS_MEMORYVIEW
#define TYPECODE_MASK (0x7f)
#define memview_offset free
#define memview_offset_max ((1LL << MP_OBJ_ARRAY_FREE_SIZE_BITS) - 1)
#else
// make (& TYPECODE_MASK) a null operation if memorview not enabled
#define TYPECODE_MASK (~(size_t)0)
Expand Down Expand Up @@ -522,6 +523,9 @@ STATIC mp_obj_t array_subscr(mp_obj_t self_in, mp_obj_t index_in, mp_obj_t value
assert(sz > 0);
#if MICROPY_PY_BUILTINS_MEMORYVIEW
if (o->base.type == &mp_type_memoryview) {
if (slice.start > memview_offset_max) {
mp_raise_msg(&mp_type_OverflowError, MP_ERROR_TEXT("memoryview offset too large"));
}
res = m_new_obj(mp_obj_array_t);
*res = *o;
res->memview_offset += slice.start;
Expand Down
5 changes: 4 additions & 1 deletion py/objarray.h
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,9 @@
// Used only for memoryview types, set in "typecode" to indicate a writable memoryview
#define MP_OBJ_ARRAY_TYPECODE_FLAG_RW (0x80)

// Bit size used for mp_obj_array_t.free member.
#define MP_OBJ_ARRAY_FREE_SIZE_BITS (8 * sizeof(size_t) - 8)

// This structure is used for all of bytearray, array.array, memoryview
// objects. Note that memoryview has different meaning for some fields,
// see comment at the beginning of objarray.c.
Expand All @@ -44,7 +47,7 @@ typedef struct _mp_obj_array_t {
// parent object. (Union is not used to not go into a complication of
// union-of-bitfields with different toolchains). See comments in
// objarray.c.
size_t free : (8 * sizeof(size_t) - 8);
size_t free : MP_OBJ_ARRAY_FREE_SIZE_BITS;
size_t len; // in elements
void *items;
} mp_obj_array_t;
Expand Down
27 changes: 27 additions & 0 deletions tests/basics/memoryview_slice_size.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# test memoryview slicing beyond the limit of what memoryview can internally index

try:
from sys import maxsize
from uctypes import bytearray_at

memoryview
except:
print("SKIP")
raise SystemExit

if maxsize <= 0xFFFF_FFFF:
slice_max = 0xFF_FFFF
else:
slice_max = 0xFF_FFFF_FFFF_FFFF

buf = bytearray_at(0, slice_max + 2)
mv = memoryview(buf)

# this should work
print(mv[slice_max : slice_max + 1])

# this should overflow the internal index for memoryview slicing
try:
print(mv[slice_max + 1 : slice_max + 2])
except OverflowError:
print("OverflowError")
2 changes: 2 additions & 0 deletions tests/basics/memoryview_slice_size.py.exp
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
<memoryview>
OverflowError

0 comments on commit 5c4153e

Please sign in to comment.