Commit 7f0bc6a6 authored by Jiri Slaby's avatar Jiri Slaby Committed by Greg Kroah-Hartman

TTY: pass flags to alloc_tty_driver

We need to allow drivers that use neither tty_port_install nor
tty_port_register_device to link a tty_port to a tty somehow.  To
avoid a race with open, this has to be performed before
tty_register_device. But currently tty_driver->ports is allocated even
in tty_register_device because we do not know whether this is the PTY
driver. The PTY driver is special here due to an excessive count of
lines it declares to handle. We cannot handle tty_ports there this
way.

To circumvent this, we start passing tty_driver flags to
alloc_tty_driver already and we create tty_alloc_driver for this
purpose. There we can allocate tty_driver->ports and do all the magic
between tty_alloc_driver and tty_register_device. Later we will
introduce tty_port_link_device function for that purpose.

All drivers should eventually switch to this new tty driver allocation
interface.
Signed-off-by: default avatarJiri Slaby <jslaby@suse.cz>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 2312e4f3
...@@ -3061,21 +3061,37 @@ void tty_unregister_device(struct tty_driver *driver, unsigned index) ...@@ -3061,21 +3061,37 @@ void tty_unregister_device(struct tty_driver *driver, unsigned index)
} }
EXPORT_SYMBOL(tty_unregister_device); EXPORT_SYMBOL(tty_unregister_device);
struct tty_driver *__alloc_tty_driver(int lines, struct module *owner) /**
* __tty_alloc_driver -- allocate tty driver
* @lines: count of lines this driver can handle at most
* @owner: module which is repsonsible for this driver
* @flags: some of TTY_DRIVER_* flags, will be set in driver->flags
*
* This should not be called directly, some of the provided macros should be
* used instead. Use IS_ERR and friends on @retval.
*/
struct tty_driver *__tty_alloc_driver(unsigned int lines, struct module *owner,
unsigned long flags)
{ {
struct tty_driver *driver; struct tty_driver *driver;
if (!lines)
return ERR_PTR(-EINVAL);
driver = kzalloc(sizeof(struct tty_driver), GFP_KERNEL); driver = kzalloc(sizeof(struct tty_driver), GFP_KERNEL);
if (driver) { if (!driver)
kref_init(&driver->kref); return ERR_PTR(-ENOMEM);
driver->magic = TTY_DRIVER_MAGIC;
driver->num = lines; kref_init(&driver->kref);
driver->owner = owner; driver->magic = TTY_DRIVER_MAGIC;
/* later we'll move allocation of tables here */ driver->num = lines;
} driver->owner = owner;
driver->flags = flags;
/* later we'll move allocation of tables here */
return driver; return driver;
} }
EXPORT_SYMBOL(__alloc_tty_driver); EXPORT_SYMBOL(__tty_alloc_driver);
static void destruct_tty_driver(struct kref *kref) static void destruct_tty_driver(struct kref *kref)
{ {
......
...@@ -296,11 +296,11 @@ struct tty_driver { ...@@ -296,11 +296,11 @@ struct tty_driver {
int name_base; /* offset of printed name */ int name_base; /* offset of printed name */
int major; /* major device number */ int major; /* major device number */
int minor_start; /* start of minor device number */ int minor_start; /* start of minor device number */
int num; /* number of devices allocated */ unsigned int num; /* number of devices allocated */
short type; /* type of tty driver */ short type; /* type of tty driver */
short subtype; /* subtype of tty driver */ short subtype; /* subtype of tty driver */
struct ktermios init_termios; /* Initial termios */ struct ktermios init_termios; /* Initial termios */
int flags; /* tty driver flags */ unsigned long flags; /* tty driver flags */
struct proc_dir_entry *proc_entry; /* /proc fs entry */ struct proc_dir_entry *proc_entry; /* /proc fs entry */
struct tty_driver *other; /* only used for the PTY driver */ struct tty_driver *other; /* only used for the PTY driver */
...@@ -322,7 +322,8 @@ struct tty_driver { ...@@ -322,7 +322,8 @@ struct tty_driver {
extern struct list_head tty_drivers; extern struct list_head tty_drivers;
extern struct tty_driver *__alloc_tty_driver(int lines, struct module *owner); extern struct tty_driver *__tty_alloc_driver(unsigned int lines,
struct module *owner, unsigned long flags);
extern void put_tty_driver(struct tty_driver *driver); extern void put_tty_driver(struct tty_driver *driver);
extern void tty_set_operations(struct tty_driver *driver, extern void tty_set_operations(struct tty_driver *driver,
const struct tty_operations *op); const struct tty_operations *op);
...@@ -330,7 +331,21 @@ extern struct tty_driver *tty_find_polling_driver(char *name, int *line); ...@@ -330,7 +331,21 @@ extern struct tty_driver *tty_find_polling_driver(char *name, int *line);
extern void tty_driver_kref_put(struct tty_driver *driver); extern void tty_driver_kref_put(struct tty_driver *driver);
#define alloc_tty_driver(lines) __alloc_tty_driver(lines, THIS_MODULE) /* Use TTY_DRIVER_* flags below */
#define tty_alloc_driver(lines, flags) \
__tty_alloc_driver(lines, THIS_MODULE, flags)
/*
* DEPRECATED Do not use this in new code, use tty_alloc_driver instead.
* (And change the return value checks.)
*/
static inline struct tty_driver *alloc_tty_driver(unsigned int lines)
{
struct tty_driver *ret = tty_alloc_driver(lines, 0);
if (IS_ERR(ret))
return NULL;
return ret;
}
static inline struct tty_driver *tty_driver_kref_get(struct tty_driver *d) static inline struct tty_driver *tty_driver_kref_get(struct tty_driver *d)
{ {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment