直接在Scipy稀疏矩阵上使用Intel mkl库来计算内存较少的A点A.T



我想从python调用mkl.mkl_scsrmultcsr。目标是计算压缩稀疏行格式的稀疏矩阵C。稀疏矩阵C是A和A的转置之间的矩阵乘积,其中A也是csr格式的稀疏矩阵。当用scipy计算C=A点(A.T)时,scipy似乎(?)分配了新的内存来保存A的转置,并且肯定为新的C矩阵分配了内存(这意味着我不能使用现有的C矩阵)。因此,我想尝试直接使用mkl c函数来减少内存使用。

以下是适用于另一个mkl函数的答案。在这个答案中,mkl函数的速度提高了4倍。

经过4天的工作,下面的版本终于可以工作了。我想知道我是否浪费了任何记忆。ctype会复制numpy数组吗?那就是测试函数中csr->csc转换的必要性吗?intel c函数可以计算(A.T)点A,或A点A,但不能计算A点(A.T)

再次感谢。

from ctypes import *
import scipy.sparse as spsp
import numpy as np
import multiprocessing as mp
# June 2nd 2016 version.
# Load the share library
mkl = cdll.LoadLibrary("libmkl_rt.so")
def get_csr_handle(A,clear=False):
if clear == True:
A.indptr[:] = 0
A.indices[:] = 0
A.data[:] = 0
a_pointer   = A.data.ctypes.data_as(POINTER(c_float))
# Array containing non-zero elements of the matrix A.
# This corresponds to data array of csr_matrix
# Its length is equal to #non zero elements in A
# (Can this be longer than actual #non-zero elements?)
assert A.data.ctypes.data % 16 == 0 # Check alignment
ja_pointer  = A.indices.ctypes.data_as(POINTER(c_int))
# Array of column indices of all non-zero elements of A.
# This corresponds to the indices array of csr_matrix
assert A.indices.ctypes.data % 16 == 0 # Check alignment
ia_pointer  = A.indptr.ctypes.data_as(POINTER(c_int))
# Array of length m+1.
# a[ia[i]:ia[i+1]] is the value of nonzero entries of
# the ith row of A.
# ja[ia[i]:ia[i+1]] is the column indices of nonzero
# entries of the ith row of A
# This corresponds to the indptr array of csr_matrix
assert A.indptr.ctypes.data % 16 == 0 # Check alignment
A_data_size    = A.data.size
A_indices_size = A.indices.size
A_indptr_size  = A.indptr.size
return (a_pointer, ja_pointer, ia_pointer, A)
def csr_dot_csr_t(A_handle, C_handle, nz=None):
# Calculate (A.T).dot(A) and put result into C
#
# This uses one-based indexing
#
# Both C.data and A.data must be in np.float32 type.
#
# Number of nonzero elements in C must be greater than
#     or equal to the size of C.data
#
# size of C.indptr must be greater than or equal to
#     1 + (num rows of A).
#
# C_data    = np.zeros((nz), dtype=np.single)
# C_indices = np.zeros((nz), dtype=np.int32)
# C_indptr  = np.zeros((m+1),dtype=np.int32)
#assert len(c_pointer._obj) >= 1 + A_shape[0]

(a_pointer, ja_pointer, ia_pointer, A) = A_handle
(c_pointer, jc_pointer, ic_pointer, C) = C_handle
#print "CCC",C
#assert type(C.data[0]) == np.float32
#assert type(A.data[0]) == np.float32
#assert C.indptr.size >= A.shape[0] + 1
#CC = A.dot(A.T)
#assert C.data.size >= nz
#assert C.indices.size >= nz
trans_pointer   = byref(c_char('T'))
sort_pointer    = byref(c_int(0))
(m, n)          = A.shape
sort_pointer        = byref(c_int(0))
m_pointer           = byref(c_int(m))     # Number of rows of matrix A
n_pointer           = byref(c_int(n))     # Number of columns of matrix A
k_pointer           = byref(c_int(n))     # Number of columns of matrix B
# should be n when trans='T'
# Otherwise, I guess should be m
###
b_pointer   = a_pointer
jb_pointer  = ja_pointer
ib_pointer  = ia_pointer
###
if nz == None:
nz = n*n #*n # m*m # Number of nonzero elements expected
# probably can use lower value for sparse
# matrices.
nzmax_pointer   = byref(c_int(nz))
# length of arrays c and jc. (which are data and
# indices of csr_matrix). So this is the number of
# nonzero elements of matrix C
#
# This parameter is used only if request=0.
# The routine stops calculation if the number of
# elements in the result matrix C exceeds the
# specified value of nzmax.
info = c_int(-3)
info_pointer = byref(info)
request_pointer_list = [byref(c_int(0)), byref(c_int(1)), byref(c_int(2))]
return_list = []
for ii in [0]:
request_pointer = request_pointer_list[ii]
ret = mkl.mkl_scsrmultcsr(trans_pointer, request_pointer, sort_pointer,
m_pointer, n_pointer, k_pointer,
a_pointer, ja_pointer, ia_pointer,
b_pointer, jb_pointer, ib_pointer,
c_pointer, jc_pointer, ic_pointer,
nzmax_pointer, info_pointer)
info_val = info.value
return_list += [ (ret,info_val) ]
return return_list

def show_csr_internal(A, indent=4):
# Print data, indptr, and indices
# of a scipy csr_matrix A
name = ['data', 'indptr', 'indices']
mat  = [A.data, A.indptr, A.indices]
for i in range(3):
str_print = ' '*indent+name[i]+':n%s'%mat[i]
str_print = str_print.replace('n', 'n'+' '*indent*2)
print(str_print)
def fix_for_scipy(C,A):
n = A.shape[1]
print "fix n", n
nz =  C.indptr[n] - 1 # -1 as this is still one based indexing.
print "fix nz", nz
data = C.data[:nz]
C.indptr[:n+1] -= 1
indptr = C.indptr[:n+1]
C.indices[:nz] -= 1
indices = C.indices[:nz]
return spsp.csr_matrix( (data, indices, indptr), shape=(n,n))
def test():
AA= [[1,0,0,1],
[1,0,1,0],
[0,0,1,0]]
AA = np.random.choice([0,1], size=(3,750000), replace=True, p=[0.99,0.01])
A_original = spsp.csr_matrix(AA)
#A = spsp.csr_matrix(A_original, dtype=np.float32)
A = A_original.astype(np.float32).tocsc()
#A_original = A.todense()
A = spsp.csr_matrix( (A.data, A.indices, A.indptr) )
print  "A:"
show_csr_internal(A)
print A.todense()
A.indptr  += 1 # convert to 1-based indexing
A.indices += 1 # convert to 1-based indexing
A_ptrs = get_csr_handle(A)
C = spsp.csr_matrix( np.ones((3,3)), dtype=np.float32)
#C.data = C.data[:16].view()
#C.indptr = C.indptr
C_ptrs = get_csr_handle(C, clear=True)
print "C:"
show_csr_internal(C)    
print "=call mkl function=" 
return_list= csr_dot_csr_t(A_ptrs, C_ptrs)
print "(ret, info):", return_list
print "C after calling mkl:"
show_csr_internal(C)    
C_fix = fix_for_scipy(C,A)
print "C_fix for scipy:"
show_csr_internal(C_fix)
print C_fix.todense()
print "Original C after fixing:"
show_csr_internal(C)
print "scipy's (A).dot(A.T)"
scipy_ans = (A_original).dot(A_original.T)
#scipy_ans = spsp.csr_matrix(scipy_ans)
show_csr_internal(scipy_ans)
print scipy_ans.todense()
if __name__ == "__main__":
test()

结果:

A:
data:
[ 1.  1.  1. ...,  1.  1.  1.]
indptr:
[    0     0     0 ..., 22673 22673 22673]
indices:
[1 0 2 ..., 2 1 2]
[[ 0.  0.  0.]
[ 0.  0.  0.]
[ 0.  0.  0.]
..., 
[ 0.  0.  0.]
[ 0.  0.  0.]
[ 0.  0.  0.]]
C:
data:
[ 0.  0.  0.  0.  0.  0.  0.  0.  0.]
indptr:
[0 0 0 0]
indices:
[0 0 0 0 0 0 0 0 0]
=call mkl function=
(ret, info): [(2, 0)]
C after calling mkl:
data:
[ 7576.    77.    83.    77.  7607.   104.    83.   104.  7490.]
indptr:
[ 1  4  7 10]
indices:
[1 2 3 1 2 3 1 2 3]
fix n 3
fix nz 9
C_fix for scipy:
data:
[ 7576.    77.    83.    77.  7607.   104.    83.   104.  7490.]
indptr:
[0 3 6 9]
indices:
[0 1 2 0 1 2 0 1 2]
[[ 7576.    77.    83.]
[   77.  7607.   104.]
[   83.   104.  7490.]]
Original C after fixing:
data:
[ 7576.    77.    83.    77.  7607.   104.    83.   104.  7490.]
indptr:
[0 3 6 9]
indices:
[0 1 2 0 1 2 0 1 2]
scipy's (A.T).dot(A)
data:
[  83   77 7576  104   77 7607   83  104 7490]
indptr:
[0 3 6 9]
indices:
[2 1 0 2 0 1 0 1 2]
[[7576   77   83]
[  77 7607  104]
[  83  104 7490]]

学到的东西:

矩阵A、B和C的
  1. 索引从1开始
  2. 我在原始代码中混淆了c_indptr和c_indexes。应该是ia=scipy csr_matrix的indptr。ja=scipy csr_matrix的索引
  3. 从这里的代码。所有内容都传递给mkl_?csrmultcsr作为指针。mkl_scsrmultcsr(&ta, &r[1], &sort, &m, &m, &m, a, ja, ia, a, ja, ia, c, jc, ic, &nzmax, &info);

  4. 我想要一个可以使用零基索引的mkl函数。mkl.mkl_scsrmultcsr函数只能使用基于一个索引。(或者我可以对所有内容进行基于一的索引。这意味着在大多数线性代数步骤中使用intel c函数而不是scipy/numpy。)

查看scipy稀疏产品的Python代码。请注意,它在2次传递中调用编译后的代码。

看起来mkl代码做了同样的事情

https://software.intel.com/en-us/node/468640

如果request=1,例程只计算长度为m+1的数组ic的值,则必须预先分配该数组的内存。退出时,值ic(m+1)-1是数组c和jc中元素的实际数量。

如果request=2,则先前已使用参数request=1调用例程,则在调用程序中分配输出数组jc和c,并且它们的长度至少为ic(m+1)-1。

首先根据C的行数分配ic(您从输入中知道这一点),然后用request=1调用mkl代码。

对于request=2,必须根据ic(m+1) - 1中的大小分配cjc阵列。这与输入阵列中nnz的数量不同。

您使用的是request1 = c_int(0),它要求c阵列的大小正确——如果不实际执行dot(或request 1),您就不知道这一点。

===============

File:        /usr/lib/python3/dist-packages/scipy/sparse/compressed.py
Definition:  A._mul_sparse_matrix(self, other)

步骤1分配indptr(纸币大小),并传递指针(此时数据无关紧要)

indptr = np.empty(major_axis + 1, dtype=idx_dtype)
fn = getattr(_sparsetools, self.format + '_matmat_pass1')
fn(M, N,
np.asarray(self.indptr, dtype=idx_dtype),
np.asarray(self.indices, dtype=idx_dtype),
np.asarray(other.indptr, dtype=idx_dtype),
np.asarray(other.indices, dtype=idx_dtype),
indptr)
nnz = indptr[-1]

步骤2分配indptr(不同大小),并基于nnz分配indicesdata

indptr = np.asarray(indptr, dtype=idx_dtype)
indices = np.empty(nnz, dtype=idx_dtype)
data = np.empty(nnz, dtype=upcast(self.dtype, other.dtype))
fn = getattr(_sparsetools, self.format + '_matmat_pass2')
fn(M, N, np.asarray(self.indptr, dtype=idx_dtype),
np.asarray(self.indices, dtype=idx_dtype),
self.data,
np.asarray(other.indptr, dtype=idx_dtype),
np.asarray(other.indices, dtype=idx_dtype),
other.data,
indptr, indices, data)

最后使用这些数组创建一个新数组。

return self.__class__((data,indices,indptr),shape=(M,N))

mkl库应该以相同的方式使用。

===================

https://github.com/scipy/scipy/blob/master/scipy/sparse/sparsetools/csr.h

具有用于csr_matmat_pass1csr_matmat_pass2的c代码

====================

如果有帮助的话,下面是这些过程的纯Python实现。一种不试图利用任何数组操作的字面翻译。

def pass1(A, B):
nrow,ncol=A.shape
Aptr=A.indptr
Bptr=B.indptr
Cp=np.zeros(nrow+1,int)
mask=np.zeros(ncol,int)-1
nnz=0
for i in range(nrow):
row_nnz=0
for jj in range(Aptr[i],Aptr[i+1]):
j=A.indices[jj]
for kk in range(Bptr[j],Bptr[j+1]):
k=B.indices[kk]
if mask[k]!=i:
mask[k]=i
row_nnz += 1
nnz += row_nnz
Cp[i+1]=nnz
return Cp

def pass2(A,B,Cnnz):
nrow,ncol=A.shape
Ap,Aj,Ax=A.indptr,A.indices,A.data
Bp,Bj,Bx=B.indptr,B.indices,B.data
next = np.zeros(ncol,int)-1
sums = np.zeros(ncol,A.dtype)

Cp=np.zeros(nrow+1,int)
Cj=np.zeros(Cnnz,int)
Cx=np.zeros(Cnnz,A.dtype)
nnz = 0
for i in range(nrow):
head = -2
length = 0
for jj in range(Ap[i], Ap[i+1]):
j, v = Aj[jj], Ax[jj]
for kk in range(Bp[j], Bp[j+1]):
k = Bj[kk]
sums[k] += v*Bx[kk]
if (next[k]==-1):
next[k], head=head, k
length += 1
print(i,sums, next)
for _ in range(length):
if sums[head] !=0:
Cj[nnz], Cx[nnz] = head, sums[head]
nnz += 1
temp = head
head = next[head]
next[temp], sums[temp] = -1, 0
Cp[i+1] = nnz            
return Cp, Cj, Cx                  

def pass0(A,B):
Cp = pass1(A,B)
nnz = Cp[-1]
Cp,Cj,Cx=pass2(A,B,nnz)
N,M=A.shape[0], B.shape[1]
C=sparse.csr_matrix((Cx, Cj, Cp), shape=(N,M))
return C

CCD_ 22和CCD_ 23都必须是CCD_。它使用转置,需要转换,例如B = A.T.tocsr()

最新更新