2019-05-27 09:55:05 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-17 02:20:36 +04:00
/*
* 32 bit - > 64 bit ioctl wrapper for raw MIDI API
* Copyright ( c ) by Takashi Iwai < tiwai @ suse . de >
*/
/* This file included from rawmidi.c */
# include <linux/compat.h>
2005-11-17 15:56:51 +03:00
struct snd_rawmidi_params32 {
2005-04-17 02:20:36 +04:00
s32 stream ;
u32 buffer_size ;
u32 avail_min ;
unsigned int no_active_sensing ; /* avoid bit-field */
unsigned char reserved [ 16 ] ;
} __attribute__ ( ( packed ) ) ;
2005-11-17 15:56:51 +03:00
static int snd_rawmidi_ioctl_params_compat ( struct snd_rawmidi_file * rfile ,
struct snd_rawmidi_params32 __user * src )
2005-04-17 02:20:36 +04:00
{
2005-11-17 15:56:51 +03:00
struct snd_rawmidi_params params ;
2005-04-17 02:20:36 +04:00
unsigned int val ;
if ( get_user ( params . stream , & src - > stream ) | |
get_user ( params . buffer_size , & src - > buffer_size ) | |
get_user ( params . avail_min , & src - > avail_min ) | |
get_user ( val , & src - > no_active_sensing ) )
return - EFAULT ;
params . no_active_sensing = val ;
switch ( params . stream ) {
case SNDRV_RAWMIDI_STREAM_OUTPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > output )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
return snd_rawmidi_output_params ( rfile - > output , & params ) ;
case SNDRV_RAWMIDI_STREAM_INPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > input )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
return snd_rawmidi_input_params ( rfile - > input , & params ) ;
}
return - EINVAL ;
}
2005-11-17 15:56:51 +03:00
struct snd_rawmidi_status32 {
2005-04-17 02:20:36 +04:00
s32 stream ;
struct compat_timespec tstamp ;
u32 avail ;
u32 xruns ;
unsigned char reserved [ 16 ] ;
} __attribute__ ( ( packed ) ) ;
2005-11-17 15:56:51 +03:00
static int snd_rawmidi_ioctl_status_compat ( struct snd_rawmidi_file * rfile ,
struct snd_rawmidi_status32 __user * src )
2005-04-17 02:20:36 +04:00
{
int err ;
2005-11-17 15:56:51 +03:00
struct snd_rawmidi_status status ;
2005-04-17 02:20:36 +04:00
if ( get_user ( status . stream , & src - > stream ) )
return - EFAULT ;
switch ( status . stream ) {
case SNDRV_RAWMIDI_STREAM_OUTPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > output )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
err = snd_rawmidi_output_status ( rfile - > output , & status ) ;
break ;
case SNDRV_RAWMIDI_STREAM_INPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > input )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
err = snd_rawmidi_input_status ( rfile - > input , & status ) ;
break ;
default :
return - EINVAL ;
}
if ( err < 0 )
return err ;
2016-02-28 13:30:53 +03:00
if ( compat_put_timespec ( & status . tstamp , & src - > tstamp ) | |
2005-04-17 02:20:36 +04:00
put_user ( status . avail , & src - > avail ) | |
put_user ( status . xruns , & src - > xruns ) )
return - EFAULT ;
return 0 ;
}
2016-02-28 13:28:08 +03:00
# ifdef CONFIG_X86_X32
/* X32 ABI has 64bit timespec and 64bit alignment */
struct snd_rawmidi_status_x32 {
s32 stream ;
u32 rsvd ; /* alignment */
struct timespec tstamp ;
u32 avail ;
u32 xruns ;
unsigned char reserved [ 16 ] ;
} __attribute__ ( ( packed ) ) ;
# define put_timespec(src, dst) copy_to_user(dst, src, sizeof(*dst))
static int snd_rawmidi_ioctl_status_x32 ( struct snd_rawmidi_file * rfile ,
struct snd_rawmidi_status_x32 __user * src )
{
int err ;
struct snd_rawmidi_status status ;
if ( get_user ( status . stream , & src - > stream ) )
return - EFAULT ;
switch ( status . stream ) {
case SNDRV_RAWMIDI_STREAM_OUTPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > output )
return - EINVAL ;
2016-02-28 13:28:08 +03:00
err = snd_rawmidi_output_status ( rfile - > output , & status ) ;
break ;
case SNDRV_RAWMIDI_STREAM_INPUT :
2018-04-19 19:16:15 +03:00
if ( ! rfile - > input )
return - EINVAL ;
2016-02-28 13:28:08 +03:00
err = snd_rawmidi_input_status ( rfile - > input , & status ) ;
break ;
default :
return - EINVAL ;
}
if ( err < 0 )
return err ;
if ( put_timespec ( & status . tstamp , & src - > tstamp ) | |
put_user ( status . avail , & src - > avail ) | |
put_user ( status . xruns , & src - > xruns ) )
return - EFAULT ;
return 0 ;
}
# endif /* CONFIG_X86_X32 */
2005-04-17 02:20:36 +04:00
enum {
2005-11-17 15:56:51 +03:00
SNDRV_RAWMIDI_IOCTL_PARAMS32 = _IOWR ( ' W ' , 0x10 , struct snd_rawmidi_params32 ) ,
SNDRV_RAWMIDI_IOCTL_STATUS32 = _IOWR ( ' W ' , 0x20 , struct snd_rawmidi_status32 ) ,
2016-02-28 13:28:08 +03:00
# ifdef CONFIG_X86_X32
SNDRV_RAWMIDI_IOCTL_STATUS_X32 = _IOWR ( ' W ' , 0x20 , struct snd_rawmidi_status_x32 ) ,
# endif /* CONFIG_X86_X32 */
2005-04-17 02:20:36 +04:00
} ;
static long snd_rawmidi_ioctl_compat ( struct file * file , unsigned int cmd , unsigned long arg )
{
2005-11-17 15:56:51 +03:00
struct snd_rawmidi_file * rfile ;
2005-04-17 02:20:36 +04:00
void __user * argp = compat_ptr ( arg ) ;
rfile = file - > private_data ;
switch ( cmd ) {
case SNDRV_RAWMIDI_IOCTL_PVERSION :
case SNDRV_RAWMIDI_IOCTL_INFO :
case SNDRV_RAWMIDI_IOCTL_DROP :
case SNDRV_RAWMIDI_IOCTL_DRAIN :
return snd_rawmidi_ioctl ( file , cmd , ( unsigned long ) argp ) ;
case SNDRV_RAWMIDI_IOCTL_PARAMS32 :
return snd_rawmidi_ioctl_params_compat ( rfile , argp ) ;
case SNDRV_RAWMIDI_IOCTL_STATUS32 :
return snd_rawmidi_ioctl_status_compat ( rfile , argp ) ;
2016-02-28 13:28:08 +03:00
# ifdef CONFIG_X86_X32
case SNDRV_RAWMIDI_IOCTL_STATUS_X32 :
return snd_rawmidi_ioctl_status_x32 ( rfile , argp ) ;
# endif /* CONFIG_X86_X32 */
2005-04-17 02:20:36 +04:00
}
return - ENOIOCTLCMD ;
}