[PATCH] usb/serial: Add compat_ioctl pass-through

From: Keith Packard
Date: Thu Nov 27 2008 - 03:29:57 EST


USB serial devices with extended IOCTLs cannot be used in a 64-bit kernel
from 32-bit user space as the compat_ioctl path is missing. This adds a
pass-through so that drivers may offer this functionality. This requires
that all drivers actually implement a compat_ioctl function if they want to
support this operation.

Signed-off-by: Keith Packard <keithp@xxxxxxxxxx>
---
drivers/usb/serial/usb-serial.c | 22 ++++++++++++++++++++++
include/linux/usb/serial.h | 2 ++
2 files changed, 24 insertions(+), 0 deletions(-)

diff --git a/drivers/usb/serial/usb-serial.c b/drivers/usb/serial/usb-serial.c
index 794b5ff..8fb4e13 100644
--- a/drivers/usb/serial/usb-serial.c
+++ b/drivers/usb/serial/usb-serial.c
@@ -381,6 +381,27 @@ static int serial_ioctl(struct tty_struct *tty, struct file *file,
return retval;
}

+static long serial_compat_ioctl(struct tty_struct *tty, struct file *file,
+ unsigned int cmd, unsigned long arg)
+{
+ struct usb_serial_port *port = tty->driver_data;
+ long retval = -ENODEV;
+
+ dbg("%s - port %d, cmd 0x%.4x", __func__, port->number, cmd);
+
+ WARN_ON(!port->port.count);
+
+ /* pass on to the driver specific version of this function
+ if it is available */
+ if (port->serial->type->compat_ioctl) {
+ lock_kernel();
+ retval = port->serial->type->compat_ioctl(tty, file, cmd, arg);
+ unlock_kernel();
+ } else
+ retval = -ENOIOCTLCMD;
+ return retval;
+}
+
static void serial_set_termios(struct tty_struct *tty, struct ktermios *old)
{
struct usb_serial_port *port = tty->driver_data;
@@ -1095,6 +1116,7 @@ static const struct tty_operations serial_ops = {
.write = serial_write,
.write_room = serial_write_room,
.ioctl = serial_ioctl,
+ .compat_ioctl = serial_compat_ioctl,
.set_termios = serial_set_termios,
.throttle = serial_throttle,
.unthrottle = serial_unthrottle,
diff --git a/include/linux/usb/serial.h b/include/linux/usb/serial.h
index 0b8617a..c15766a 100644
--- a/include/linux/usb/serial.h
+++ b/include/linux/usb/serial.h
@@ -228,6 +228,8 @@ struct usb_serial_driver {
int (*write_room)(struct tty_struct *tty);
int (*ioctl)(struct tty_struct *tty, struct file *file,
unsigned int cmd, unsigned long arg);
+ long (*compat_ioctl)(struct tty_struct *tty, struct file *file,
+ unsigned int cmd, unsigned long arg);
void (*set_termios)(struct tty_struct *tty,
struct usb_serial_port *port, struct ktermios *old);
void (*break_ctl)(struct tty_struct *tty, int break_state);
--
1.5.6.5

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/