2013-05-17 09:05:21 +09:30
# include <linux/uaccess.h>
# include <linux/export.h>
# include <linux/uio.h>
/*
* Copy iovec to kernel . Returns - EFAULT on error .
*
* Note : this modifies the original iovec .
*/
int memcpy_fromiovec ( unsigned char * kdata , struct iovec * iov , int len )
{
while ( len > 0 ) {
if ( iov - > iov_len ) {
int copy = min_t ( unsigned int , len , iov - > iov_len ) ;
if ( copy_from_user ( kdata , iov - > iov_base , copy ) )
return - EFAULT ;
len - = copy ;
kdata + = copy ;
iov - > iov_base + = copy ;
iov - > iov_len - = copy ;
}
iov + + ;
}
return 0 ;
}
EXPORT_SYMBOL ( memcpy_fromiovec ) ;
/*
* Copy kernel to iovec . Returns - EFAULT on error .
*
* Note : this modifies the original iovec .
*/
int memcpy_toiovec ( struct iovec * iov , unsigned char * kdata , int len )
{
while ( len > 0 ) {
if ( iov - > iov_len ) {
int copy = min_t ( unsigned int , iov - > iov_len , len ) ;
if ( copy_to_user ( iov - > iov_base , kdata , copy ) )
return - EFAULT ;
kdata + = copy ;
len - = copy ;
iov - > iov_len - = copy ;
iov - > iov_base + = copy ;
}
iov + + ;
}
return 0 ;
}
EXPORT_SYMBOL ( memcpy_toiovec ) ;
2014-06-19 21:22:56 +03:00
/*
* Copy kernel to iovec . Returns - EFAULT on error .
*/
int memcpy_toiovecend ( const struct iovec * iov , unsigned char * kdata ,
int offset , int len )
{
int copy ;
for ( ; len > 0 ; + + iov ) {
/* Skip over the finished iovecs */
if ( unlikely ( offset > = iov - > iov_len ) ) {
offset - = iov - > iov_len ;
continue ;
}
copy = min_t ( unsigned int , iov - > iov_len - offset , len ) ;
if ( copy_to_user ( iov - > iov_base + offset , kdata , copy ) )
return - EFAULT ;
offset = 0 ;
kdata + = copy ;
len - = copy ;
}
return 0 ;
}
EXPORT_SYMBOL ( memcpy_toiovecend ) ;
/*
* Copy iovec to kernel . Returns - EFAULT on error .
*/
int memcpy_fromiovecend ( unsigned char * kdata , const struct iovec * iov ,
int offset , int len )
{
2014-07-31 23:00:35 -04:00
/* No data? Done! */
if ( len = = 0 )
return 0 ;
2014-06-19 21:22:56 +03:00
/* Skip over the finished iovecs */
while ( offset > = iov - > iov_len ) {
offset - = iov - > iov_len ;
iov + + ;
}
while ( len > 0 ) {
u8 __user * base = iov - > iov_base + offset ;
int copy = min_t ( unsigned int , len , iov - > iov_len - offset ) ;
offset = 0 ;
if ( copy_from_user ( kdata , base , copy ) )
return - EFAULT ;
len - = copy ;
kdata + = copy ;
iov + + ;
}
return 0 ;
}
EXPORT_SYMBOL ( memcpy_fromiovecend ) ;