2009-07-13 16:02:34 -07:00
/*
*
* Copyright ( c ) 2009 , Microsoft Corporation .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
* You should have received a copy of the GNU General Public License along with
* this program ; if not , write to the Free Software Foundation , Inc . , 59 Temple
* Place - Suite 330 , Boston , MA 02111 - 1307 USA .
*
* Authors :
* Haiyang Zhang < haiyangz @ microsoft . com >
* Hank Janssen < hjanssen @ microsoft . com >
2011-05-10 07:55:30 -07:00
* K . Y . Srinivasan < kys @ microsoft . com >
2009-07-13 16:02:34 -07:00
*
*/
2011-03-29 13:58:47 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-07-13 16:02:34 -07:00
2009-08-17 17:22:08 -07:00
# include <linux/kernel.h>
# include <linux/mm.h>
2011-10-04 12:29:52 -07:00
# include <linux/hyperv.h>
2014-02-01 19:02:20 -08:00
# include <linux/uio.h>
2016-09-02 05:58:20 -07:00
# include <linux/vmalloc.h>
# include <linux/slab.h>
2011-05-12 19:34:15 -07:00
2011-05-12 19:34:28 -07:00
# include "hyperv_vmbus.h"
2009-07-13 16:02:34 -07:00
2012-12-01 06:46:32 -08:00
void hv_begin_read ( struct hv_ring_buffer_info * rbi )
{
rbi - > ring_buffer - > interrupt_mask = 1 ;
2016-04-02 17:59:48 -07:00
virt_mb ( ) ;
2012-12-01 06:46:32 -08:00
}
u32 hv_end_read ( struct hv_ring_buffer_info * rbi )
{
rbi - > ring_buffer - > interrupt_mask = 0 ;
2016-04-02 17:59:48 -07:00
virt_mb ( ) ;
2012-12-01 06:46:32 -08:00
/*
* Now check to see if the ring buffer is still empty .
* If it is not , we raced and we need to process new
* incoming messages .
*/
2016-04-02 17:59:46 -07:00
return hv_get_bytes_to_read ( rbi ) ;
2012-12-01 06:46:32 -08:00
}
2012-12-01 06:46:36 -08:00
/*
* When we write to the ring buffer , check if the host needs to
* be signaled . Here is the details of this protocol :
*
* 1. The host guarantees that while it is draining the
* ring buffer , it will set the interrupt_mask to
* indicate it does not need to be interrupted when
* new data is placed .
*
* 2. The host guarantees that it will completely drain
* the ring buffer before exiting the read loop . Further ,
* once the ring buffer is empty , it will clear the
* interrupt_mask and re - check to see if new data has
* arrived .
*/
2016-07-01 16:26:35 -07:00
static bool hv_need_to_signal ( u32 old_write , struct hv_ring_buffer_info * rbi ,
enum hv_signal_policy policy )
2012-12-01 06:46:36 -08:00
{
2016-04-02 17:59:48 -07:00
virt_mb ( ) ;
2016-04-02 17:59:47 -07:00
if ( READ_ONCE ( rbi - > ring_buffer - > interrupt_mask ) )
2012-12-01 06:46:36 -08:00
return false ;
2016-07-01 16:26:35 -07:00
/*
* When the client wants to control signaling ,
* we only honour the host interrupt mask .
*/
if ( policy = = HV_SIGNAL_POLICY_EXPLICIT )
return true ;
2013-06-20 12:58:57 +08:00
/* check interrupt_mask before read_index */
2016-04-02 17:59:48 -07:00
virt_rmb ( ) ;
2012-12-01 06:46:36 -08:00
/*
* This is the only case we need to signal when the
* ring transitions from being empty to non - empty .
*/
2016-04-02 17:59:47 -07:00
if ( old_write = = READ_ONCE ( rbi - > ring_buffer - > read_index ) )
2012-12-01 06:46:36 -08:00
return true ;
return false ;
}
2015-12-14 19:01:57 -08:00
/* Get the next write location for the specified ring buffer. */
2009-07-14 15:09:36 -07:00
static inline u32
2011-05-10 07:55:29 -07:00
hv_get_next_write_location ( struct hv_ring_buffer_info * ring_info )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
u32 next = ring_info - > ring_buffer - > write_index ;
2009-07-13 16:02:34 -07:00
return next ;
}
2015-12-14 19:01:57 -08:00
/* Set the next write location for the specified ring buffer. */
2009-07-13 16:02:34 -07:00
static inline void
2011-05-10 07:55:29 -07:00
hv_set_next_write_location ( struct hv_ring_buffer_info * ring_info ,
2010-11-08 14:04:46 -08:00
u32 next_write_location )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > write_index = next_write_location ;
2009-07-13 16:02:34 -07:00
}
2015-12-14 19:01:57 -08:00
/* Get the next read location for the specified ring buffer. */
2009-07-14 15:09:36 -07:00
static inline u32
2011-05-10 07:55:29 -07:00
hv_get_next_read_location ( struct hv_ring_buffer_info * ring_info )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
u32 next = ring_info - > ring_buffer - > read_index ;
2009-07-13 16:02:34 -07:00
return next ;
}
2011-05-10 07:55:30 -07:00
/*
* Get the next read location + offset for the specified ring buffer .
2015-12-14 19:01:57 -08:00
* This allows the caller to skip .
2011-05-10 07:55:30 -07:00
*/
2009-07-14 15:09:36 -07:00
static inline u32
2011-05-10 07:55:29 -07:00
hv_get_next_readlocation_withoffset ( struct hv_ring_buffer_info * ring_info ,
2010-11-08 14:04:47 -08:00
u32 offset )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
u32 next = ring_info - > ring_buffer - > read_index ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
next + = offset ;
next % = ring_info - > ring_datasize ;
2009-07-13 16:02:34 -07:00
return next ;
}
2015-12-14 19:01:57 -08:00
/* Set the next read location for the specified ring buffer. */
2009-07-13 16:02:34 -07:00
static inline void
2011-05-10 07:55:29 -07:00
hv_set_next_read_location ( struct hv_ring_buffer_info * ring_info ,
2010-11-08 14:04:46 -08:00
u32 next_read_location )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > read_index = next_read_location ;
2016-04-02 17:59:51 -07:00
ring_info - > priv_read_index = next_read_location ;
2009-07-13 16:02:34 -07:00
}
2015-12-14 19:01:57 -08:00
/* Get the size of the ring buffer. */
2009-07-14 15:09:36 -07:00
static inline u32
2011-05-10 07:55:29 -07:00
hv_get_ring_buffersize ( struct hv_ring_buffer_info * ring_info )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
return ring_info - > ring_datasize ;
2009-07-13 16:02:34 -07:00
}
2015-12-14 19:01:57 -08:00
/* Get the read and write indices as u64 of the specified ring buffer. */
2009-07-14 15:10:26 -07:00
static inline u64
2011-05-10 07:55:29 -07:00
hv_get_ring_bufferindices ( struct hv_ring_buffer_info * ring_info )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
return ( u64 ) ring_info - > ring_buffer - > write_index < < 32 ;
2009-07-13 16:02:34 -07:00
}
2011-05-10 07:55:31 -07:00
/*
* Helper routine to copy to source from ring buffer .
* Assume there is enough room . Handles wrap - around in src case only ! !
*/
static u32 hv_copyfrom_ringbuffer (
struct hv_ring_buffer_info * ring_info ,
void * dest ,
u32 destlen ,
u32 start_read_offset )
{
void * ring_buffer = hv_get_ring_buffer ( ring_info ) ;
u32 ring_buffer_size = hv_get_ring_buffersize ( ring_info ) ;
2016-09-02 05:58:21 -07:00
memcpy ( dest , ring_buffer + start_read_offset , destlen ) ;
2011-05-10 07:55:31 -07:00
start_read_offset + = destlen ;
start_read_offset % = ring_buffer_size ;
return start_read_offset ;
}
2011-05-10 07:55:32 -07:00
/*
* Helper routine to copy from source to ring buffer .
* Assume there is enough room . Handles wrap - around in dest case only ! !
*/
static u32 hv_copyto_ringbuffer (
2010-11-08 14:04:46 -08:00
struct hv_ring_buffer_info * ring_info ,
u32 start_write_offset ,
void * src ,
2011-05-10 07:55:32 -07:00
u32 srclen )
{
void * ring_buffer = hv_get_ring_buffer ( ring_info ) ;
u32 ring_buffer_size = hv_get_ring_buffersize ( ring_info ) ;
2016-09-02 05:58:21 -07:00
memcpy ( ring_buffer + start_write_offset , src , srclen ) ;
2009-07-13 16:02:34 -07:00
2011-05-10 07:55:32 -07:00
start_write_offset + = srclen ;
start_write_offset % = ring_buffer_size ;
return start_write_offset ;
}
2009-07-13 16:02:34 -07:00
2015-12-14 19:01:57 -08:00
/* Get various debug metrics for the specified ring buffer. */
2011-05-10 07:55:28 -07:00
void hv_ringbuffer_get_debuginfo ( struct hv_ring_buffer_info * ring_info ,
2010-07-27 11:37:32 -07:00
struct hv_ring_buffer_debug_info * debug_info )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
u32 bytes_avail_towrite ;
u32 bytes_avail_toread ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
if ( ring_info - > ring_buffer ) {
2011-05-10 07:55:29 -07:00
hv_get_ringbuffer_availbytes ( ring_info ,
2010-11-08 14:04:46 -08:00
& bytes_avail_toread ,
& bytes_avail_towrite ) ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
debug_info - > bytes_avail_toread = bytes_avail_toread ;
debug_info - > bytes_avail_towrite = bytes_avail_towrite ;
2010-11-08 14:04:45 -08:00
debug_info - > current_read_index =
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > read_index ;
2010-11-08 14:04:45 -08:00
debug_info - > current_write_index =
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > write_index ;
2010-11-08 14:04:45 -08:00
debug_info - > current_interrupt_mask =
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > interrupt_mask ;
2009-07-13 16:02:34 -07:00
}
}
2015-12-14 19:01:57 -08:00
/* Initialize the ring buffer. */
2011-05-10 07:55:21 -07:00
int hv_ringbuffer_init ( struct hv_ring_buffer_info * ring_info ,
2016-09-02 05:58:20 -07:00
struct page * pages , u32 page_cnt )
2009-07-13 16:02:34 -07:00
{
2016-09-02 05:58:20 -07:00
int i ;
struct page * * pages_wraparound ;
BUILD_BUG_ON ( ( sizeof ( struct hv_ring_buffer ) ! = PAGE_SIZE ) ) ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
memset ( ring_info , 0 , sizeof ( struct hv_ring_buffer_info ) ) ;
2009-07-13 16:02:34 -07:00
2016-09-02 05:58:20 -07:00
/*
* First page holds struct hv_ring_buffer , do wraparound mapping for
* the rest .
*/
pages_wraparound = kzalloc ( sizeof ( struct page * ) * ( page_cnt * 2 - 1 ) ,
GFP_KERNEL ) ;
if ( ! pages_wraparound )
return - ENOMEM ;
pages_wraparound [ 0 ] = pages ;
for ( i = 0 ; i < 2 * ( page_cnt - 1 ) ; i + + )
pages_wraparound [ i + 1 ] = & pages [ i % ( page_cnt - 1 ) + 1 ] ;
ring_info - > ring_buffer = ( struct hv_ring_buffer * )
vmap ( pages_wraparound , page_cnt * 2 - 1 , VM_MAP , PAGE_KERNEL ) ;
kfree ( pages_wraparound ) ;
if ( ! ring_info - > ring_buffer )
return - ENOMEM ;
2010-11-08 14:04:46 -08:00
ring_info - > ring_buffer - > read_index =
ring_info - > ring_buffer - > write_index = 0 ;
2009-07-13 16:02:34 -07:00
2015-12-14 19:01:57 -08:00
/* Set the feature bit for enabling flow control. */
2014-09-05 17:29:12 -07:00
ring_info - > ring_buffer - > feature_bits . value = 1 ;
2016-09-02 05:58:20 -07:00
ring_info - > ring_size = page_cnt < < PAGE_SHIFT ;
ring_info - > ring_datasize = ring_info - > ring_size -
sizeof ( struct hv_ring_buffer ) ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
spin_lock_init ( & ring_info - > ring_lock ) ;
2009-07-13 16:02:34 -07:00
return 0 ;
}
2015-12-14 19:01:57 -08:00
/* Cleanup the ring buffer. */
2011-05-10 07:55:22 -07:00
void hv_ringbuffer_cleanup ( struct hv_ring_buffer_info * ring_info )
2009-07-13 16:02:34 -07:00
{
2016-09-02 05:58:20 -07:00
vunmap ( ring_info - > ring_buffer ) ;
2009-07-13 16:02:34 -07:00
}
2015-12-14 19:01:57 -08:00
/* Write to the ring buffer. */
2011-05-10 07:55:23 -07:00
int hv_ringbuffer_write ( struct hv_ring_buffer_info * outring_info ,
2016-07-01 16:26:35 -07:00
struct kvec * kv_list , u32 kv_count , bool * signal , bool lock ,
enum hv_signal_policy policy )
2009-07-13 16:02:34 -07:00
{
2010-02-03 15:34:27 +00:00
int i = 0 ;
2010-11-08 14:04:46 -08:00
u32 bytes_avail_towrite ;
u32 totalbytes_towrite = 0 ;
2009-07-13 16:02:34 -07:00
2011-05-10 07:55:33 -07:00
u32 next_write_location ;
2012-12-01 06:46:36 -08:00
u32 old_write ;
2010-11-08 14:04:46 -08:00
u64 prev_indices = 0 ;
2016-01-27 22:29:45 -08:00
unsigned long flags = 0 ;
2009-07-13 16:02:34 -07:00
2014-02-01 19:02:20 -08:00
for ( i = 0 ; i < kv_count ; i + + )
totalbytes_towrite + = kv_list [ i ] . iov_len ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
totalbytes_towrite + = sizeof ( u64 ) ;
2009-07-13 16:02:34 -07:00
2016-01-27 22:29:45 -08:00
if ( lock )
spin_lock_irqsave ( & outring_info - > ring_lock , flags ) ;
2009-07-13 16:02:34 -07:00
2016-04-02 17:59:46 -07:00
bytes_avail_towrite = hv_get_bytes_to_write ( outring_info ) ;
2009-07-13 16:02:34 -07:00
2015-12-14 19:01:57 -08:00
/*
* If there is only room for the packet , assume it is full .
* Otherwise , the next time around , we think the ring buffer
* is empty since the read index = = write index .
*/
2010-11-08 14:04:46 -08:00
if ( bytes_avail_towrite < = totalbytes_towrite ) {
2016-01-27 22:29:45 -08:00
if ( lock )
spin_unlock_irqrestore ( & outring_info - > ring_lock , flags ) ;
2011-08-25 09:48:58 -07:00
return - EAGAIN ;
2009-07-13 16:02:34 -07:00
}
2009-07-27 16:47:24 -04:00
/* Write to the ring buffer */
2011-05-10 07:55:29 -07:00
next_write_location = hv_get_next_write_location ( outring_info ) ;
2009-07-13 16:02:34 -07:00
2012-12-01 06:46:36 -08:00
old_write = next_write_location ;
2014-02-01 19:02:20 -08:00
for ( i = 0 ; i < kv_count ; i + + ) {
2011-05-10 07:55:29 -07:00
next_write_location = hv_copyto_ringbuffer ( outring_info ,
2010-11-08 14:04:46 -08:00
next_write_location ,
2014-02-01 19:02:20 -08:00
kv_list [ i ] . iov_base ,
kv_list [ i ] . iov_len ) ;
2009-07-13 16:02:34 -07:00
}
2009-07-27 16:47:24 -04:00
/* Set previous packet start */
2011-05-10 07:55:29 -07:00
prev_indices = hv_get_ring_bufferindices ( outring_info ) ;
2009-07-13 16:02:34 -07:00
2011-05-10 07:55:29 -07:00
next_write_location = hv_copyto_ringbuffer ( outring_info ,
2010-11-08 14:04:46 -08:00
next_write_location ,
& prev_indices ,
2009-07-30 17:37:23 +02:00
sizeof ( u64 ) ) ;
2009-07-13 16:02:34 -07:00
2012-12-01 06:46:36 -08:00
/* Issue a full memory barrier before updating the write index */
2016-04-02 17:59:48 -07:00
virt_mb ( ) ;
2009-07-13 16:02:34 -07:00
2009-07-27 16:47:24 -04:00
/* Now, update the write location */
2011-05-10 07:55:29 -07:00
hv_set_next_write_location ( outring_info , next_write_location ) ;
2009-07-13 16:02:34 -07:00
2016-01-27 22:29:45 -08:00
if ( lock )
spin_unlock_irqrestore ( & outring_info - > ring_lock , flags ) ;
2012-12-01 06:46:36 -08:00
2016-07-01 16:26:35 -07:00
* signal = hv_need_to_signal ( old_write , outring_info , policy ) ;
2009-07-13 16:02:34 -07:00
return 0 ;
}
2015-12-14 19:02:01 -08:00
int hv_ringbuffer_read ( struct hv_ring_buffer_info * inring_info ,
void * buffer , u32 buflen , u32 * buffer_actual_len ,
u64 * requestid , bool * signal , bool raw )
2009-07-13 16:02:34 -07:00
{
2010-11-08 14:04:46 -08:00
u32 bytes_avail_toread ;
u32 next_read_location = 0 ;
u64 prev_indices = 0 ;
2015-12-14 19:02:01 -08:00
struct vmpacket_descriptor desc ;
u32 offset ;
u32 packetlen ;
int ret = 0 ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:46 -08:00
if ( buflen < = 0 )
2010-05-05 15:27:50 -04:00
return - EINVAL ;
2009-07-13 16:02:34 -07:00
2015-12-14 19:02:01 -08:00
* buffer_actual_len = 0 ;
* requestid = 0 ;
2016-04-02 17:59:46 -07:00
bytes_avail_toread = hv_get_bytes_to_read ( inring_info ) ;
2009-07-27 16:47:24 -04:00
/* Make sure there is something to read */
2015-12-14 19:02:01 -08:00
if ( bytes_avail_toread < sizeof ( desc ) ) {
/*
* No error is set when there is even no header , drivers are
* supposed to analyze buffer_actual_len .
*/
2016-01-27 22:29:44 -08:00
return ret ;
2015-12-14 19:02:01 -08:00
}
2009-07-13 16:02:34 -07:00
2015-12-14 19:02:01 -08:00
next_read_location = hv_get_next_read_location ( inring_info ) ;
next_read_location = hv_copyfrom_ringbuffer ( inring_info , & desc ,
sizeof ( desc ) ,
next_read_location ) ;
offset = raw ? 0 : ( desc . offset8 < < 3 ) ;
packetlen = ( desc . len8 < < 3 ) - offset ;
* buffer_actual_len = packetlen ;
* requestid = desc . trans_id ;
2016-01-27 22:29:44 -08:00
if ( bytes_avail_toread < packetlen + offset )
return - EAGAIN ;
2015-12-14 19:02:01 -08:00
2016-01-27 22:29:44 -08:00
if ( packetlen > buflen )
return - ENOBUFS ;
2009-07-13 16:02:34 -07:00
2010-11-08 14:04:47 -08:00
next_read_location =
2011-05-10 07:55:29 -07:00
hv_get_next_readlocation_withoffset ( inring_info , offset ) ;
2009-07-13 16:02:34 -07:00
2011-05-10 07:55:29 -07:00
next_read_location = hv_copyfrom_ringbuffer ( inring_info ,
2010-11-08 14:04:46 -08:00
buffer ,
2015-12-14 19:02:01 -08:00
packetlen ,
2010-11-08 14:04:46 -08:00
next_read_location ) ;
2009-07-13 16:02:34 -07:00
2011-05-10 07:55:29 -07:00
next_read_location = hv_copyfrom_ringbuffer ( inring_info ,
2010-11-08 14:04:46 -08:00
& prev_indices ,
2010-02-03 15:34:27 +00:00
sizeof ( u64 ) ,
2010-11-08 14:04:46 -08:00
next_read_location ) ;
2009-07-13 16:02:34 -07:00
2015-12-14 19:01:57 -08:00
/*
* Make sure all reads are done before we update the read index since
* the writer may start writing to the read area once the read index
* is updated .
*/
2016-04-02 17:59:48 -07:00
virt_mb ( ) ;
2009-07-13 16:02:34 -07:00
2009-07-27 16:47:24 -04:00
/* Update the read index */
2011-05-10 07:55:29 -07:00
hv_set_next_read_location ( inring_info , next_read_location ) ;
2009-07-13 16:02:34 -07:00
2016-04-02 16:17:38 -07:00
* signal = hv_need_to_signal_on_read ( inring_info ) ;
2012-12-01 06:46:57 -08:00
2015-12-14 19:02:01 -08:00
return ret ;
2015-12-14 19:01:59 -08:00
}