Commit 7c3f0d3d authored by Takashi Iwai's avatar Takashi Iwai

ALSA: seq: Check the conflicting port at port creation

We didn't check if a port with the given port number has been already
present at creating a new port.  Check it and return -EBUSY properly
if the port number conflicts.
Reviewed-by: default avatarJaroslav Kysela <perex@perex.cz>
Link: https://lore.kernel.org/r/20230523075358.9672-22-tiwai@suse.deSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent 94c5b717
...@@ -1194,15 +1194,19 @@ static int snd_seq_ioctl_create_port(struct snd_seq_client *client, void *arg) ...@@ -1194,15 +1194,19 @@ static int snd_seq_ioctl_create_port(struct snd_seq_client *client, void *arg)
struct snd_seq_port_info *info = arg; struct snd_seq_port_info *info = arg;
struct snd_seq_client_port *port; struct snd_seq_client_port *port;
struct snd_seq_port_callback *callback; struct snd_seq_port_callback *callback;
int port_idx; int port_idx, err;
/* it is not allowed to create the port for an another client */ /* it is not allowed to create the port for an another client */
if (info->addr.client != client->number) if (info->addr.client != client->number)
return -EPERM; return -EPERM;
port = snd_seq_create_port(client, (info->flags & SNDRV_SEQ_PORT_FLG_GIVEN_PORT) ? info->addr.port : -1); if (info->flags & SNDRV_SEQ_PORT_FLG_GIVEN_PORT)
if (port == NULL) port_idx = info->addr.port;
return -ENOMEM; else
port_idx = -1;
err = snd_seq_create_port(client, port_idx, &port);
if (err < 0)
return err;
if (client->type == USER_CLIENT && info->kernel) { if (client->type == USER_CLIENT && info->kernel) {
port_idx = port->addr.port; port_idx = port->addr.port;
......
...@@ -107,33 +107,34 @@ static void port_subs_info_init(struct snd_seq_port_subs_info *grp) ...@@ -107,33 +107,34 @@ static void port_subs_info_init(struct snd_seq_port_subs_info *grp)
} }
/* create a port, port number is returned (-1 on failure); /* create a port, port number or a negative error code is returned
* the caller needs to unref the port via snd_seq_port_unlock() appropriately * the caller needs to unref the port via snd_seq_port_unlock() appropriately
*/ */
struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client, int snd_seq_create_port(struct snd_seq_client *client, int port,
int port) struct snd_seq_client_port **port_ret)
{ {
struct snd_seq_client_port *new_port, *p; struct snd_seq_client_port *new_port, *p;
int num = -1; int num;
*port_ret = NULL;
/* sanity check */ /* sanity check */
if (snd_BUG_ON(!client)) if (snd_BUG_ON(!client))
return NULL; return -EINVAL;
if (client->num_ports >= SNDRV_SEQ_MAX_PORTS) { if (client->num_ports >= SNDRV_SEQ_MAX_PORTS) {
pr_warn("ALSA: seq: too many ports for client %d\n", client->number); pr_warn("ALSA: seq: too many ports for client %d\n", client->number);
return NULL; return -EINVAL;
} }
/* create a new port */ /* create a new port */
new_port = kzalloc(sizeof(*new_port), GFP_KERNEL); new_port = kzalloc(sizeof(*new_port), GFP_KERNEL);
if (!new_port) if (!new_port)
return NULL; /* failure, out of memory */ return -ENOMEM; /* failure, out of memory */
/* init port data */ /* init port data */
new_port->addr.client = client->number; new_port->addr.client = client->number;
new_port->addr.port = -1; new_port->addr.port = -1;
new_port->owner = THIS_MODULE; new_port->owner = THIS_MODULE;
sprintf(new_port->name, "port-%d", num);
snd_use_lock_init(&new_port->use_lock); snd_use_lock_init(&new_port->use_lock);
port_subs_info_init(&new_port->c_src); port_subs_info_init(&new_port->c_src);
port_subs_info_init(&new_port->c_dest); port_subs_info_init(&new_port->c_dest);
...@@ -143,6 +144,10 @@ struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client, ...@@ -143,6 +144,10 @@ struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client,
mutex_lock(&client->ports_mutex); mutex_lock(&client->ports_mutex);
write_lock_irq(&client->ports_lock); write_lock_irq(&client->ports_lock);
list_for_each_entry(p, &client->ports_list_head, list) { list_for_each_entry(p, &client->ports_list_head, list) {
if (p->addr.port == port) {
num = -EBUSY;
goto unlock;
}
if (p->addr.port > num) if (p->addr.port > num)
break; break;
if (port < 0) /* auto-probe mode */ if (port < 0) /* auto-probe mode */
...@@ -153,10 +158,12 @@ struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client, ...@@ -153,10 +158,12 @@ struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client,
client->num_ports++; client->num_ports++;
new_port->addr.port = num; /* store the port number in the port */ new_port->addr.port = num; /* store the port number in the port */
sprintf(new_port->name, "port-%d", num); sprintf(new_port->name, "port-%d", num);
*port_ret = new_port;
unlock:
write_unlock_irq(&client->ports_lock); write_unlock_irq(&client->ports_lock);
mutex_unlock(&client->ports_mutex); mutex_unlock(&client->ports_mutex);
return new_port; return num;
} }
/* */ /* */
......
...@@ -86,8 +86,9 @@ struct snd_seq_client_port *snd_seq_port_query_nearest(struct snd_seq_client *cl ...@@ -86,8 +86,9 @@ struct snd_seq_client_port *snd_seq_port_query_nearest(struct snd_seq_client *cl
/* unlock the port */ /* unlock the port */
#define snd_seq_port_unlock(port) snd_use_lock_free(&(port)->use_lock) #define snd_seq_port_unlock(port) snd_use_lock_free(&(port)->use_lock)
/* create a port, port number is returned (-1 on failure) */ /* create a port, port number or a negative error code is returned */
struct snd_seq_client_port *snd_seq_create_port(struct snd_seq_client *client, int port_index); int snd_seq_create_port(struct snd_seq_client *client, int port_index,
struct snd_seq_client_port **port_ret);
/* delete a port */ /* delete a port */
int snd_seq_delete_port(struct snd_seq_client *client, int port); int snd_seq_delete_port(struct snd_seq_client *client, int port);
......
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