mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-17 18:56:24 +00:00
7e1621de14
In ALSA firewire stack, some AV/C commands are supported, including vendor's extensions. Drivers includes response parser of each command, according to its requirements, while the parser is written with loose fashion in two points; error check and length check. This doesn't cause any issues such as kernel corruption, but should be improved. This commit modifies evaluations of return value on each parsers. Reported-by: Dan Carpenter <dan.carpenter@oracle.com> Signed-off-by: Takashi Sakamoto <o-takashi@sakamocchi.jp> Signed-off-by: Takashi Iwai <tiwai@suse.de>
297 lines
6.9 KiB
C
297 lines
6.9 KiB
C
/*
|
|
* bebob_command.c - driver for BeBoB based devices
|
|
*
|
|
* Copyright (c) 2013-2014 Takashi Sakamoto
|
|
*
|
|
* Licensed under the terms of the GNU General Public License, version 2.
|
|
*/
|
|
|
|
#include "./bebob.h"
|
|
|
|
int avc_audio_set_selector(struct fw_unit *unit, unsigned int subunit_id,
|
|
unsigned int fb_id, unsigned int num)
|
|
{
|
|
u8 *buf;
|
|
int err;
|
|
|
|
buf = kzalloc(12, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return -ENOMEM;
|
|
|
|
buf[0] = 0x00; /* AV/C CONTROL */
|
|
buf[1] = 0x08 | (0x07 & subunit_id); /* AUDIO SUBUNIT ID */
|
|
buf[2] = 0xb8; /* FUNCTION BLOCK */
|
|
buf[3] = 0x80; /* type is 'selector'*/
|
|
buf[4] = 0xff & fb_id; /* function block id */
|
|
buf[5] = 0x10; /* control attribute is CURRENT */
|
|
buf[6] = 0x02; /* selector length is 2 */
|
|
buf[7] = 0xff & num; /* input function block plug number */
|
|
buf[8] = 0x01; /* control selector is SELECTOR_CONTROL */
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(7) | BIT(8));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 9)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else
|
|
err = 0;
|
|
|
|
kfree(buf);
|
|
return err;
|
|
}
|
|
|
|
int avc_audio_get_selector(struct fw_unit *unit, unsigned int subunit_id,
|
|
unsigned int fb_id, unsigned int *num)
|
|
{
|
|
u8 *buf;
|
|
int err;
|
|
|
|
buf = kzalloc(12, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return -ENOMEM;
|
|
|
|
buf[0] = 0x01; /* AV/C STATUS */
|
|
buf[1] = 0x08 | (0x07 & subunit_id); /* AUDIO SUBUNIT ID */
|
|
buf[2] = 0xb8; /* FUNCTION BLOCK */
|
|
buf[3] = 0x80; /* type is 'selector'*/
|
|
buf[4] = 0xff & fb_id; /* function block id */
|
|
buf[5] = 0x10; /* control attribute is CURRENT */
|
|
buf[6] = 0x02; /* selector length is 2 */
|
|
buf[7] = 0xff; /* input function block plug number */
|
|
buf[8] = 0x01; /* control selector is SELECTOR_CONTROL */
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(8));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 9)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
*num = buf[7];
|
|
err = 0;
|
|
end:
|
|
kfree(buf);
|
|
return err;
|
|
}
|
|
|
|
static inline void
|
|
avc_bridgeco_fill_extension_addr(u8 *buf, u8 *addr)
|
|
{
|
|
buf[1] = addr[0];
|
|
memcpy(buf + 4, addr + 1, 5);
|
|
}
|
|
|
|
static inline void
|
|
avc_bridgeco_fill_plug_info_extension_command(u8 *buf, u8 *addr,
|
|
unsigned int itype)
|
|
{
|
|
buf[0] = 0x01; /* AV/C STATUS */
|
|
buf[2] = 0x02; /* AV/C GENERAL PLUG INFO */
|
|
buf[3] = 0xc0; /* BridgeCo extension */
|
|
avc_bridgeco_fill_extension_addr(buf, addr);
|
|
buf[9] = itype; /* info type */
|
|
}
|
|
|
|
int avc_bridgeco_get_plug_type(struct fw_unit *unit,
|
|
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
|
enum avc_bridgeco_plug_type *type)
|
|
{
|
|
u8 *buf;
|
|
int err;
|
|
|
|
buf = kzalloc(12, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return -ENOMEM;
|
|
|
|
/* Info type is 'plug type'. */
|
|
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x00);
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(7) | BIT(9));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 11)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
*type = buf[10];
|
|
err = 0;
|
|
end:
|
|
kfree(buf);
|
|
return err;
|
|
}
|
|
|
|
int avc_bridgeco_get_plug_ch_pos(struct fw_unit *unit,
|
|
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
|
u8 *buf, unsigned int len)
|
|
{
|
|
int err;
|
|
|
|
/* Info type is 'channel position'. */
|
|
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x03);
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, 256,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) |
|
|
BIT(5) | BIT(6) | BIT(7) | BIT(9));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 11)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
/* Pick up specific data. */
|
|
memmove(buf, buf + 10, err - 10);
|
|
err = 0;
|
|
end:
|
|
return err;
|
|
}
|
|
|
|
int avc_bridgeco_get_plug_section_type(struct fw_unit *unit,
|
|
u8 addr[AVC_BRIDGECO_ADDR_BYTES],
|
|
unsigned int id, u8 *type)
|
|
{
|
|
u8 *buf;
|
|
int err;
|
|
|
|
/* section info includes charactors but this module don't need it */
|
|
buf = kzalloc(12, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return -ENOMEM;
|
|
|
|
/* Info type is 'section info'. */
|
|
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x07);
|
|
buf[10] = 0xff & ++id; /* section id */
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, 12,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(7) | BIT(9) | BIT(10));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 12)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
*type = buf[11];
|
|
err = 0;
|
|
end:
|
|
kfree(buf);
|
|
return err;
|
|
}
|
|
|
|
int avc_bridgeco_get_plug_input(struct fw_unit *unit,
|
|
u8 addr[AVC_BRIDGECO_ADDR_BYTES], u8 input[7])
|
|
{
|
|
int err;
|
|
u8 *buf;
|
|
|
|
buf = kzalloc(18, GFP_KERNEL);
|
|
if (buf == NULL)
|
|
return -ENOMEM;
|
|
|
|
/* Info type is 'plug input'. */
|
|
avc_bridgeco_fill_plug_info_extension_command(buf, addr, 0x05);
|
|
|
|
err = fcp_avc_transaction(unit, buf, 16, buf, 16,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(7));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 16)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
memcpy(input, buf + 10, 5);
|
|
err = 0;
|
|
end:
|
|
kfree(buf);
|
|
return err;
|
|
}
|
|
|
|
int avc_bridgeco_get_plug_strm_fmt(struct fw_unit *unit,
|
|
u8 addr[AVC_BRIDGECO_ADDR_BYTES], u8 *buf,
|
|
unsigned int *len, unsigned int eid)
|
|
{
|
|
int err;
|
|
|
|
/* check given buffer */
|
|
if ((buf == NULL) || (*len < 12)) {
|
|
err = -EINVAL;
|
|
goto end;
|
|
}
|
|
|
|
buf[0] = 0x01; /* AV/C STATUS */
|
|
buf[2] = 0x2f; /* AV/C STREAM FORMAT SUPPORT */
|
|
buf[3] = 0xc1; /* Bridgeco extension - List Request */
|
|
avc_bridgeco_fill_extension_addr(buf, addr);
|
|
buf[10] = 0xff & eid; /* Entry ID */
|
|
|
|
err = fcp_avc_transaction(unit, buf, 12, buf, *len,
|
|
BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) |
|
|
BIT(6) | BIT(7) | BIT(10));
|
|
if (err < 0)
|
|
;
|
|
else if (err < 12)
|
|
err = -EIO;
|
|
else if (buf[0] == 0x08) /* NOT IMPLEMENTED */
|
|
err = -ENOSYS;
|
|
else if (buf[0] == 0x0a) /* REJECTED */
|
|
err = -EINVAL;
|
|
else if (buf[0] == 0x0b) /* IN TRANSITION */
|
|
err = -EAGAIN;
|
|
else if (buf[10] != eid)
|
|
err = -EIO;
|
|
if (err < 0)
|
|
goto end;
|
|
|
|
/* Pick up 'stream format info'. */
|
|
memmove(buf, buf + 11, err - 11);
|
|
*len = err - 11;
|
|
err = 0;
|
|
end:
|
|
return err;
|
|
}
|