Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
nexedi
cython
Commits
abf0d29c
Commit
abf0d29c
authored
Jul 26, 2008
by
Dag Sverre Seljebotn
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Passes proper buffer flags (including auto-detected readonly)
parent
aaedcdea
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
107 additions
and
18 deletions
+107
-18
Cython/Compiler/Buffer.py
Cython/Compiler/Buffer.py
+21
-16
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+5
-0
Cython/Compiler/ParseTreeTransforms.py
Cython/Compiler/ParseTreeTransforms.py
+1
-1
Cython/Compiler/PyrexTypes.py
Cython/Compiler/PyrexTypes.py
+1
-1
Cython/Compiler/Symtab.py
Cython/Compiler/Symtab.py
+2
-0
tests/run/bufaccess.pyx
tests/run/bufaccess.pyx
+77
-0
No files found.
Cython/Compiler/Buffer.py
View file @
abf0d29c
...
@@ -8,6 +8,11 @@ from Cython.Compiler.Errors import CompileError
...
@@ -8,6 +8,11 @@ from Cython.Compiler.Errors import CompileError
import
PyrexTypes
import
PyrexTypes
from
sets
import
Set
as
set
from
sets
import
Set
as
set
def
get_flags
(
buffer_aux
,
buffer_type
):
flags
=
'PyBUF_FORMAT | PyBUF_INDIRECT'
if
buffer_aux
.
writable_needed
:
flags
+=
"| PyBUF_WRITABLE"
return
flags
def
used_buffer_aux_vars
(
entry
):
def
used_buffer_aux_vars
(
entry
):
buffer_aux
=
entry
.
buffer_aux
buffer_aux
=
entry
.
buffer_aux
buffer_aux
.
buffer_info_var
.
used
=
True
buffer_aux
.
buffer_info_var
.
used
=
True
...
@@ -33,19 +38,26 @@ def put_zero_buffer_aux_into_scope(buffer_aux, code):
...
@@ -33,19 +38,26 @@ def put_zero_buffer_aux_into_scope(buffer_aux, code):
code
.
putln
(
" "
.
join
([
"%s = 0;"
%
s
.
cname
code
.
putln
(
" "
.
join
([
"%s = 0;"
%
s
.
cname
for
s
in
buffer_aux
.
shapevars
]))
for
s
in
buffer_aux
.
shapevars
]))
def
getbuffer_cond_code
(
obj_cname
,
buffer_aux
,
flags
,
ndim
):
bufstruct
=
buffer_aux
.
buffer_info_var
.
cname
checker
=
buffer_aux
.
tschecker
return
"PyObject_GetBuffer(%s, &%s, %s) == -1 || %s(&%s, %d) == -1"
%
(
obj_cname
,
bufstruct
,
flags
,
checker
,
bufstruct
,
ndim
)
def
put_acquire_arg_buffer
(
entry
,
code
,
pos
):
def
put_acquire_arg_buffer
(
entry
,
code
,
pos
):
buffer_aux
=
entry
.
buffer_aux
buffer_aux
=
entry
.
buffer_aux
cname
=
entry
.
cname
cname
=
entry
.
cname
bufstruct
=
buffer_aux
.
buffer_info_var
.
cname
bufstruct
=
buffer_aux
.
buffer_info_var
.
cname
flags
=
'0'
flags
=
get_flags
(
buffer_aux
,
entry
.
type
)
# Acquire any new buffer
# Acquire any new buffer
code
.
put
(
'if (%s != Py_None) '
%
cname
)
code
.
put
(
'if (%s != Py_None) '
%
cname
)
code
.
begin_block
()
code
.
begin_block
()
code
.
putln
(
'%s.buf = 0;'
%
bufstruct
)
# PEP requirement
code
.
putln
(
'%s.buf = 0;'
%
bufstruct
)
# PEP requirement
code
.
put
(
code
.
error_goto_if
(
code
.
put
(
code
.
error_goto_if
(
getbuffer_cond_code
(
cname
,
'PyObject_GetBuffer(%s, &%s, %s) == -1 || %s(&%s, %d) == -1'
%
(
buffer_aux
,
cname
,
bufstruct
,
flags
,
buffer_aux
.
tschecker
,
bufstruct
,
entry
.
type
.
ndim
),
flags
,
pos
))
entry
.
type
.
ndim
),
pos
))
# An exception raised in arg parsing cannot be catched, so no
# An exception raised in arg parsing cannot be catched, so no
# need to do care about the buffer then.
# need to do care about the buffer then.
put_unpack_buffer_aux_into_scope
(
buffer_aux
,
code
)
put_unpack_buffer_aux_into_scope
(
buffer_aux
,
code
)
...
@@ -58,7 +70,7 @@ def put_release_buffer(entry, code):
...
@@ -58,7 +70,7 @@ def put_release_buffer(entry, code):
def
put_assign_to_buffer
(
lhs_cname
,
rhs_cname
,
buffer_aux
,
buffer_type
,
def
put_assign_to_buffer
(
lhs_cname
,
rhs_cname
,
buffer_aux
,
buffer_type
,
is_initialized
,
pos
,
code
):
is_initialized
,
pos
,
code
):
bufstruct
=
buffer_aux
.
buffer_info_var
.
cname
bufstruct
=
buffer_aux
.
buffer_info_var
.
cname
flags
=
'0'
flags
=
get_flags
(
buffer_aux
,
buffer_type
)
if
is_initialized
:
if
is_initialized
:
# Release any existing buffer
# Release any existing buffer
...
@@ -71,14 +83,7 @@ def put_assign_to_buffer(lhs_cname, rhs_cname, buffer_aux, buffer_type,
...
@@ -71,14 +83,7 @@ def put_assign_to_buffer(lhs_cname, rhs_cname, buffer_aux, buffer_type,
code
.
put
(
'if (%s != Py_None) '
%
rhs_cname
)
code
.
put
(
'if (%s != Py_None) '
%
rhs_cname
)
code
.
begin_block
()
code
.
begin_block
()
code
.
putln
(
'%s.buf = 0;'
%
bufstruct
)
# PEP requirement
code
.
putln
(
'%s.buf = 0;'
%
bufstruct
)
# PEP requirement
code
.
put
(
'if (%s) '
%
code
.
unlikely
(
code
.
put
(
'if (%s) '
%
code
.
unlikely
(
getbuffer_cond_code
(
rhs_cname
,
buffer_aux
,
flags
,
buffer_type
.
ndim
)))
'PyObject_GetBuffer(%s, &%s, %s) == -1'
%
(
rhs_cname
,
bufstruct
,
flags
)
+
' || %s(&%s, %d) == -1'
%
(
buffer_aux
.
tschecker
,
bufstruct
,
buffer_type
.
ndim
)))
code
.
begin_block
()
code
.
begin_block
()
# If acquisition failed, attempt to reacquire the old buffer
# If acquisition failed, attempt to reacquire the old buffer
# before raising the exception. A failure of reacquisition
# before raising the exception. A failure of reacquisition
...
@@ -86,8 +91,8 @@ def put_assign_to_buffer(lhs_cname, rhs_cname, buffer_aux, buffer_type,
...
@@ -86,8 +91,8 @@ def put_assign_to_buffer(lhs_cname, rhs_cname, buffer_aux, buffer_type,
# can consider working around this later.
# can consider working around this later.
if
is_initialized
:
if
is_initialized
:
put_zero_buffer_aux_into_scope
(
buffer_aux
,
code
)
put_zero_buffer_aux_into_scope
(
buffer_aux
,
code
)
code
.
put
(
'if (%s != Py_None &&
PyObject_GetBuffer(%s, &%s, %s) == -1) '
%
(
code
.
put
(
'if (%s != Py_None &&
(%s)) '
%
(
rhs_cname
,
lhs_cname
,
lhs_cname
,
bufstruct
,
flags
))
getbuffer_cond_code
(
rhs_cname
,
buffer_aux
,
flags
,
buffer_type
.
ndim
)
))
code
.
begin_block
()
code
.
begin_block
()
put_zero_buffer_aux_into_scope
(
buffer_aux
,
code
)
put_zero_buffer_aux_into_scope
(
buffer_aux
,
code
)
code
.
end_block
()
code
.
end_block
()
...
...
Cython/Compiler/ExprNodes.py
View file @
abf0d29c
...
@@ -1371,6 +1371,11 @@ class IndexNode(ExprNode):
...
@@ -1371,6 +1371,11 @@ class IndexNode(ExprNode):
# we only need a temp because result_code isn't refactored to
# we only need a temp because result_code isn't refactored to
# generation time, but this seems an ok shortcut to take
# generation time, but this seems an ok shortcut to take
self
.
is_temp
=
True
self
.
is_temp
=
True
if
setting
:
if
not
self
.
base
.
entry
.
type
.
writable
:
error
(
self
.
pos
,
"Writing to readonly buffer"
)
else
:
self
.
base
.
entry
.
buffer_aux
.
writable_needed
=
True
else
:
else
:
if
isinstance
(
self
.
index
,
TupleNode
):
if
isinstance
(
self
.
index
,
TupleNode
):
self
.
index
.
analyse_types
(
env
,
skip_children
=
skip_child_analysis
)
self
.
index
.
analyse_types
(
env
,
skip_children
=
skip_child_analysis
)
...
...
Cython/Compiler/ParseTreeTransforms.py
View file @
abf0d29c
...
@@ -182,7 +182,7 @@ class PostParse(CythonTransform):
...
@@ -182,7 +182,7 @@ class PostParse(CythonTransform):
node
.
ndim
=
int
(
ndimnode
.
value
)
node
.
ndim
=
int
(
ndimnode
.
value
)
else
:
else
:
node
.
ndim
=
1
node
.
ndim
=
1
# We're done with the parse tree args
# We're done with the parse tree args
node
.
positional_args
=
None
node
.
positional_args
=
None
node
.
keyword_args
=
None
node
.
keyword_args
=
None
...
...
Cython/Compiler/PyrexTypes.py
View file @
abf0d29c
...
@@ -198,7 +198,7 @@ class BufferType(BaseType):
...
@@ -198,7 +198,7 @@ class BufferType(BaseType):
# ndim int
# ndim int
is_buffer
=
1
is_buffer
=
1
writable
=
True
def
__init__
(
self
,
base
,
dtype
,
ndim
):
def
__init__
(
self
,
base
,
dtype
,
ndim
):
self
.
base
=
base
self
.
base
=
base
self
.
dtype
=
dtype
self
.
dtype
=
dtype
...
...
Cython/Compiler/Symtab.py
View file @
abf0d29c
...
@@ -20,6 +20,8 @@ possible_identifier = re.compile(ur"(?![0-9])\w+$", re.U).match
...
@@ -20,6 +20,8 @@ possible_identifier = re.compile(ur"(?![0-9])\w+$", re.U).match
nice_identifier = re.compile('^[a-zA-Z0-0_]+$').match
nice_identifier = re.compile('^[a-zA-Z0-0_]+$').match
class BufferAux:
class BufferAux:
writable_needed = False
def __init__(self, buffer_info_var, stridevars, shapevars, tschecker):
def __init__(self, buffer_info_var, stridevars, shapevars, tschecker):
self.buffer_info_var = buffer_info_var
self.buffer_info_var = buffer_info_var
self.stridevars = stridevars
self.stridevars = stridevars
...
...
tests/run/bufaccess.pyx
View file @
abf0d29c
cimport
__cython__
cimport
__cython__
# Tests the buffer access syntax functionality by constructing
# mock buffer objects.
#
# Note that the buffers are mock objects created for testing
# the buffer access behaviour -- for instance there is no flag
# checking in the buffer objects (why test our test case?), rather
# what we want to test is what is passed into the flags argument.
#
cimport
stdlib
cimport
stdlib
cimport
python_buffer
# Add all test_X function docstrings as unit tests
# Add all test_X function docstrings as unit tests
__test__
=
{}
__test__
=
{}
...
@@ -251,6 +262,50 @@ def ndim1(object[int, 2] buf):
...
@@ -251,6 +262,50 @@ def ndim1(object[int, 2] buf):
ValueError: Buffer has wrong number of dimensions (expected 2, got 1)
ValueError: Buffer has wrong number of dimensions (expected 2, got 1)
"""
"""
#
# Test which flags are passed.
#
@
testcase
def
readonly
(
obj
):
"""
>>> R = UnsignedShortMockBuffer("R", range(27), shape=(3, 3, 3))
>>> readonly(R)
acquired R
25
released R
>>> R.recieved_flags
['FORMAT', 'INDIRECT', 'ND', 'STRIDES']
"""
cdef
object
[
unsigned
short
int
,
3
]
buf
=
obj
print
buf
[
2
,
2
,
1
]
@
testcase
def
writable
(
obj
):
"""
>>> R = UnsignedShortMockBuffer("R", range(27), shape=(3, 3, 3))
>>> writable(R)
acquired R
released R
>>> R.recieved_flags
['FORMAT', 'INDIRECT', 'ND', 'STRIDES', 'WRITABLE']
"""
cdef
object
[
unsigned
short
int
,
3
]
buf
=
obj
buf
[
2
,
2
,
1
]
=
23
#
# Coercions
#
@
testcase
def
coercions
(
object
[
unsigned
char
]
uc
):
"""
TODO
"""
print
type
(
uc
[
0
])
uc
[
0
]
=
-
1
print
uc
[
0
]
uc
[
0
]
=
<
int
>
3.14
print
uc
[
0
]
@
testcase
@
testcase
def
printbuf_float
(
o
,
shape
):
def
printbuf_float
(
o
,
shape
):
...
@@ -270,6 +325,14 @@ def printbuf_float(o, shape):
...
@@ -270,6 +325,14 @@ def printbuf_float(o, shape):
print
print
available_flags
=
(
(
'FORMAT'
,
python_buffer
.
PyBUF_FORMAT
),
(
'INDIRECT'
,
python_buffer
.
PyBUF_INDIRECT
),
(
'ND'
,
python_buffer
.
PyBUF_ND
),
(
'STRIDES'
,
python_buffer
.
PyBUF_STRIDES
),
(
'WRITABLE'
,
python_buffer
.
PyBUF_WRITABLE
)
)
cdef
class
MockBuffer
:
cdef
class
MockBuffer
:
cdef
object
format
cdef
object
format
cdef
char
*
buffer
cdef
char
*
buffer
...
@@ -277,6 +340,7 @@ cdef class MockBuffer:
...
@@ -277,6 +340,7 @@ cdef class MockBuffer:
cdef
Py_ssize_t
*
strides
cdef
Py_ssize_t
*
strides
cdef
Py_ssize_t
*
shape
cdef
Py_ssize_t
*
shape
cdef
object
label
,
log
cdef
object
label
,
log
cdef
readonly
object
recieved_flags
def
__init__
(
self
,
label
,
data
,
shape
=
None
,
strides
=
None
,
format
=
None
):
def
__init__
(
self
,
label
,
data
,
shape
=
None
,
strides
=
None
,
format
=
None
):
self
.
label
=
label
self
.
label
=
label
...
@@ -313,6 +377,12 @@ cdef class MockBuffer:
...
@@ -313,6 +377,12 @@ cdef class MockBuffer:
if
buffer
is
NULL
:
if
buffer
is
NULL
:
print
u"locking!"
print
u"locking!"
return
return
self
.
recieved_flags
=
[]
for
name
,
value
in
available_flags
:
if
(
value
&
flags
)
==
value
:
self
.
recieved_flags
.
append
(
name
)
buffer
.
buf
=
self
.
buffer
buffer
.
buf
=
self
.
buffer
buffer
.
len
=
self
.
len
buffer
.
len
=
self
.
len
buffer
.
readonly
=
0
buffer
.
readonly
=
0
...
@@ -363,6 +433,13 @@ cdef class IntMockBuffer(MockBuffer):
...
@@ -363,6 +433,13 @@ cdef class IntMockBuffer(MockBuffer):
return
0
return
0
cdef
get_itemsize
(
self
):
return
sizeof
(
int
)
cdef
get_itemsize
(
self
):
return
sizeof
(
int
)
cdef
get_default_format
(
self
):
return
"=i"
cdef
get_default_format
(
self
):
return
"=i"
cdef
class
UnsignedShortMockBuffer
(
MockBuffer
):
cdef
int
write
(
self
,
char
*
buf
,
object
value
)
except
-
1
:
(
<
unsigned
short
*>
buf
)[
0
]
=
<
unsigned
short
>
value
return
0
cdef
get_itemsize
(
self
):
return
sizeof
(
unsigned
short
)
cdef
get_default_format
(
self
):
return
"=H"
cdef
class
ErrorBuffer
:
cdef
class
ErrorBuffer
:
cdef
object
label
cdef
object
label
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment