2014-02-16 11:34:30 -08:00
/*
* An implementation of file copy service .
*
* Copyright ( C ) 2014 , Microsoft , Inc .
*
* Author : K . Y . Srinivasan < ksrinivasan @ novell . com >
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE , GOOD TITLE or
* NON INFRINGEMENT . See the GNU General Public License for more
* details .
*
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/nls.h>
# include <linux/workqueue.h>
# include <linux/hyperv.h>
# include <linux/sched.h>
# include "hyperv_vmbus.h"
2015-04-11 18:07:53 -07:00
# include "hv_utils_transport.h"
2014-02-16 11:34:30 -08:00
# define WIN8_SRV_MAJOR 1
# define WIN8_SRV_MINOR 1
# define WIN8_SRV_VERSION (WIN8_SRV_MAJOR << 16 | WIN8_SRV_MINOR)
/*
* Global state maintained for transaction that is being processed .
* For a class of integration services , including the " file copy service " ,
* the specified protocol is a " request/response " protocol which means that
* there can only be single outstanding transaction from the host at any
* given point in time . We use this to simplify memory management in this
* driver - we cache and process only one message at a time .
*
* While the request / response protocol is guaranteed by the host , we further
* ensure this by serializing packet processing in this driver - we do not
* read additional packets from the VMBUs until the current packet is fully
* handled .
*/
static struct {
2015-04-11 18:07:49 -07:00
int state ; /* hvutil_device_state */
2014-02-16 11:34:30 -08:00
int recv_len ; /* number of bytes received. */
struct hv_fcopy_hdr * fcopy_msg ; /* current message */
struct vmbus_channel * recv_channel ; /* chn we got the request */
u64 recv_req_id ; /* request ID. */
void * fcopy_context ; /* for the channel callback */
} fcopy_transaction ;
static void fcopy_respond_to_host ( int error ) ;
2015-04-11 18:07:53 -07:00
static void fcopy_send_data ( struct work_struct * dummy ) ;
2015-04-11 18:07:45 -07:00
static void fcopy_timeout_func ( struct work_struct * dummy ) ;
static DECLARE_DELAYED_WORK ( fcopy_timeout_work , fcopy_timeout_func ) ;
2015-04-11 18:07:53 -07:00
static DECLARE_WORK ( fcopy_send_work , fcopy_send_data ) ;
static const char fcopy_devname [ ] = " vmbus/hv_fcopy " ;
2014-02-16 11:34:30 -08:00
static u8 * recv_buffer ;
2015-04-11 18:07:53 -07:00
static struct hvutil_transport * hvt ;
2015-04-11 18:07:58 -07:00
/*
* This state maintains the version number registered by the daemon .
*/
static int dm_reg_value ;
2014-02-16 11:34:30 -08:00
2015-04-11 18:07:45 -07:00
static void fcopy_timeout_func ( struct work_struct * dummy )
2014-02-16 11:34:30 -08:00
{
/*
* If the timer fires , the user - mode component has not responded ;
* process the pending transaction .
*/
fcopy_respond_to_host ( HV_E_FAIL ) ;
2015-01-14 01:55:10 -08:00
2015-04-11 18:07:49 -07:00
/* Transaction is finished, reset the state. */
if ( fcopy_transaction . state > HVUTIL_READY )
fcopy_transaction . state = HVUTIL_READY ;
2015-04-11 18:07:42 -07:00
hv_poll_channel ( fcopy_transaction . fcopy_context ,
hv_fcopy_onchannelcallback ) ;
2014-02-16 11:34:30 -08:00
}
static int fcopy_handle_handshake ( u32 version )
{
2015-04-11 18:07:58 -07:00
u32 our_ver = FCOPY_CURRENT_VERSION ;
2014-02-16 11:34:30 -08:00
switch ( version ) {
2015-04-11 18:07:58 -07:00
case FCOPY_VERSION_0 :
/* Daemon doesn't expect us to reply */
dm_reg_value = version ;
break ;
case FCOPY_VERSION_1 :
/* Daemon expects us to reply with our own version */
if ( hvutil_transport_send ( hvt , & our_ver , sizeof ( our_ver ) ) )
return - EFAULT ;
dm_reg_value = version ;
2014-02-16 11:34:30 -08:00
break ;
default :
/*
* For now we will fail the registration .
* If and when we have multiple versions to
* deal with , we will be backward compatible .
* We will add this code when needed .
*/
return - EINVAL ;
}
pr_info ( " FCP: user-mode registering done. Daemon version: %d \n " ,
version ) ;
2015-04-11 18:07:49 -07:00
fcopy_transaction . state = HVUTIL_READY ;
2015-04-11 18:07:42 -07:00
hv_poll_channel ( fcopy_transaction . fcopy_context ,
hv_fcopy_onchannelcallback ) ;
2014-02-16 11:34:30 -08:00
return 0 ;
}
2015-04-11 18:07:53 -07:00
static void fcopy_send_data ( struct work_struct * dummy )
2014-02-16 11:34:30 -08:00
{
2015-04-11 18:07:53 -07:00
struct hv_start_fcopy smsg_out ;
2014-02-16 11:34:30 -08:00
int operation = fcopy_transaction . fcopy_msg - > operation ;
struct hv_start_fcopy * smsg_in ;
2015-04-11 18:07:53 -07:00
void * out_src ;
int rc , out_len ;
2014-02-16 11:34:30 -08:00
/*
* The strings sent from the host are encoded in
* in utf16 ; convert it to utf8 strings .
* The host assures us that the utf16 strings will not exceed
* the max lengths specified . We will however , reserve room
* for the string terminating character - in the utf16s_utf8s ( )
* function we limit the size of the buffer where the converted
* string is placed to W_MAX_PATH - 1 to guarantee
* that the strings can be properly terminated !
*/
switch ( operation ) {
case START_FILE_COPY :
2015-04-11 18:07:53 -07:00
out_len = sizeof ( struct hv_start_fcopy ) ;
memset ( & smsg_out , 0 , out_len ) ;
smsg_out . hdr . operation = operation ;
2014-02-16 11:34:30 -08:00
smsg_in = ( struct hv_start_fcopy * ) fcopy_transaction . fcopy_msg ;
utf16s_to_utf8s ( ( wchar_t * ) smsg_in - > file_name , W_MAX_PATH ,
UTF16_LITTLE_ENDIAN ,
2015-04-11 18:07:53 -07:00
( __u8 * ) & smsg_out . file_name , W_MAX_PATH - 1 ) ;
2014-02-16 11:34:30 -08:00
utf16s_to_utf8s ( ( wchar_t * ) smsg_in - > path_name , W_MAX_PATH ,
UTF16_LITTLE_ENDIAN ,
2015-04-11 18:07:53 -07:00
( __u8 * ) & smsg_out . path_name , W_MAX_PATH - 1 ) ;
2014-02-16 11:34:30 -08:00
2015-04-11 18:07:53 -07:00
smsg_out . copy_flags = smsg_in - > copy_flags ;
smsg_out . file_size = smsg_in - > file_size ;
out_src = & smsg_out ;
2014-02-16 11:34:30 -08:00
break ;
default :
2015-04-11 18:07:53 -07:00
out_src = fcopy_transaction . fcopy_msg ;
out_len = fcopy_transaction . recv_len ;
2014-02-16 11:34:30 -08:00
break ;
}
2015-04-11 18:07:53 -07:00
fcopy_transaction . state = HVUTIL_USERSPACE_REQ ;
rc = hvutil_transport_send ( hvt , out_src , out_len ) ;
if ( rc ) {
pr_debug ( " FCP: failed to communicate to the daemon: %d \n " , rc ) ;
if ( cancel_delayed_work_sync ( & fcopy_timeout_work ) ) {
fcopy_respond_to_host ( HV_E_FAIL ) ;
fcopy_transaction . state = HVUTIL_READY ;
}
}
2014-02-16 11:34:30 -08:00
return ;
}
/*
* Send a response back to the host .
*/
static void
fcopy_respond_to_host ( int error )
{
struct icmsg_hdr * icmsghdr ;
u32 buf_len ;
struct vmbus_channel * channel ;
u64 req_id ;
/*
* Copy the global state for completing the transaction . Note that
* only one transaction can be active at a time . This is guaranteed
* by the file copy protocol implemented by the host . Furthermore ,
* the " transaction active " state we maintain ensures that there can
* only be one active transaction at a time .
*/
buf_len = fcopy_transaction . recv_len ;
channel = fcopy_transaction . recv_channel ;
req_id = fcopy_transaction . recv_req_id ;
icmsghdr = ( struct icmsg_hdr * )
& recv_buffer [ sizeof ( struct vmbuspipe_hdr ) ] ;
if ( channel - > onchannel_callback = = NULL )
/*
* We have raced with util driver being unloaded ;
* silently return .
*/
return ;
icmsghdr - > status = error ;
icmsghdr - > icflags = ICMSGHDRFLAG_TRANSACTION | ICMSGHDRFLAG_RESPONSE ;
vmbus_sendpacket ( channel , recv_buffer , buf_len , req_id ,
VM_PKT_DATA_INBAND , 0 ) ;
}
void hv_fcopy_onchannelcallback ( void * context )
{
struct vmbus_channel * channel = context ;
u32 recvlen ;
u64 requestid ;
struct hv_fcopy_hdr * fcopy_msg ;
struct icmsg_hdr * icmsghdr ;
struct icmsg_negotiate * negop = NULL ;
int util_fw_version ;
int fcopy_srv_version ;
2015-04-11 18:07:49 -07:00
if ( fcopy_transaction . state > HVUTIL_READY ) {
2014-02-16 11:34:30 -08:00
/*
* We will defer processing this callback once
* the current transaction is complete .
*/
fcopy_transaction . fcopy_context = context ;
return ;
}
2015-04-11 18:07:42 -07:00
fcopy_transaction . fcopy_context = NULL ;
2014-02-16 11:34:30 -08:00
vmbus_recvpacket ( channel , recv_buffer , PAGE_SIZE * 2 , & recvlen ,
& requestid ) ;
if ( recvlen < = 0 )
return ;
icmsghdr = ( struct icmsg_hdr * ) & recv_buffer [
sizeof ( struct vmbuspipe_hdr ) ] ;
if ( icmsghdr - > icmsgtype = = ICMSGTYPE_NEGOTIATE ) {
util_fw_version = UTIL_FW_VERSION ;
fcopy_srv_version = WIN8_SRV_VERSION ;
vmbus_prep_negotiate_resp ( icmsghdr , negop , recv_buffer ,
util_fw_version , fcopy_srv_version ) ;
} else {
fcopy_msg = ( struct hv_fcopy_hdr * ) & recv_buffer [
sizeof ( struct vmbuspipe_hdr ) +
sizeof ( struct icmsg_hdr ) ] ;
/*
* Stash away this global state for completing the
* transaction ; note transactions are serialized .
*/
fcopy_transaction . recv_len = recvlen ;
fcopy_transaction . recv_channel = channel ;
fcopy_transaction . recv_req_id = requestid ;
fcopy_transaction . fcopy_msg = fcopy_msg ;
2015-04-11 18:07:49 -07:00
if ( fcopy_transaction . state < HVUTIL_READY ) {
/* Userspace is not registered yet */
fcopy_respond_to_host ( HV_E_FAIL ) ;
return ;
}
fcopy_transaction . state = HVUTIL_HOSTMSG_RECEIVED ;
2014-02-16 11:34:30 -08:00
/*
* Send the information to the user - level daemon .
*/
2015-04-11 18:07:53 -07:00
schedule_work ( & fcopy_send_work ) ;
2015-04-11 18:07:45 -07:00
schedule_delayed_work ( & fcopy_timeout_work , 5 * HZ ) ;
2014-02-16 11:34:30 -08:00
return ;
}
icmsghdr - > icflags = ICMSGHDRFLAG_TRANSACTION | ICMSGHDRFLAG_RESPONSE ;
vmbus_sendpacket ( channel , recv_buffer , recvlen , requestid ,
VM_PKT_DATA_INBAND , 0 ) ;
}
2015-04-11 18:07:53 -07:00
/* Callback when data is received from userspace */
static int fcopy_on_msg ( void * msg , int len )
2014-02-16 11:34:30 -08:00
{
2015-04-11 18:07:53 -07:00
int * val = ( int * ) msg ;
2014-02-16 11:34:30 -08:00
2015-04-11 18:07:53 -07:00
if ( len ! = sizeof ( int ) )
2014-02-16 11:34:30 -08:00
return - EINVAL ;
2015-04-11 18:07:53 -07:00
if ( fcopy_transaction . state = = HVUTIL_DEVICE_INIT )
return fcopy_handle_handshake ( * val ) ;
2014-02-16 11:34:30 -08:00
2015-04-11 18:07:49 -07:00
if ( fcopy_transaction . state ! = HVUTIL_USERSPACE_REQ )
return - EINVAL ;
2014-02-16 11:34:30 -08:00
/*
* Complete the transaction by forwarding the result
* to the host . But first , cancel the timeout .
*/
2015-04-11 18:07:49 -07:00
if ( cancel_delayed_work_sync ( & fcopy_timeout_work ) ) {
fcopy_transaction . state = HVUTIL_USERSPACE_RECV ;
2015-04-11 18:07:53 -07:00
fcopy_respond_to_host ( * val ) ;
2015-04-11 18:07:49 -07:00
fcopy_transaction . state = HVUTIL_READY ;
2015-04-11 18:07:42 -07:00
hv_poll_channel ( fcopy_transaction . fcopy_context ,
hv_fcopy_onchannelcallback ) ;
}
2014-02-16 11:34:30 -08:00
return 0 ;
}
2015-04-11 18:07:53 -07:00
static void fcopy_on_reset ( void )
2014-02-16 11:34:30 -08:00
{
/*
* The daemon has exited ; reset the state .
*/
2015-04-11 18:07:49 -07:00
fcopy_transaction . state = HVUTIL_DEVICE_INIT ;
2015-01-14 01:55:10 -08:00
2015-04-11 18:07:53 -07:00
if ( cancel_delayed_work_sync ( & fcopy_timeout_work ) )
2015-01-14 01:55:10 -08:00
fcopy_respond_to_host ( HV_E_FAIL ) ;
2014-02-16 11:34:30 -08:00
}
int hv_fcopy_init ( struct hv_util_service * srv )
{
recv_buffer = srv - > recv_buffer ;
/*
* When this driver loads , the user level daemon that
* processes the host requests may not yet be running .
* Defer processing channel callbacks until the daemon
* has registered .
*/
2015-04-11 18:07:49 -07:00
fcopy_transaction . state = HVUTIL_DEVICE_INIT ;
2014-02-16 11:34:30 -08:00
2015-04-11 18:07:53 -07:00
hvt = hvutil_transport_init ( fcopy_devname , 0 , 0 ,
fcopy_on_msg , fcopy_on_reset ) ;
if ( ! hvt )
return - EFAULT ;
return 0 ;
2014-02-16 11:34:30 -08:00
}
void hv_fcopy_deinit ( void )
{
2015-04-11 18:07:49 -07:00
fcopy_transaction . state = HVUTIL_DEVICE_DYING ;
2015-04-11 18:07:45 -07:00
cancel_delayed_work_sync ( & fcopy_timeout_work ) ;
2015-04-11 18:07:53 -07:00
hvutil_transport_destroy ( hvt ) ;
2014-02-16 11:34:30 -08:00
}