2010-05-04 15:55:05 -07:00
/*
* Copyright ( c ) 2010 , 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-03-29 13:58:49 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2010-05-04 15:55:05 -07:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/sysctl.h>
2010-05-04 08:26:23 -07:00
# include <linux/reboot.h>
2011-10-04 12:29:52 -07:00
# include <linux/hyperv.h>
2010-05-04 15:55:05 -07:00
2011-09-18 10:31:33 -07:00
static void shutdown_onchannelcallback ( void * context ) ;
static struct hv_util_service util_shutdown = {
. util_cb = shutdown_onchannelcallback ,
} ;
static void timesync_onchannelcallback ( void * context ) ;
static struct hv_util_service util_timesynch = {
. util_cb = timesync_onchannelcallback ,
} ;
static void heartbeat_onchannelcallback ( void * context ) ;
static struct hv_util_service util_heartbeat = {
. util_cb = heartbeat_onchannelcallback ,
} ;
static struct hv_util_service util_kvp = {
. util_cb = hv_kvp_onchannelcallback ,
. util_init = hv_kvp_init ,
. util_deinit = hv_kvp_deinit ,
} ;
2010-05-04 15:55:05 -07:00
2010-05-04 14:31:18 -07:00
static void shutdown_onchannelcallback ( void * context )
2010-05-04 15:55:05 -07:00
{
struct vmbus_channel * channel = context ;
2010-12-13 16:23:36 -08:00
u32 recvlen ;
2010-05-04 15:55:05 -07:00
u64 requestid ;
u8 execute_shutdown = false ;
2011-09-18 10:31:33 -07:00
u8 * shut_txf_buf = util_shutdown . recv_buffer ;
2010-05-04 15:55:05 -07:00
struct shutdown_msg_data * shutdown_msg ;
struct icmsg_hdr * icmsghdrp ;
struct icmsg_negotiate * negop = NULL ;
2010-12-13 16:23:36 -08:00
vmbus_recvpacket ( channel , shut_txf_buf ,
PAGE_SIZE , & recvlen , & requestid ) ;
2010-05-04 15:55:05 -07:00
if ( recvlen > 0 ) {
2010-12-13 16:23:36 -08:00
icmsghdrp = ( struct icmsg_hdr * ) & shut_txf_buf [
2010-05-04 15:55:05 -07:00
sizeof ( struct vmbuspipe_hdr ) ] ;
if ( icmsghdrp - > icmsgtype = = ICMSGTYPE_NEGOTIATE ) {
2012-05-12 13:44:58 -07:00
vmbus_prep_negotiate_resp ( icmsghdrp , negop ,
shut_txf_buf , MAX_SRV_VER , MAX_SRV_VER ) ;
2010-05-04 15:55:05 -07:00
} else {
2010-12-13 16:23:36 -08:00
shutdown_msg =
( struct shutdown_msg_data * ) & shut_txf_buf [
sizeof ( struct vmbuspipe_hdr ) +
sizeof ( struct icmsg_hdr ) ] ;
2010-05-04 15:55:05 -07:00
switch ( shutdown_msg - > flags ) {
case 0 :
case 1 :
icmsghdrp - > status = HV_S_OK ;
execute_shutdown = true ;
2011-03-29 13:58:49 -07:00
pr_info ( " Shutdown request received - "
2011-04-13 12:14:05 -07:00
" graceful shutdown initiated \n " ) ;
2010-05-04 15:55:05 -07:00
break ;
default :
icmsghdrp - > status = HV_E_FAIL ;
execute_shutdown = false ;
2011-03-29 13:58:49 -07:00
pr_info ( " Shutdown request received - "
" Invalid request \n " ) ;
2010-05-04 15:55:05 -07:00
break ;
2011-04-10 14:31:35 -07:00
}
2010-05-04 15:55:05 -07:00
}
icmsghdrp - > icflags = ICMSGHDRFLAG_TRANSACTION
| ICMSGHDRFLAG_RESPONSE ;
2010-12-13 16:23:36 -08:00
vmbus_sendpacket ( channel , shut_txf_buf ,
2010-05-04 15:55:05 -07:00
recvlen , requestid ,
2011-01-26 12:12:13 -08:00
VM_PKT_DATA_INBAND , 0 ) ;
2010-05-04 15:55:05 -07:00
}
if ( execute_shutdown = = true )
2011-08-27 11:31:30 -07:00
orderly_poweroff ( true ) ;
2010-05-04 15:55:05 -07:00
}
2010-05-05 19:23:46 +00:00
/*
2010-05-11 15:11:24 +00:00
* Set guest time to host UTC time .
2010-05-05 19:23:46 +00:00
*/
2010-05-11 15:11:24 +00:00
static inline void do_adj_guesttime ( u64 hosttime )
2010-05-05 19:23:46 +00:00
{
s64 host_tns ;
struct timespec host_ts ;
host_tns = ( hosttime - WLTIMEDELTA ) * 100 ;
host_ts = ns_to_timespec ( host_tns ) ;
2010-05-11 15:11:24 +00:00
do_settimeofday ( & host_ts ) ;
}
2011-08-27 11:31:31 -07:00
/*
* Set the host time in a process context .
*/
struct adj_time_work {
struct work_struct work ;
u64 host_time ;
} ;
static void hv_set_host_time ( struct work_struct * work )
{
struct adj_time_work * wrk ;
wrk = container_of ( work , struct adj_time_work , work ) ;
do_adj_guesttime ( wrk - > host_time ) ;
kfree ( wrk ) ;
}
2010-05-11 15:11:24 +00:00
/*
* Synchronize time with host after reboot , restore , etc .
*
* ICTIMESYNCFLAG_SYNC flag bit indicates reboot , restore events of the VM .
* After reboot the flag ICTIMESYNCFLAG_SYNC is included in the first time
* message after the timesync channel is opened . Since the hv_utils module is
* loaded after hv_vmbus , the first message is usually missed . The other
* thing is , systime is automatically set to emulated hardware clock which may
* not be UTC time or in the same time zone . So , to override these effects , we
* use the first 50 time samples for initial system time setting .
*/
static inline void adj_guesttime ( u64 hosttime , u8 flags )
{
2011-08-27 11:31:31 -07:00
struct adj_time_work * wrk ;
2010-05-11 15:11:24 +00:00
static s32 scnt = 50 ;
2011-08-27 11:31:31 -07:00
wrk = kmalloc ( sizeof ( struct adj_time_work ) , GFP_ATOMIC ) ;
if ( wrk = = NULL )
return ;
wrk - > host_time = hosttime ;
2010-05-05 19:23:46 +00:00
if ( ( flags & ICTIMESYNCFLAG_SYNC ) ! = 0 ) {
2011-08-27 11:31:31 -07:00
INIT_WORK ( & wrk - > work , hv_set_host_time ) ;
schedule_work ( & wrk - > work ) ;
2010-05-05 19:23:46 +00:00
return ;
}
2010-05-11 15:11:24 +00:00
if ( ( flags & ICTIMESYNCFLAG_SAMPLE ) ! = 0 & & scnt > 0 ) {
2010-05-05 19:23:46 +00:00
scnt - - ;
2011-08-27 11:31:31 -07:00
INIT_WORK ( & wrk - > work , hv_set_host_time ) ;
schedule_work ( & wrk - > work ) ;
} else
kfree ( wrk ) ;
2010-05-05 19:23:46 +00:00
}
/*
* Time Sync Channel message handler .
*/
static void timesync_onchannelcallback ( void * context )
{
struct vmbus_channel * channel = context ;
2010-12-13 16:23:36 -08:00
u32 recvlen ;
2010-05-05 19:23:46 +00:00
u64 requestid ;
struct icmsg_hdr * icmsghdrp ;
struct ictimesync_data * timedatap ;
2011-09-18 10:31:33 -07:00
u8 * time_txf_buf = util_timesynch . recv_buffer ;
2010-05-05 19:23:46 +00:00
2010-12-13 16:23:36 -08:00
vmbus_recvpacket ( channel , time_txf_buf ,
PAGE_SIZE , & recvlen , & requestid ) ;
2010-05-05 19:23:46 +00:00
if ( recvlen > 0 ) {
2010-12-13 16:23:36 -08:00
icmsghdrp = ( struct icmsg_hdr * ) & time_txf_buf [
2010-05-05 19:23:46 +00:00
sizeof ( struct vmbuspipe_hdr ) ] ;
if ( icmsghdrp - > icmsgtype = = ICMSGTYPE_NEGOTIATE ) {
2012-05-12 13:44:58 -07:00
vmbus_prep_negotiate_resp ( icmsghdrp , NULL , time_txf_buf ,
MAX_SRV_VER , MAX_SRV_VER ) ;
2010-05-05 19:23:46 +00:00
} else {
2010-12-13 16:23:36 -08:00
timedatap = ( struct ictimesync_data * ) & time_txf_buf [
2010-05-05 19:23:46 +00:00
sizeof ( struct vmbuspipe_hdr ) +
sizeof ( struct icmsg_hdr ) ] ;
adj_guesttime ( timedatap - > parenttime , timedatap - > flags ) ;
}
icmsghdrp - > icflags = ICMSGHDRFLAG_TRANSACTION
| ICMSGHDRFLAG_RESPONSE ;
2010-12-13 16:23:36 -08:00
vmbus_sendpacket ( channel , time_txf_buf ,
2010-05-05 19:23:46 +00:00
recvlen , requestid ,
2011-01-26 12:12:13 -08:00
VM_PKT_DATA_INBAND , 0 ) ;
2010-05-05 19:23:46 +00:00
}
}
2010-05-15 14:39:58 -07:00
/*
* Heartbeat functionality .
* Every two seconds , Hyper - V send us a heartbeat request message .
* we respond to this message , and Hyper - V knows we are alive .
*/
static void heartbeat_onchannelcallback ( void * context )
{
struct vmbus_channel * channel = context ;
2010-12-13 16:23:36 -08:00
u32 recvlen ;
2010-05-15 14:39:58 -07:00
u64 requestid ;
struct icmsg_hdr * icmsghdrp ;
struct heartbeat_msg_data * heartbeat_msg ;
2011-09-18 10:31:33 -07:00
u8 * hbeat_txf_buf = util_heartbeat . recv_buffer ;
2010-05-15 14:39:58 -07:00
2010-12-13 16:23:36 -08:00
vmbus_recvpacket ( channel , hbeat_txf_buf ,
PAGE_SIZE , & recvlen , & requestid ) ;
2010-05-15 14:39:58 -07:00
if ( recvlen > 0 ) {
2010-12-13 16:23:36 -08:00
icmsghdrp = ( struct icmsg_hdr * ) & hbeat_txf_buf [
2010-05-15 14:39:58 -07:00
sizeof ( struct vmbuspipe_hdr ) ] ;
if ( icmsghdrp - > icmsgtype = = ICMSGTYPE_NEGOTIATE ) {
2012-05-12 13:44:58 -07:00
vmbus_prep_negotiate_resp ( icmsghdrp , NULL ,
hbeat_txf_buf , MAX_SRV_VER , MAX_SRV_VER ) ;
2010-05-15 14:39:58 -07:00
} else {
2010-12-13 16:23:36 -08:00
heartbeat_msg =
( struct heartbeat_msg_data * ) & hbeat_txf_buf [
sizeof ( struct vmbuspipe_hdr ) +
sizeof ( struct icmsg_hdr ) ] ;
2010-05-15 14:39:58 -07:00
heartbeat_msg - > seq_num + = 1 ;
}
icmsghdrp - > icflags = ICMSGHDRFLAG_TRANSACTION
| ICMSGHDRFLAG_RESPONSE ;
2010-12-13 16:23:36 -08:00
vmbus_sendpacket ( channel , hbeat_txf_buf ,
2010-05-15 14:39:58 -07:00
recvlen , requestid ,
2011-01-26 12:12:13 -08:00
VM_PKT_DATA_INBAND , 0 ) ;
2010-05-15 14:39:58 -07:00
}
}
2010-05-05 19:23:46 +00:00
2011-09-13 10:59:38 -07:00
static int util_probe ( struct hv_device * dev ,
const struct hv_vmbus_device_id * dev_id )
2011-08-25 09:48:36 -07:00
{
2011-09-18 10:31:33 -07:00
struct hv_util_service * srv =
( struct hv_util_service * ) dev_id - > driver_data ;
int ret ;
2012-08-13 10:06:51 -07:00
srv - > recv_buffer = kmalloc ( PAGE_SIZE * 2 , GFP_KERNEL ) ;
2011-09-18 10:31:33 -07:00
if ( ! srv - > recv_buffer )
return - ENOMEM ;
if ( srv - > util_init ) {
ret = srv - > util_init ( srv ) ;
if ( ret ) {
2011-09-18 10:31:34 -07:00
ret = - ENODEV ;
goto error1 ;
2011-09-18 10:31:33 -07:00
}
}
2012-08-13 10:06:51 -07:00
ret = vmbus_open ( dev - > channel , 4 * PAGE_SIZE , 4 * PAGE_SIZE , NULL , 0 ,
2011-09-18 10:31:34 -07:00
srv - > util_cb , dev - > channel ) ;
if ( ret )
goto error ;
2011-09-18 10:31:33 -07:00
hv_set_drvdata ( dev , srv ) ;
2011-08-25 09:48:36 -07:00
return 0 ;
2011-09-18 10:31:34 -07:00
error :
if ( srv - > util_deinit )
srv - > util_deinit ( ) ;
error1 :
kfree ( srv - > recv_buffer ) ;
return ret ;
2011-08-25 09:48:36 -07:00
}
static int util_remove ( struct hv_device * dev )
{
2011-09-18 10:31:33 -07:00
struct hv_util_service * srv = hv_get_drvdata ( dev ) ;
2011-09-18 10:31:34 -07:00
vmbus_close ( dev - > channel ) ;
2011-09-18 10:31:33 -07:00
if ( srv - > util_deinit )
srv - > util_deinit ( ) ;
kfree ( srv - > recv_buffer ) ;
2011-08-25 09:48:36 -07:00
return 0 ;
}
static const struct hv_vmbus_device_id id_table [ ] = {
2011-08-25 11:41:33 -07:00
/* Shutdown guid */
{ VMBUS_DEVICE ( 0x31 , 0x60 , 0x0B , 0 X0E , 0x13 , 0x52 , 0x34 , 0x49 ,
2011-09-18 10:31:33 -07:00
0x81 , 0x8B , 0x38 , 0 XD9 , 0x0C , 0xED , 0x39 , 0xDB )
. driver_data = ( unsigned long ) & util_shutdown } ,
2011-08-25 11:41:33 -07:00
/* Time synch guid */
{ VMBUS_DEVICE ( 0x30 , 0xe6 , 0x27 , 0x95 , 0xae , 0xd0 , 0x7b , 0x49 ,
2011-09-18 10:31:33 -07:00
0xad , 0xce , 0xe8 , 0x0a , 0xb0 , 0x17 , 0x5c , 0xaf )
. driver_data = ( unsigned long ) & util_timesynch } ,
2011-08-25 11:41:33 -07:00
/* Heartbeat guid */
{ VMBUS_DEVICE ( 0x39 , 0x4f , 0x16 , 0x57 , 0x15 , 0x91 , 0x78 , 0x4e ,
2011-09-18 10:31:33 -07:00
0xab , 0x55 , 0x38 , 0x2f , 0x3b , 0xd5 , 0x42 , 0x2d )
. driver_data = ( unsigned long ) & util_heartbeat } ,
2011-08-25 11:41:33 -07:00
/* KVP guid */
{ VMBUS_DEVICE ( 0xe7 , 0xf4 , 0xa0 , 0xa9 , 0x45 , 0x5a , 0x96 , 0x4d ,
2011-09-18 10:31:33 -07:00
0xb8 , 0x27 , 0x8a , 0x84 , 0x1e , 0x8c , 0x3 , 0xe6 )
. driver_data = ( unsigned long ) & util_kvp } ,
2011-08-25 11:41:33 -07:00
{ } ,
2011-08-25 09:48:36 -07:00
} ;
MODULE_DEVICE_TABLE ( vmbus , id_table ) ;
/* The one and only one */
static struct hv_driver util_drv = {
2011-08-25 15:07:32 -07:00
. name = " hv_util " ,
2011-08-25 09:48:36 -07:00
. id_table = id_table ,
. probe = util_probe ,
. remove = util_remove ,
} ;
2010-05-04 15:55:05 -07:00
static int __init init_hyperv_utils ( void )
{
2011-03-29 13:58:49 -07:00
pr_info ( " Registering HyperV Utility Driver \n " ) ;
2010-05-04 15:55:05 -07:00
2011-09-18 10:31:34 -07:00
return vmbus_driver_register ( & util_drv ) ;
2010-05-04 15:55:05 -07:00
}
static void exit_hyperv_utils ( void )
{
2011-03-29 13:58:49 -07:00
pr_info ( " De-Registered HyperV Utility Driver \n " ) ;
2010-05-04 15:55:05 -07:00
2011-08-25 15:07:32 -07:00
vmbus_driver_unregister ( & util_drv ) ;
2010-05-04 15:55:05 -07:00
}
module_init ( init_hyperv_utils ) ;
module_exit ( exit_hyperv_utils ) ;
MODULE_DESCRIPTION ( " Hyper-V Utilities " ) ;
MODULE_VERSION ( HV_DRV_VERSION ) ;
MODULE_LICENSE ( " GPL " ) ;