2005-04-17 02:20:36 +04:00
/*
* iovec manipulation routines .
*
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
* Fixes :
* Andrew Lunn : Errors in iovec copying .
* Pedro Roque : Added memcpy_fromiovecend and
* csum_ . . . _fromiovecend .
* Andi Kleen : fixed error handling for 2.1
* Alexey Kuznetsov : 2.1 optimisations
* Andi Kleen : Fix csum * fromiovecend for IPv6 .
*/
# include <linux/errno.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/net.h>
# include <linux/in6.h>
# include <asm/uaccess.h>
# include <asm/byteorder.h>
# include <net/checksum.h>
# include <net/sock.h>
/*
* Verify iovec . The caller must ensure that the iovec is big enough
* to hold the message iovec .
*
2005-05-01 19:59:08 +04:00
* Save time not doing access_ok . copy_ * _user will make this work
2005-04-17 02:20:36 +04:00
* in any case .
*/
2012-03-11 16:51:50 +04:00
int verify_iovec ( struct msghdr * m , struct iovec * iov , struct sockaddr_storage * address , int mode )
2005-04-17 02:20:36 +04:00
{
2010-10-28 22:41:55 +04:00
int size , ct , err ;
2007-02-09 17:24:36 +03:00
2014-07-26 21:26:58 +04:00
if ( m - > msg_name & & m - > msg_namelen ) {
2005-04-17 02:20:36 +04:00
if ( mode = = VERIFY_READ ) {
2010-09-08 07:48:47 +04:00
void __user * namep ;
namep = ( void __user __force * ) m - > msg_name ;
err = move_addr_to_kernel ( namep , m - > msg_namelen ,
2005-04-17 02:20:36 +04:00
address ) ;
if ( err < 0 )
return err ;
}
2014-07-26 21:26:58 +04:00
m - > msg_name = address ;
2005-04-17 02:20:36 +04:00
} else {
m - > msg_name = NULL ;
2014-07-26 21:26:58 +04:00
m - > msg_namelen = 0 ;
2005-04-17 02:20:36 +04:00
}
size = m - > msg_iovlen * sizeof ( struct iovec ) ;
2010-09-08 07:48:47 +04:00
if ( copy_from_user ( iov , ( void __user __force * ) m - > msg_iov , size ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
m - > msg_iov = iov ;
err = 0 ;
for ( ct = 0 ; ct < m - > msg_iovlen ; ct + + ) {
2010-10-28 22:41:55 +04:00
size_t len = iov [ ct ] . iov_len ;
if ( len > INT_MAX - err ) {
len = INT_MAX - err ;
iov [ ct ] . iov_len = len ;
}
err + = len ;
2005-04-17 02:20:36 +04:00
}
return err ;
}
/*
* And now for the all - in - one : copy and checksum from a user iovec
* directly to a datagram
* Calls to csum_partial but the last must be in 32 bit chunks
*
* ip_build_xmit must ensure that when fragmenting only the last
* call to this function will be unaligned also .
*/
int csum_partial_copy_fromiovecend ( unsigned char * kdata , struct iovec * iov ,
2006-11-15 08:36:14 +03:00
int offset , unsigned int len , __wsum * csump )
2005-04-17 02:20:36 +04:00
{
2006-11-15 08:36:14 +03:00
__wsum csum = * csump ;
2005-04-17 02:20:36 +04:00
int partial_cnt = 0 , err = 0 ;
/* 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 ;
/* There is a remnant from previous iov. */
if ( partial_cnt ) {
int par_len = 4 - partial_cnt ;
/* iov component is too short ... */
if ( par_len > copy ) {
if ( copy_from_user ( kdata , base , copy ) )
goto out_fault ;
kdata + = copy ;
base + = copy ;
partial_cnt + = copy ;
len - = copy ;
iov + + ;
if ( len )
continue ;
* csump = csum_partial ( kdata - partial_cnt ,
partial_cnt , csum ) ;
goto out ;
}
if ( copy_from_user ( kdata , base , par_len ) )
goto out_fault ;
csum = csum_partial ( kdata - partial_cnt , 4 , csum ) ;
kdata + = par_len ;
base + = par_len ;
copy - = par_len ;
len - = par_len ;
partial_cnt = 0 ;
}
if ( len > copy ) {
partial_cnt = copy % 4 ;
if ( partial_cnt ) {
copy - = partial_cnt ;
if ( copy_from_user ( kdata + copy , base + copy ,
2007-02-09 17:24:36 +03:00
partial_cnt ) )
2005-04-17 02:20:36 +04:00
goto out_fault ;
}
}
if ( copy ) {
csum = csum_and_copy_from_user ( base , kdata , copy ,
csum , & err ) ;
if ( err )
goto out ;
}
len - = copy + partial_cnt ;
kdata + = copy + partial_cnt ;
iov + + ;
}
2007-02-09 17:24:36 +03:00
* csump = csum ;
2005-04-17 02:20:36 +04:00
out :
return err ;
out_fault :
err = - EFAULT ;
goto out ;
}
EXPORT_SYMBOL ( csum_partial_copy_fromiovecend ) ;
2013-08-06 13:45:03 +04:00
unsigned long iov_pages ( const struct iovec * iov , int offset ,
unsigned long nr_segs )
{
unsigned long seg , base ;
int pages = 0 , len , size ;
while ( nr_segs & & ( offset > = iov - > iov_len ) ) {
offset - = iov - > iov_len ;
+ + iov ;
- - nr_segs ;
}
for ( seg = 0 ; seg < nr_segs ; seg + + ) {
base = ( unsigned long ) iov [ seg ] . iov_base + offset ;
len = iov [ seg ] . iov_len - offset ;
size = ( ( base & ~ PAGE_MASK ) + len + ~ PAGE_MASK ) > > PAGE_SHIFT ;
pages + = size ;
offset = 0 ;
}
return pages ;
}
EXPORT_SYMBOL ( iov_pages ) ;