Skip to content
项目
群组
代码片段
帮助
当前项目
正在载入...
登录 / 注册
切换导航面板
P
pytensor
项目
项目
详情
活动
周期分析
仓库
仓库
文件
提交
分支
标签
贡献者
图表
比较
统计图
议题
0
议题
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
CI / CD
CI / CD
流水线
作业
日程
统计图
Wiki
Wiki
代码片段
代码片段
成员
成员
折叠边栏
关闭边栏
活动
图像
聊天
创建新问题
作业
提交
问题看板
Open sidebar
testgroup
pytensor
Commits
761c7f9f
提交
761c7f9f
authored
4月 07, 2008
作者:
olivier@olivier-desktop
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
added a few important files
上级
87103eff
隐藏空白字符变更
内嵌
并排
正在显示
2 个修改的文件
包含
400 行增加
和
0 行删除
+400
-0
_test_elemwise2.py
_test_elemwise2.py
+180
-0
elemwise_cgen.py
elemwise_cgen.py
+220
-0
没有找到文件。
_test_elemwise2.py
0 → 100644
浏览文件 @
761c7f9f
import
time
import
unittest
from
gof
import
ResultBase
,
Op
,
Env
,
modes
import
gof
from
scalar_ops
import
*
from
elemwise2
import
*
def
inputs
():
x
=
modes
.
build
(
Tensor
(
'float64'
,
(
0
,
0
),
name
=
'x'
))
y
=
modes
.
build
(
Tensor
(
'float64'
,
(
1
,
0
),
name
=
'y'
))
z
=
modes
.
build
(
Tensor
(
'float64'
,
(
0
,
0
),
name
=
'z'
))
return
x
,
y
,
z
def
env
(
inputs
,
outputs
,
validate
=
True
,
features
=
[]):
return
Env
(
inputs
,
outputs
,
features
=
features
,
consistency_check
=
validate
)
class
_test_DimShuffle
(
unittest
.
TestCase
):
def
test_straightforward
(
self
):
x
,
y
,
z
=
inputs
()
e0
=
DimShuffle
(
x
,
[
1
,
'x'
,
0
])
.
out
f
=
gof
.
PerformLinker
(
env
([
x
],
[
e0
]))
.
make_function
(
inplace
=
True
)
assert
f
(
numpy
.
ones
((
2
,
3
)))
.
shape
==
(
3
,
1
,
2
)
class
_test_Broadcast
(
unittest
.
TestCase
):
def
with_linker
(
self
,
linker
):
for
xsh
,
ysh
in
[((
5
,
5
),
(
5
,
5
)),
((
5
,
5
),
(
1
,
5
)),
((
5
,
5
),
(
5
,
1
)),
((
1
,
5
),
(
5
,
1
)),
((
1
,
1
),
(
1
,
1
)),
((
2
,
3
,
4
,
5
),
(
2
,
3
,
4
,
5
)),
((
2
,
3
,
4
,
5
),
(
1
,
3
,
1
,
5
)),
((
2
,
3
,
4
,
5
),
(
1
,
1
,
1
,
1
)),
((),
())]:
x
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
*
(
entry
==
1
)
for
entry
in
xsh
],
name
=
'x'
))
y
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
*
(
entry
==
1
)
for
entry
in
ysh
],
name
=
'y'
))
e
=
Broadcast
(
Add
,
(
x
,
y
))
.
out
f
=
linker
(
env
([
x
,
y
],
[
e
]))
.
make_function
(
inplace
=
False
)
# xv = numpy.array(range(numpy.product(xsh)))
# xv = xv.reshape(xsh)
# yv = numpy.array(range(numpy.product(ysh)))
# yv = yv.reshape(ysh)
xv
=
numpy
.
asarray
(
numpy
.
random
.
rand
(
*
xsh
))
yv
=
numpy
.
asarray
(
numpy
.
random
.
rand
(
*
ysh
))
zv
=
xv
+
yv
self
.
failUnless
((
f
(
xv
,
yv
)
==
zv
)
.
all
())
def
with_linker_inplace
(
self
,
linker
):
for
xsh
,
ysh
in
[((
5
,
5
),
(
5
,
5
)),
((
5
,
5
),
(
1
,
5
)),
((
5
,
5
),
(
5
,
1
)),
((
1
,
1
),
(
1
,
1
)),
((
2
,
3
,
4
,
5
),
(
2
,
3
,
4
,
5
)),
((
2
,
3
,
4
,
5
),
(
1
,
3
,
1
,
5
)),
((
2
,
3
,
4
,
5
),
(
1
,
1
,
1
,
1
)),
((),
())]:
x
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
*
(
entry
==
1
)
for
entry
in
xsh
],
name
=
'x'
))
y
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
*
(
entry
==
1
)
for
entry
in
ysh
],
name
=
'y'
))
e
=
Broadcast
(
Add
,
(
x
,
y
),
{
0
:
0
})
.
out
f
=
linker
(
env
([
x
,
y
],
[
e
]))
.
make_function
(
inplace
=
False
)
xv
=
numpy
.
asarray
(
numpy
.
random
.
rand
(
*
xsh
))
yv
=
numpy
.
asarray
(
numpy
.
random
.
rand
(
*
ysh
))
zv
=
xv
+
yv
f
(
xv
,
yv
)
self
.
failUnless
((
xv
==
zv
)
.
all
())
def
test_perform
(
self
):
self
.
with_linker
(
gof
.
PerformLinker
)
def
test_c
(
self
):
self
.
with_linker
(
gof
.
CLinker
)
def
test_perform_inplace
(
self
):
self
.
with_linker_inplace
(
gof
.
PerformLinker
)
def
test_c_inplace
(
self
):
self
.
with_linker_inplace
(
gof
.
CLinker
)
def
test_fill
(
self
):
x
=
modes
.
build
(
Tensor
(
'float64'
,
[
0
,
0
],
name
=
'x'
))
y
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
,
1
],
name
=
'y'
))
e
=
Broadcast
(
Second
,
(
x
,
y
),
{
0
:
0
})
.
out
f
=
gof
.
CLinker
(
env
([
x
,
y
],
[
e
]))
.
make_function
(
inplace
=
False
)
xv
=
numpy
.
ones
((
5
,
5
))
yv
=
numpy
.
random
.
rand
(
1
,
1
)
f
(
xv
,
yv
)
assert
(
xv
==
yv
)
.
all
()
class
_test_CAReduce
(
unittest
.
TestCase
):
def
with_linker
(
self
,
linker
):
for
xsh
,
tosum
in
[((
5
,
6
),
(
0
,
1
)),
((
5
,
6
),
(
0
,
)),
((
5
,
6
),
(
1
,
)),
((
2
,
3
,
4
,
5
),
(
0
,
1
,
3
))]:
x
=
modes
.
build
(
Tensor
(
'float64'
,
[
1
*
(
entry
==
1
)
for
entry
in
xsh
],
name
=
'x'
))
e
=
CAReduce
(
Add
,
[
x
],
dimensions_to_reduce
=
tosum
)
.
out
f
=
linker
(
env
([
x
],
[
e
]))
.
make_function
(
inplace
=
False
)
xv
=
numpy
.
asarray
(
numpy
.
random
.
rand
(
*
xsh
))
zv
=
xv
for
axis
in
reversed
(
sorted
(
tosum
)):
zv
=
numpy
.
add
.
reduce
(
zv
,
axis
)
self
.
failUnless
((
f
(
xv
)
-
zv
<
1e-10
)
.
all
())
def
test_perform
(
self
):
self
.
with_linker
(
gof
.
PerformLinker
)
def
test_c
(
self
):
self
.
with_linker
(
gof
.
CLinker
)
if
__name__
==
'__main__'
:
# unittest.main()
x
=
modes
.
build
(
Tensor
(
'float64'
,
[
0
,
0
],
name
=
'x'
))
y
=
modes
.
build
(
Tensor
(
'float64'
,
[
0
,
0
],
name
=
'y'
))
e
=
Broadcast
(
SquareDiff
,
(
x
,
y
),
{
0
:
0
})
.
out
f
=
gof
.
CLinker
(
env
([
x
,
y
],
[
e
]))
.
make_function
(
inplace
=
False
)
xv
=
numpy
.
random
.
rand
(
1000
,
1000
)
yv
=
numpy
.
random
.
rand
(
1000
,
1000
)
zv
=
numpy
.
random
.
rand
(
1000
,
1000
)
add
=
numpy
.
frompyfunc
(
lambda
x
,
y
:
x
+
y
,
2
,
1
)
t0
=
time
.
time
()
for
i
in
xrange
(
100
):
xv
-=
yv
xv
*=
xv
# xv += yv
print
time
.
time
()
-
t0
t0
=
time
.
time
()
for
i
in
xrange
(
100
):
f
(
xv
,
yv
)
print
time
.
time
()
-
t0
# def test_straightforward(self):
# x, y, z = inputs()
# e0 = CAReduce(Add, [x]).out
# # print e0.owner
# f = gof.PerformLinker(env([x], [e0])).make_function(inplace=True)
# assert f(numpy.ones((2, 2))) == 4.0
##########
##########
# def test_straightforward(self):
# x, y, z = inputs()
# e0 = Broadcast(Add, (x, y)).out
# f = gof.PerformLinker(env([x, y], [e0])).make_function(inplace=True)
# assert (f(numpy.ones((2, 2)), numpy.ones((1, 2))) == numpy.ones((2, 2))*2).all()
# # for result in e0.owner.grad(e0.owner.inputs, (z, )):
# # print env([x, y, z], [result])
# def test_c(self):
# x = modes.build(Tensor('float64', (0, 0), name = 'x'))
# y = modes.build(Tensor('float64', (0, 1), name = 'y'))
# z = modes.build(Tensor('float64', (0, 0), name = 'z'))
# # x = modes.build(Tensor('float64', (), name = 'x'))
# # y = modes.build(Tensor('float64', (), name = 'y'))
# # x, y, z = inputs()
# e0 = Broadcast(Add, (x, y)).out
# f = gof.CLinker(env([x, y], [e0])).make_function(inplace=True)
# print f(numpy.ones((4, 4), order = 'f'), numpy.array([[1], [2], [3], [4]]))
# # print f(numpy.ones(()), numpy.ones(()))
# assert (f(numpy.ones((2, 2)), numpy.ones((2, 1))) == numpy.ones((2, 2))*2).all()
elemwise_cgen.py
0 → 100644
浏览文件 @
761c7f9f
def
make_declare
(
loop_orders
,
dtypes
,
sub
):
decl
=
""
for
i
,
(
loop_order
,
dtype
)
in
enumerate
(
zip
(
loop_orders
,
dtypes
)):
var
=
sub
[
'lv
%
i'
%
i
]
decl
+=
"""
%(dtype)
s*
%(var)
s_iter;
int
%(var)
s_nd;
"""
%
locals
()
for
j
,
value
in
enumerate
(
loop_order
):
if
value
!=
'x'
:
decl
+=
"""
int
%(var)
s_n
%(value)
i;
int
%(var)
s_stride
%(value)
i;
int
%(var)
s_jump
%(value)
i_
%(j)
i;
"""
%
locals
()
else
:
decl
+=
"""
int
%(var)
s_jump
%(value)
s_
%(j)
i;
"""
%
locals
()
return
decl
def
make_checks
(
loop_orders
,
dtypes
,
sub
):
init
=
""
for
i
,
(
loop_order
,
dtype
)
in
enumerate
(
zip
(
loop_orders
,
dtypes
)):
var
=
"
%%
(lv
%
i)s"
%
i
nonx
=
[
x
for
x
in
loop_order
if
x
!=
'x'
]
if
nonx
:
min_nd
=
max
(
nonx
)
init
+=
"""
if (
%(var)
s->nd <
%(min_nd)
s) {
PyErr_SetString(PyExc_ValueError, "Not enough dimensions on input.");
%%(fail)
s
}
"""
%
locals
()
adjust
=
[]
for
j
,
index
in
reversed
([
aaa
for
aaa
in
enumerate
(
loop_order
)]):
if
index
!=
'x'
:
jump
=
" - "
.
join
([
"
%(var)
s_stride
%(index)
s"
%
locals
()]
+
adjust
)
init
+=
"""
%(var)
s_n
%(index)
s =
%(var)
s->dimensions[
%(index)
s];
%(var)
s_stride
%(index)
s =
%(var)
s->strides[
%(index)
s] / sizeof(
%(dtype)
s);
%(var)
s_jump
%(index)
s_
%(j)
s =
%(jump)
s;
//printf("
%(var)
s_jump
%(index)
s_
%(j)
s is:");
//std::cout <<
%(var)
s_jump
%(index)
s_
%(j)
s << std::endl;
"""
%
locals
()
adjust
=
[
"
%(var)
s_n
%(index)
s*
%(var)
s_stride
%(index)
s"
%
locals
()]
else
:
jump
=
" - "
.
join
([
"0"
]
+
adjust
)
init
+=
"""
%(var)
s_jump
%(index)
s_
%(j)
s =
%(jump)
s;
//printf("
%(var)
s_jump
%(index)
s_
%(j)
s is:");
//std::cout <<
%(var)
s_jump
%(index)
s_
%(j)
s << std::endl;
"""
%
locals
()
adjust
=
[]
check
=
""
for
matches
in
zip
(
*
loop_orders
):
to_compare
=
[(
j
,
x
)
for
j
,
x
in
enumerate
(
matches
)
if
x
!=
"x"
]
if
len
(
to_compare
)
<
2
:
continue
j
,
x
=
to_compare
[
0
]
first
=
"
%%
(lv
%(j)
s)s_n
%(x)
s"
%
locals
()
cond
=
" || "
.
join
([
"
%(first)
s !=
%%
(lv
%(j)
s)s_n
%(x)
s"
%
locals
()
for
j
,
x
in
to_compare
[
1
:]])
if
cond
:
check
+=
"""
if (
%(cond)
s) {
PyErr_SetString(PyExc_ValueError, "Input dimensions do not match.");
%%(fail)
s
}
"""
%
locals
()
return
init
%
sub
+
check
%
sub
def
make_alloc
(
loop_orders
,
dtype
,
sub
):
nd
=
len
(
loop_orders
[
0
])
init_dims
=
""
for
i
,
candidates
in
enumerate
(
zip
(
*
loop_orders
)):
for
j
,
candidate
in
enumerate
(
candidates
):
if
candidate
!=
'x'
:
var
=
sub
[
'lv
%
i'
%
j
]
init_dims
+=
"dims[
%(i)
s] =
%(var)
s_n
%(candidate)
s;
\n
"
%
locals
()
break
else
:
init_dims
+=
"dims[
%(i)
s] = 1;
\n
"
%
locals
()
#raise Exception("For each looping dimension, at least one input must have a non-broadcastable dimension.")
return
"""
{
npy_intp dims[
%(nd)
s];
//npy_intp* dims = (npy_intp*)malloc(
%(nd)
s * sizeof(npy_intp));
%(init_dims)
s
if (!
%(olv)
s) {
%(olv)
s = (PyArrayObject*)PyArray_EMPTY(
%(nd)
s, dims, type_num_
%(olv)
s, 0);
}
else {
PyArray_Dims new_dims;
new_dims.len =
%(nd)
s;
new_dims.ptr = dims;
PyObject* success = PyArray_Resize(
%(olv)
s, &new_dims, 0, PyArray_CORDER);
if (!success) {
// If we can't resize the ndarray we have we can allocate a new one.
PyErr_Clear();
Py_XDECREF(
%(olv)
s);
%(olv)
s = (PyArrayObject*)PyArray_EMPTY(
%(nd)
s, dims, type_num_
%(olv)
s, 0);
}
}
if (!
%(olv)
s) {
%(fail)
s
}
}
"""
%
dict
(
locals
(),
**
sub
)
def
make_loop
(
loop_orders
,
dtypes
,
loop_tasks
,
sub
):
"""
Make a nested loop over several arrays and associate specific code
to each level of nesting.
loop_orders: list of N tuples of length M. Each value of each
tuple can be either the index of a dimension to loop over or
the letter 'x' which means there is no looping to be done
over that variable at that point (in other words we broadcast
over that dimension). If an entry is an integer, it will become
an alias of the entry of that rank.
loop_tasks: list of M+1 pieces of code. The ith loop_task is code
to be executed just before going to the next element of the
ith dimension. The last is code to be executed at the very end.
sub: a dictionary that maps 'lv#' to a suitable variable name.
The 'lvi' variable corresponds to the ith element of loop_orders.
"""
def
loop_over
(
preloop
,
code
,
indices
,
i
):
iterv
=
'ITER_
%
i'
%
i
update
=
""
suitable_n
=
"1"
for
j
,
index
in
enumerate
(
indices
):
var
=
sub
[
'lv
%
i'
%
j
]
update
+=
"
%(var)
s_iter +=
%(var)
s_jump
%(index)
s_
%(i)
s;
\n
"
%
locals
()
if
index
!=
'x'
:
suitable_n
=
"
%(var)
s_n
%(index)
s"
%
locals
()
return
"""
%(preloop)
s
for (int
%(iterv)
s =
%(suitable_n)
s;
%(iterv)
s;
%(iterv)
s--) {
%(code)
s
%(update)
s
}
"""
%
locals
()
preloops
=
{}
for
i
,
(
loop_order
,
dtype
)
in
enumerate
(
zip
(
loop_orders
,
dtypes
)):
for
j
,
index
in
enumerate
(
loop_order
):
if
index
!=
'x'
:
preloops
.
setdefault
(
j
,
""
)
preloops
[
j
]
+=
(
"
%%
(lv
%(i)
s)s_iter = (
%(dtype)
s*)(
%%
(lv
%(i)
s)s->data);
\n
"
%
locals
())
%
sub
break
else
:
# all broadcastable
preloops
.
setdefault
(
0
,
""
)
preloops
[
0
]
+=
(
"
%%
(lv
%(i)
s)s_iter = (
%(dtype)
s*)(
%%
(lv
%(i)
s)s->data);
\n
"
%
locals
())
%
sub
if
len
(
loop_tasks
)
==
1
:
s
=
preloops
.
get
(
0
,
""
)
else
:
s
=
""
for
i
,
(
pre_task
,
task
),
indices
in
reversed
(
zip
(
xrange
(
len
(
loop_tasks
)
-
1
),
loop_tasks
,
zip
(
*
loop_orders
))):
s
=
loop_over
(
preloops
.
get
(
i
,
""
)
+
pre_task
,
s
+
task
,
indices
,
i
)
s
+=
loop_tasks
[
-
1
]
return
"{
%
s}"
%
s
# print make_declare(((0, 1, 2, 3), ('x', 1, 0, 3), ('x', 'x', 'x', 0)),
# ('double', 'int', 'float'),
# dict(lv0='x', lv1='y', lv2='z', fail="FAIL;"))
# print make_checks(((0, 1, 2, 3), ('x', 1, 0, 3), ('x', 'x', 'x', 0)),
# ('double', 'int', 'float'),
# dict(lv0='x', lv1='y', lv2='z', fail="FAIL;"))
# print make_alloc(((0, 1, 2, 3), ('x', 1, 0, 3), ('x', 'x', 'x', 0)),
# 'double',
# dict(olv='out', lv0='x', lv1='y', lv2='z', fail="FAIL;"))
# print make_loop(((0, 1, 2, 3), ('x', 1, 0, 3), ('x', 'x', 'x', 0)),
# ('double', 'int', 'float'),
# (("C00;", "C%01;"), ("C10;", "C11;"), ("C20;", "C21;"), ("C30;", "C31;"),"C4;"),
# dict(lv0='x', lv1='y', lv2='z', fail="FAIL;"))
# print make_loop(((0, 1, 2, 3), (3, 'x', 0, 'x'), (0, 'x', 'x', 'x')),
# ('double', 'int', 'float'),
# (("C00;", "C01;"), ("C10;", "C11;"), ("C20;", "C21;"), ("C30;", "C31;"),"C4;"),
# dict(lv0='x', lv1='y', lv2='z', fail="FAIL;"))
##################
### DimShuffle ###
##################
#################
### Broadcast ###
#################
################
### CAReduce ###
################
编写
预览
Markdown
格式
0%
重试
或
添加新文件
添加附件
取消
您添加了
0
人
到此讨论。请谨慎行事。
请先完成此评论的编辑!
取消
请
注册
或者
登录
后发表评论