2017-12-02 04:05:25 +09:00
/*
* Support for decoding of KVM_ * ioctl commands .
*
* Copyright ( c ) 2017 Masatake YAMATO < yamato @ redhat . com >
* Copyright ( c ) 2017 Red Hat , Inc .
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
# include "defs.h"
# ifdef HAVE_LINUX_KVM_H
# include <linux / kvm.h>
2017-12-04 22:08:14 +09:00
# include "print_fields.h"
2017-12-04 22:08:15 +09:00
# include "arch_kvm.c"
2017-12-02 04:05:25 +09:00
2017-12-02 04:05:28 +09:00
static int
kvm_ioctl_create_vcpu ( struct tcb * const tcp , const kernel_ulong_t arg )
{
uint32_t cpuid = arg ;
tprintf ( " , %u " , cpuid ) ;
return RVAL_IOCTL_DECODED | RVAL_FD ;
}
2017-12-04 22:08:14 +09:00
# ifdef HAVE_STRUCT_KVM_USERSPACE_MEMORY_REGION
# include "xlat / kvm_mem_flags.h"
static int
kvm_ioctl_set_user_memory_region ( struct tcb * const tcp , const kernel_ulong_t arg )
{
struct kvm_userspace_memory_region u_memory_region ;
tprints ( " , " ) ;
if ( umove_or_printaddr ( tcp , arg , & u_memory_region ) )
return RVAL_IOCTL_DECODED ;
PRINT_FIELD_U ( " { " , u_memory_region , slot ) ;
PRINT_FIELD_FLAGS ( " , " , u_memory_region , flags , kvm_mem_flags ,
" KVM_MEM_??? " ) ;
PRINT_FIELD_X ( " , " , u_memory_region , guest_phys_addr ) ;
PRINT_FIELD_U ( " , " , u_memory_region , memory_size ) ;
PRINT_FIELD_X ( " , " , u_memory_region , userspace_addr ) ;
tprints ( " } " ) ;
return RVAL_IOCTL_DECODED ;
}
# endif /* HAVE_STRUCT_KVM_USERSPACE_MEMORY_REGION */
2017-12-04 22:08:15 +09:00
# ifdef HAVE_STRUCT_KVM_REGS
static int
kvm_ioctl_decode_regs ( struct tcb * const tcp , const unsigned int code ,
const kernel_ulong_t arg )
{
struct kvm_regs regs ;
if ( code = = KVM_GET_REGS & & entering ( tcp ) )
return 0 ;
tprints ( " , " ) ;
if ( ! umove_or_printaddr ( tcp , arg , & regs ) )
arch_print_kvm_regs ( tcp , arg , & regs ) ;
return RVAL_IOCTL_DECODED ;
}
# endif /* HAVE_STRUCT_KVM_REGS */
2017-12-04 22:08:16 +09:00
# ifdef HAVE_STRUCT_KVM_SREGS
static int
kvm_ioctl_decode_sregs ( struct tcb * const tcp , const unsigned int code ,
const kernel_ulong_t arg )
{
struct kvm_sregs sregs ;
if ( code = = KVM_GET_SREGS & & entering ( tcp ) )
return 0 ;
tprints ( " , " ) ;
if ( ! umove_or_printaddr ( tcp , arg , & sregs ) )
arch_print_kvm_sregs ( tcp , arg , & sregs ) ;
return RVAL_IOCTL_DECODED ;
}
# endif /* HAVE_STRUCT_KVM_SREGS */
2017-12-02 04:05:25 +09:00
int
kvm_ioctl ( struct tcb * const tcp , const unsigned int code , const kernel_ulong_t arg )
{
switch ( code ) {
2017-12-02 04:05:28 +09:00
case KVM_CREATE_VCPU :
return kvm_ioctl_create_vcpu ( tcp , arg ) ;
2017-12-04 22:08:14 +09:00
# ifdef HAVE_STRUCT_KVM_USERSPACE_MEMORY_REGION
case KVM_SET_USER_MEMORY_REGION :
return kvm_ioctl_set_user_memory_region ( tcp , arg ) ;
# endif
2017-12-04 22:08:15 +09:00
# ifdef HAVE_STRUCT_KVM_REGS
case KVM_SET_REGS :
case KVM_GET_REGS :
return kvm_ioctl_decode_regs ( tcp , code , arg ) ;
# endif
2017-12-04 22:08:16 +09:00
# ifdef HAVE_STRUCT_KVM_SREGS
case KVM_SET_SREGS :
case KVM_GET_SREGS :
return kvm_ioctl_decode_sregs ( tcp , code , arg ) ;
# endif
2017-12-02 04:05:27 +09:00
case KVM_CREATE_VM :
return RVAL_DECODED | RVAL_FD ;
2017-12-02 04:05:26 +09:00
case KVM_RUN :
case KVM_GET_VCPU_MMAP_SIZE :
case KVM_GET_API_VERSION :
2017-12-02 04:05:25 +09:00
default :
return RVAL_DECODED ;
}
}
# endif /* HAVE_LINUX_KVM_H */