2019-05-23 12:14:39 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2009-07-14 23:20:37 +04:00
/*
* Asynchronous RAID - 6 recovery calculations ASYNC_TX API .
* Copyright ( c ) 2009 Intel Corporation
*
* based on raid6recov . c :
* Copyright 2002 H . Peter Anvin
*/
# include <linux/kernel.h>
# include <linux/interrupt.h>
2011-05-27 22:41:48 +04:00
# include <linux/module.h>
2009-07-14 23:20:37 +04:00
# include <linux/dma-mapping.h>
# include <linux/raid/pq.h>
# include <linux/async_tx.h>
2013-10-18 21:35:28 +04:00
# include <linux/dmaengine.h>
2009-07-14 23:20:37 +04:00
static struct dma_async_tx_descriptor *
2020-08-20 16:22:11 +03:00
async_sum_product ( struct page * dest , unsigned int d_off ,
struct page * * srcs , unsigned int * src_offs , unsigned char * coef ,
size_t len , struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_chan * chan = async_tx_find_channel ( submit , DMA_PQ ,
& dest , 1 , srcs , 2 , len ) ;
struct dma_device * dma = chan ? chan - > device : NULL ;
2013-10-18 21:35:28 +04:00
struct dmaengine_unmap_data * unmap = NULL ;
2009-07-14 23:20:37 +04:00
const u8 * amul , * bmul ;
u8 ax , bx ;
u8 * a , * b , * c ;
2013-10-18 21:35:28 +04:00
if ( dma )
2016-01-07 03:02:34 +03:00
unmap = dmaengine_get_unmap_data ( dma - > dev , 3 , GFP_NOWAIT ) ;
2013-10-18 21:35:28 +04:00
if ( unmap ) {
2009-07-14 23:20:37 +04:00
struct device * dev = dma - > dev ;
2013-10-18 21:35:28 +04:00
dma_addr_t pq [ 2 ] ;
2009-07-14 23:20:37 +04:00
struct dma_async_tx_descriptor * tx ;
2013-10-18 21:35:33 +04:00
enum dma_ctrl_flags dma_flags = DMA_PREP_PQ_DISABLE_P ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
if ( submit - > flags & ASYNC_TX_FENCE )
dma_flags | = DMA_PREP_FENCE ;
2020-08-20 16:22:11 +03:00
unmap - > addr [ 0 ] = dma_map_page ( dev , srcs [ 0 ] , src_offs [ 0 ] ,
len , DMA_TO_DEVICE ) ;
unmap - > addr [ 1 ] = dma_map_page ( dev , srcs [ 1 ] , src_offs [ 1 ] ,
len , DMA_TO_DEVICE ) ;
2013-10-18 21:35:28 +04:00
unmap - > to_cnt = 2 ;
2020-08-20 16:22:11 +03:00
unmap - > addr [ 2 ] = dma_map_page ( dev , dest , d_off ,
len , DMA_BIDIRECTIONAL ) ;
2013-10-18 21:35:28 +04:00
unmap - > bidi_cnt = 1 ;
/* engine only looks at Q, but expects it to follow P */
pq [ 1 ] = unmap - > addr [ 2 ] ;
unmap - > len = len ;
tx = dma - > device_prep_dma_pq ( chan , pq , unmap - > addr , 2 , coef ,
2009-07-14 23:20:37 +04:00
len , dma_flags ) ;
if ( tx ) {
2013-10-18 21:35:28 +04:00
dma_set_unmap ( tx , unmap ) ;
2009-07-14 23:20:37 +04:00
async_tx_submit ( chan , tx , submit ) ;
2013-10-18 21:35:28 +04:00
dmaengine_unmap_put ( unmap ) ;
2009-07-14 23:20:37 +04:00
return tx ;
}
2009-09-21 21:47:40 +04:00
/* could not get a descriptor, unmap and fall through to
* the synchronous path
*/
2013-10-18 21:35:28 +04:00
dmaengine_unmap_put ( unmap ) ;
2009-07-14 23:20:37 +04:00
}
/* run the operation synchronously */
async_tx_quiesce ( & submit - > depend_tx ) ;
amul = raid6_gfmul [ coef [ 0 ] ] ;
bmul = raid6_gfmul [ coef [ 1 ] ] ;
2020-08-20 16:22:11 +03:00
a = page_address ( srcs [ 0 ] ) + src_offs [ 0 ] ;
b = page_address ( srcs [ 1 ] ) + src_offs [ 1 ] ;
c = page_address ( dest ) + d_off ;
2009-07-14 23:20:37 +04:00
while ( len - - ) {
ax = amul [ * a + + ] ;
bx = bmul [ * b + + ] ;
* c + + = ax ^ bx ;
}
return NULL ;
}
static struct dma_async_tx_descriptor *
2020-08-20 16:22:11 +03:00
async_mult ( struct page * dest , unsigned int d_off , struct page * src ,
unsigned int s_off , u8 coef , size_t len ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_chan * chan = async_tx_find_channel ( submit , DMA_PQ ,
& dest , 1 , & src , 1 , len ) ;
struct dma_device * dma = chan ? chan - > device : NULL ;
2013-10-18 21:35:28 +04:00
struct dmaengine_unmap_data * unmap = NULL ;
2009-07-14 23:20:37 +04:00
const u8 * qmul ; /* Q multiplier table */
u8 * d , * s ;
2013-10-18 21:35:28 +04:00
if ( dma )
2016-01-07 03:02:34 +03:00
unmap = dmaengine_get_unmap_data ( dma - > dev , 3 , GFP_NOWAIT ) ;
2013-10-18 21:35:28 +04:00
if ( unmap ) {
2009-07-14 23:20:37 +04:00
dma_addr_t dma_dest [ 2 ] ;
struct device * dev = dma - > dev ;
struct dma_async_tx_descriptor * tx ;
2013-10-18 21:35:33 +04:00
enum dma_ctrl_flags dma_flags = DMA_PREP_PQ_DISABLE_P ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
if ( submit - > flags & ASYNC_TX_FENCE )
dma_flags | = DMA_PREP_FENCE ;
2020-08-20 16:22:11 +03:00
unmap - > addr [ 0 ] = dma_map_page ( dev , src , s_off ,
len , DMA_TO_DEVICE ) ;
2013-10-18 21:35:28 +04:00
unmap - > to_cnt + + ;
2020-08-20 16:22:11 +03:00
unmap - > addr [ 1 ] = dma_map_page ( dev , dest , d_off ,
len , DMA_BIDIRECTIONAL ) ;
2013-10-18 21:35:28 +04:00
dma_dest [ 1 ] = unmap - > addr [ 1 ] ;
unmap - > bidi_cnt + + ;
unmap - > len = len ;
/* this looks funny, but the engine looks for Q at
* dma_dest [ 1 ] and ignores dma_dest [ 0 ] as a dest
* due to DMA_PREP_PQ_DISABLE_P
*/
tx = dma - > device_prep_dma_pq ( chan , dma_dest , unmap - > addr ,
1 , & coef , len , dma_flags ) ;
2009-07-14 23:20:37 +04:00
if ( tx ) {
2013-10-18 21:35:28 +04:00
dma_set_unmap ( tx , unmap ) ;
dmaengine_unmap_put ( unmap ) ;
2009-07-14 23:20:37 +04:00
async_tx_submit ( chan , tx , submit ) ;
return tx ;
}
2009-09-21 21:47:40 +04:00
/* could not get a descriptor, unmap and fall through to
* the synchronous path
*/
2013-10-18 21:35:28 +04:00
dmaengine_unmap_put ( unmap ) ;
2009-07-14 23:20:37 +04:00
}
/* no channel available, or failed to allocate a descriptor, so
* perform the operation synchronously
*/
async_tx_quiesce ( & submit - > depend_tx ) ;
qmul = raid6_gfmul [ coef ] ;
2020-08-20 16:22:11 +03:00
d = page_address ( dest ) + d_off ;
s = page_address ( src ) + s_off ;
2009-07-14 23:20:37 +04:00
while ( len - - )
* d + + = qmul [ * s + + ] ;
return NULL ;
}
static struct dma_async_tx_descriptor *
2009-10-20 01:05:12 +04:00
__2data_recov_4 ( int disks , size_t bytes , int faila , int failb ,
2020-08-20 16:22:11 +03:00
struct page * * blocks , unsigned int * offs ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_async_tx_descriptor * tx = NULL ;
struct page * p , * q , * a , * b ;
2020-08-20 16:22:11 +03:00
unsigned int p_off , q_off , a_off , b_off ;
2009-07-14 23:20:37 +04:00
struct page * srcs [ 2 ] ;
2020-08-20 16:22:11 +03:00
unsigned int src_offs [ 2 ] ;
2009-07-14 23:20:37 +04:00
unsigned char coef [ 2 ] ;
enum async_tx_flags flags = submit - > flags ;
dma_async_tx_callback cb_fn = submit - > cb_fn ;
void * cb_param = submit - > cb_param ;
void * scribble = submit - > scribble ;
2009-10-20 01:05:12 +04:00
p = blocks [ disks - 2 ] ;
2020-08-20 16:22:11 +03:00
p_off = offs [ disks - 2 ] ;
2009-10-20 01:05:12 +04:00
q = blocks [ disks - 1 ] ;
2020-08-20 16:22:11 +03:00
q_off = offs [ disks - 1 ] ;
2009-07-14 23:20:37 +04:00
a = blocks [ faila ] ;
2020-08-20 16:22:11 +03:00
a_off = offs [ faila ] ;
2009-07-14 23:20:37 +04:00
b = blocks [ failb ] ;
2020-08-20 16:22:11 +03:00
b_off = offs [ failb ] ;
2009-07-14 23:20:37 +04:00
/* in the 4 disk case P + Pxy == P and Q + Qxy == Q */
/* Dx = A*(P+Pxy) + B*(Q+Qxy) */
srcs [ 0 ] = p ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = p_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = q ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = q_off ;
2009-07-14 23:20:37 +04:00
coef [ 0 ] = raid6_gfexi [ failb - faila ] ;
coef [ 1 ] = raid6_gfinv [ raid6_gfexp [ faila ] ^ raid6_gfexp [ failb ] ] ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_sum_product ( b , b_off , srcs , src_offs , coef , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Dy = P+Pxy+Dx */
srcs [ 0 ] = p ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = p_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = b ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = b_off ;
2009-07-14 23:20:37 +04:00
init_async_submit ( submit , flags | ASYNC_TX_XOR_ZERO_DST , tx , cb_fn ,
cb_param , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( a , a_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
return tx ;
}
static struct dma_async_tx_descriptor *
2009-10-20 01:05:12 +04:00
__2data_recov_5 ( int disks , size_t bytes , int faila , int failb ,
2020-08-20 16:22:11 +03:00
struct page * * blocks , unsigned int * offs ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_async_tx_descriptor * tx = NULL ;
struct page * p , * q , * g , * dp , * dq ;
2020-08-20 16:22:11 +03:00
unsigned int p_off , q_off , g_off , dp_off , dq_off ;
2009-07-14 23:20:37 +04:00
struct page * srcs [ 2 ] ;
2020-08-20 16:22:11 +03:00
unsigned int src_offs [ 2 ] ;
2009-07-14 23:20:37 +04:00
unsigned char coef [ 2 ] ;
enum async_tx_flags flags = submit - > flags ;
dma_async_tx_callback cb_fn = submit - > cb_fn ;
void * cb_param = submit - > cb_param ;
void * scribble = submit - > scribble ;
2009-10-20 01:05:12 +04:00
int good_srcs , good , i ;
2009-07-14 23:20:37 +04:00
2009-10-20 01:05:12 +04:00
good_srcs = 0 ;
good = - 1 ;
for ( i = 0 ; i < disks - 2 ; i + + ) {
if ( blocks [ i ] = = NULL )
continue ;
2009-07-14 23:20:37 +04:00
if ( i = = faila | | i = = failb )
continue ;
2009-10-20 01:05:12 +04:00
good = i ;
good_srcs + + ;
2009-07-14 23:20:37 +04:00
}
2009-10-20 01:05:12 +04:00
BUG_ON ( good_srcs > 1 ) ;
2009-07-14 23:20:37 +04:00
2009-10-20 01:05:12 +04:00
p = blocks [ disks - 2 ] ;
2020-08-20 16:22:11 +03:00
p_off = offs [ disks - 2 ] ;
2009-10-20 01:05:12 +04:00
q = blocks [ disks - 1 ] ;
2020-08-20 16:22:11 +03:00
q_off = offs [ disks - 1 ] ;
2009-07-14 23:20:37 +04:00
g = blocks [ good ] ;
2020-08-20 16:22:11 +03:00
g_off = offs [ good ] ;
2009-07-14 23:20:37 +04:00
/* Compute syndrome with zero for the missing data pages
* Use the dead data pages as temporary storage for delta p and
* delta q
*/
dp = blocks [ faila ] ;
2020-08-20 16:22:11 +03:00
dp_off = offs [ faila ] ;
2009-07-14 23:20:37 +04:00
dq = blocks [ failb ] ;
2020-08-20 16:22:11 +03:00
dq_off = offs [ failb ] ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_memcpy ( dp , g , dp_off , g_off , bytes , submit ) ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_mult ( dq , dq_off , g , g_off ,
raid6_gfexp [ good ] , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* compute P + Pxy */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = p ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = p_off ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE | ASYNC_TX_XOR_DROP_DST , tx ,
NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dp , dp_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* compute Q + Qxy */
srcs [ 0 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dq_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = q ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = q_off ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE | ASYNC_TX_XOR_DROP_DST , tx ,
NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dq , dq_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Dx = A*(P+Pxy) + B*(Q+Qxy) */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
coef [ 0 ] = raid6_gfexi [ failb - faila ] ;
coef [ 1 ] = raid6_gfinv [ raid6_gfexp [ faila ] ^ raid6_gfexp [ failb ] ] ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_sum_product ( dq , dq_off , srcs , src_offs , coef , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Dy = P+Pxy+Dx */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
init_async_submit ( submit , flags | ASYNC_TX_XOR_DROP_DST , tx , cb_fn ,
cb_param , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dp , dp_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
return tx ;
}
static struct dma_async_tx_descriptor *
__2data_recov_n ( int disks , size_t bytes , int faila , int failb ,
2020-08-20 16:22:11 +03:00
struct page * * blocks , unsigned int * offs ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_async_tx_descriptor * tx = NULL ;
struct page * p , * q , * dp , * dq ;
2020-08-20 16:22:11 +03:00
unsigned int p_off , q_off , dp_off , dq_off ;
2009-07-14 23:20:37 +04:00
struct page * srcs [ 2 ] ;
2020-08-20 16:22:11 +03:00
unsigned int src_offs [ 2 ] ;
2009-07-14 23:20:37 +04:00
unsigned char coef [ 2 ] ;
enum async_tx_flags flags = submit - > flags ;
dma_async_tx_callback cb_fn = submit - > cb_fn ;
void * cb_param = submit - > cb_param ;
void * scribble = submit - > scribble ;
p = blocks [ disks - 2 ] ;
2020-08-20 16:22:11 +03:00
p_off = offs [ disks - 2 ] ;
2009-07-14 23:20:37 +04:00
q = blocks [ disks - 1 ] ;
2020-08-20 16:22:11 +03:00
q_off = offs [ disks - 1 ] ;
2009-07-14 23:20:37 +04:00
/* Compute syndrome with zero for the missing data pages
* Use the dead data pages as temporary storage for
* delta p and delta q
*/
dp = blocks [ faila ] ;
2020-08-20 16:22:11 +03:00
dp_off = offs [ faila ] ;
2009-10-16 09:40:25 +04:00
blocks [ faila ] = NULL ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 2 ] = dp ;
2020-08-20 16:22:11 +03:00
offs [ disks - 2 ] = dp_off ;
2009-07-14 23:20:37 +04:00
dq = blocks [ failb ] ;
2020-08-20 16:22:11 +03:00
dq_off = offs [ failb ] ;
2009-10-16 09:40:25 +04:00
blocks [ failb ] = NULL ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 1 ] = dq ;
2020-08-20 16:22:11 +03:00
offs [ disks - 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_gen_syndrome ( blocks , offs , disks , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Restore pointer table */
blocks [ faila ] = dp ;
2020-08-20 16:22:11 +03:00
offs [ faila ] = dp_off ;
2009-07-14 23:20:37 +04:00
blocks [ failb ] = dq ;
2020-08-20 16:22:11 +03:00
offs [ failb ] = dq_off ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 2 ] = p ;
2020-08-20 16:22:11 +03:00
offs [ disks - 2 ] = p_off ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 1 ] = q ;
2020-08-20 16:22:11 +03:00
offs [ disks - 1 ] = q_off ;
2009-07-14 23:20:37 +04:00
/* compute P + Pxy */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = p ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = p_off ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE | ASYNC_TX_XOR_DROP_DST , tx ,
NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dp , dp_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* compute Q + Qxy */
srcs [ 0 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dq_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = q ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = q_off ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE | ASYNC_TX_XOR_DROP_DST , tx ,
NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dq , dq_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Dx = A*(P+Pxy) + B*(Q+Qxy) */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
coef [ 0 ] = raid6_gfexi [ failb - faila ] ;
coef [ 1 ] = raid6_gfinv [ raid6_gfexp [ faila ] ^ raid6_gfexp [ failb ] ] ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_sum_product ( dq , dq_off , srcs , src_offs , coef , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
/* Dy = P+Pxy+Dx */
srcs [ 0 ] = dp ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dp_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
init_async_submit ( submit , flags | ASYNC_TX_XOR_DROP_DST , tx , cb_fn ,
cb_param , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dp , dp_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
return tx ;
}
/**
* async_raid6_2data_recov - asynchronously calculate two missing data blocks
* @ disks : number of disks in the RAID - 6 array
* @ bytes : block size
* @ faila : first failed drive index
* @ failb : second failed drive index
* @ blocks : array of source pointers where the last two entries are p and q
2020-08-20 16:22:11 +03:00
* @ offs : array of offset for pages in blocks
2009-07-14 23:20:37 +04:00
* @ submit : submission / completion modifiers
*/
struct dma_async_tx_descriptor *
async_raid6_2data_recov ( int disks , size_t bytes , int faila , int failb ,
2020-08-20 16:22:11 +03:00
struct page * * blocks , unsigned int * offs ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
2010-05-05 07:41:56 +04:00
void * scribble = submit - > scribble ;
2009-10-20 01:05:12 +04:00
int non_zero_srcs , i ;
2009-07-14 23:20:37 +04:00
BUG_ON ( faila = = failb ) ;
if ( failb < faila )
swap ( faila , failb ) ;
pr_debug ( " %s: disks: %d len: %zu \n " , __func__ , disks , bytes ) ;
2010-05-05 07:41:56 +04:00
/* if a dma resource is not available or a scribble buffer is not
* available punt to the synchronous path . In the ' dma not
* available ' case be sure to use the scribble buffer to
* preserve the content of ' blocks ' as the caller intended .
2009-07-14 23:20:37 +04:00
*/
2010-05-05 07:41:56 +04:00
if ( ! async_dma_find_channel ( DMA_PQ ) | | ! scribble ) {
void * * ptrs = scribble ? scribble : ( void * * ) blocks ;
2009-07-14 23:20:37 +04:00
async_tx_quiesce ( & submit - > depend_tx ) ;
for ( i = 0 ; i < disks ; i + + )
2009-10-16 09:40:25 +04:00
if ( blocks [ i ] = = NULL )
2009-10-20 01:05:12 +04:00
ptrs [ i ] = ( void * ) raid6_empty_zero_page ;
2009-10-16 09:40:25 +04:00
else
2020-08-20 16:22:11 +03:00
ptrs [ i ] = page_address ( blocks [ i ] ) + offs [ i ] ;
2009-07-14 23:20:37 +04:00
raid6_2data_recov ( disks , bytes , faila , failb , ptrs ) ;
async_tx_sync_epilog ( submit ) ;
return NULL ;
}
2009-10-20 01:05:12 +04:00
non_zero_srcs = 0 ;
for ( i = 0 ; i < disks - 2 & & non_zero_srcs < 4 ; i + + )
if ( blocks [ i ] )
non_zero_srcs + + ;
switch ( non_zero_srcs ) {
case 0 :
case 1 :
/* There must be at least 2 sources - the failed devices. */
BUG ( ) ;
case 2 :
2009-07-14 23:20:37 +04:00
/* dma devices do not uniformly understand a zero source pq
* operation ( in contrast to the synchronous case ) , so
2009-10-20 01:05:12 +04:00
* explicitly handle the special case of a 4 disk array with
* both data disks missing .
2009-07-14 23:20:37 +04:00
*/
2020-08-20 16:22:11 +03:00
return __2data_recov_4 ( disks , bytes , faila , failb ,
blocks , offs , submit ) ;
2009-10-20 01:05:12 +04:00
case 3 :
2009-07-14 23:20:37 +04:00
/* dma devices do not uniformly understand a single
* source pq operation ( in contrast to the synchronous
2009-10-20 01:05:12 +04:00
* case ) , so explicitly handle the special case of a 5 disk
* array with 2 of 3 data disks missing .
2009-07-14 23:20:37 +04:00
*/
2020-08-20 16:22:11 +03:00
return __2data_recov_5 ( disks , bytes , faila , failb ,
blocks , offs , submit ) ;
2009-07-14 23:20:37 +04:00
default :
2020-08-20 16:22:11 +03:00
return __2data_recov_n ( disks , bytes , faila , failb ,
blocks , offs , submit ) ;
2009-07-14 23:20:37 +04:00
}
}
EXPORT_SYMBOL_GPL ( async_raid6_2data_recov ) ;
/**
* async_raid6_datap_recov - asynchronously calculate a data and the ' p ' block
* @ disks : number of disks in the RAID - 6 array
* @ bytes : block size
* @ faila : failed drive index
* @ blocks : array of source pointers where the last two entries are p and q
2020-08-20 16:22:11 +03:00
* @ offs : array of offset for pages in blocks
2009-07-14 23:20:37 +04:00
* @ submit : submission / completion modifiers
*/
struct dma_async_tx_descriptor *
async_raid6_datap_recov ( int disks , size_t bytes , int faila ,
2020-08-20 16:22:11 +03:00
struct page * * blocks , unsigned int * offs ,
struct async_submit_ctl * submit )
2009-07-14 23:20:37 +04:00
{
struct dma_async_tx_descriptor * tx = NULL ;
struct page * p , * q , * dq ;
2020-08-20 16:22:11 +03:00
unsigned int p_off , q_off , dq_off ;
2009-07-14 23:20:37 +04:00
u8 coef ;
enum async_tx_flags flags = submit - > flags ;
dma_async_tx_callback cb_fn = submit - > cb_fn ;
void * cb_param = submit - > cb_param ;
void * scribble = submit - > scribble ;
2009-10-20 01:05:12 +04:00
int good_srcs , good , i ;
2009-07-14 23:20:37 +04:00
struct page * srcs [ 2 ] ;
2020-08-20 16:22:11 +03:00
unsigned int src_offs [ 2 ] ;
2009-07-14 23:20:37 +04:00
pr_debug ( " %s: disks: %d len: %zu \n " , __func__ , disks , bytes ) ;
2010-05-05 07:41:56 +04:00
/* if a dma resource is not available or a scribble buffer is not
* available punt to the synchronous path . In the ' dma not
* available ' case be sure to use the scribble buffer to
* preserve the content of ' blocks ' as the caller intended .
2009-07-14 23:20:37 +04:00
*/
2010-05-05 07:41:56 +04:00
if ( ! async_dma_find_channel ( DMA_PQ ) | | ! scribble ) {
void * * ptrs = scribble ? scribble : ( void * * ) blocks ;
2009-07-14 23:20:37 +04:00
async_tx_quiesce ( & submit - > depend_tx ) ;
for ( i = 0 ; i < disks ; i + + )
2009-10-16 09:40:25 +04:00
if ( blocks [ i ] = = NULL )
ptrs [ i ] = ( void * ) raid6_empty_zero_page ;
else
2020-08-20 16:22:11 +03:00
ptrs [ i ] = page_address ( blocks [ i ] ) + offs [ i ] ;
2009-07-14 23:20:37 +04:00
raid6_datap_recov ( disks , bytes , faila , ptrs ) ;
async_tx_sync_epilog ( submit ) ;
return NULL ;
}
2009-10-20 01:05:12 +04:00
good_srcs = 0 ;
good = - 1 ;
for ( i = 0 ; i < disks - 2 ; i + + ) {
if ( i = = faila )
continue ;
if ( blocks [ i ] ) {
good = i ;
good_srcs + + ;
if ( good_srcs > 1 )
break ;
}
}
BUG_ON ( good_srcs = = 0 ) ;
2009-07-14 23:20:37 +04:00
p = blocks [ disks - 2 ] ;
2020-08-20 16:22:11 +03:00
p_off = offs [ disks - 2 ] ;
2009-07-14 23:20:37 +04:00
q = blocks [ disks - 1 ] ;
2020-08-20 16:22:11 +03:00
q_off = offs [ disks - 1 ] ;
2009-07-14 23:20:37 +04:00
/* Compute syndrome with zero for the missing data page
* Use the dead data page as temporary storage for delta q
*/
dq = blocks [ faila ] ;
2020-08-20 16:22:11 +03:00
dq_off = offs [ faila ] ;
2009-10-16 09:40:25 +04:00
blocks [ faila ] = NULL ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 1 ] = dq ;
2020-08-20 16:22:11 +03:00
offs [ disks - 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
2009-10-20 01:05:12 +04:00
/* in the 4-disk case we only need to perform a single source
* multiplication with the one good data block .
2009-07-14 23:20:37 +04:00
*/
2009-10-20 01:05:12 +04:00
if ( good_srcs = = 1 ) {
2009-07-14 23:20:37 +04:00
struct page * g = blocks [ good ] ;
2020-08-20 16:22:11 +03:00
unsigned int g_off = offs [ good ] ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL ,
scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_memcpy ( p , g , p_off , g_off , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL ,
scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_mult ( dq , dq_off , g , g_off ,
raid6_gfexp [ good ] , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
} else {
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL ,
scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_gen_syndrome ( blocks , offs , disks , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
}
/* Restore pointer table */
blocks [ faila ] = dq ;
2020-08-20 16:22:11 +03:00
offs [ faila ] = dq_off ;
2009-07-14 23:20:37 +04:00
blocks [ disks - 1 ] = q ;
2020-08-20 16:22:11 +03:00
offs [ disks - 1 ] = q_off ;
2009-07-14 23:20:37 +04:00
/* calculate g^{-faila} */
coef = raid6_gfinv [ raid6_gfexp [ faila ] ] ;
srcs [ 0 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = dq_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = q ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = q_off ;
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE | ASYNC_TX_XOR_DROP_DST , tx ,
NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( dq , dq_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
2009-09-09 04:42:50 +04:00
init_async_submit ( submit , ASYNC_TX_FENCE , tx , NULL , NULL , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_mult ( dq , dq_off , dq , dq_off , coef , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
srcs [ 0 ] = p ;
2020-08-20 16:22:11 +03:00
src_offs [ 0 ] = p_off ;
2009-07-14 23:20:37 +04:00
srcs [ 1 ] = dq ;
2020-08-20 16:22:11 +03:00
src_offs [ 1 ] = dq_off ;
2009-07-14 23:20:37 +04:00
init_async_submit ( submit , flags | ASYNC_TX_XOR_DROP_DST , tx , cb_fn ,
cb_param , scribble ) ;
2020-08-20 16:22:11 +03:00
tx = async_xor_offs ( p , p_off , srcs , src_offs , 2 , bytes , submit ) ;
2009-07-14 23:20:37 +04:00
return tx ;
}
EXPORT_SYMBOL_GPL ( async_raid6_datap_recov ) ;
MODULE_AUTHOR ( " Dan Williams <dan.j.williams@intel.com> " ) ;
MODULE_DESCRIPTION ( " asynchronous RAID-6 recovery api " ) ;
MODULE_LICENSE ( " GPL " ) ;