[media] drx-j: Split firmware size check from the main routine

The firmware upload routine is already complex enough. Split the
first loop that verifies the firmware size into a separate routine,
making the code more readable.

Acked-by: Devin Heitmueller <dheitmueller@kernellabs.com>
Signed-off-by: Mauro Carvalho Chehab <m.chehab@samsung.com>
This commit is contained in:
Mauro Carvalho Chehab 2014-01-18 19:11:11 -03:00
parent b48293db4a
commit 2cd63e487c
1 changed files with 122 additions and 93 deletions

View File

@ -933,6 +933,86 @@ static u16 u_code_compute_crc(u8 *block_data, u16 nr_words)
/*============================================================================*/ /*============================================================================*/
static int check_firmware(struct drx_demod_instance *demod, u8 *mc_data,
unsigned size)
{
struct drxu_code_block_hdr block_hdr;
int i;
unsigned count = 2 * sizeof(u16);
u32 mc_dev_type, mc_version, mc_base_version;
u16 mc_nr_of_blks = u_code_read16(mc_data + sizeof(u16));
/*
* Scan microcode blocks first for version info
* and firmware check
*/
/* Clear version block */
DRX_ATTR_MCRECORD(demod).aux_type = 0;
DRX_ATTR_MCRECORD(demod).mc_dev_type = 0;
DRX_ATTR_MCRECORD(demod).mc_version = 0;
DRX_ATTR_MCRECORD(demod).mc_base_version = 0;
for (i = 0; i < mc_nr_of_blks; i++) {
if (count + 3 * sizeof(u16) + sizeof(u32) > size)
goto eof;
/* Process block header */
block_hdr.addr = u_code_read32(mc_data + count);
count += sizeof(u32);
block_hdr.size = u_code_read16(mc_data + count);
count += sizeof(u16);
block_hdr.flags = u_code_read16(mc_data + count);
count += sizeof(u16);
block_hdr.CRC = u_code_read16(mc_data + count);
count += sizeof(u16);
pr_debug("%u: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
count, block_hdr.addr, block_hdr.size, block_hdr.flags,
block_hdr.CRC);
if (block_hdr.flags & 0x8) {
u8 *auxblk = ((void *)mc_data) + block_hdr.addr;
u16 auxtype;
if (block_hdr.addr + sizeof(u16) > size)
goto eof;
auxtype = u_code_read16(auxblk);
/* Aux block. Check type */
if (DRX_ISMCVERTYPE(auxtype)) {
if (block_hdr.addr + 2 * sizeof(u16) + 2 * sizeof (u32) > size)
goto eof;
auxblk += sizeof(u16);
mc_dev_type = u_code_read32(auxblk);
auxblk += sizeof(u32);
mc_version = u_code_read32(auxblk);
auxblk += sizeof(u32);
mc_base_version = u_code_read32(auxblk);
DRX_ATTR_MCRECORD(demod).aux_type = auxtype;
DRX_ATTR_MCRECORD(demod).mc_dev_type = mc_dev_type;
DRX_ATTR_MCRECORD(demod).mc_version = mc_version;
DRX_ATTR_MCRECORD(demod).mc_base_version = mc_base_version;
pr_info("Firmware dev %x, ver %x, base ver %x\n",
mc_dev_type, mc_version, mc_base_version);
}
} else if (count + block_hdr.size * sizeof(u16) > size)
goto eof;
count += block_hdr.size * sizeof(u16);
}
return 0;
eof:
pr_err("Firmware is truncated at pos %u/%u\n", count, size);
return -EINVAL;
}
/** /**
* \brief Handle microcode upload or verify. * \brief Handle microcode upload or verify.
* \param dev_addr: Address of device. * \param dev_addr: Address of device.
@ -962,24 +1042,15 @@ ctrl_u_code(struct drx_demod_instance *demod,
u16 mc_magic_word = 0; u16 mc_magic_word = 0;
const u8 *mc_data_init = NULL; const u8 *mc_data_init = NULL;
u8 *mc_data = NULL; u8 *mc_data = NULL;
unsigned size;
char *mc_file = mc_info->mc_file; char *mc_file = mc_info->mc_file;
/* Check arguments */ /* Check arguments */
if (!mc_info || !mc_file) if (!mc_info || !mc_file)
return -EINVAL; return -EINVAL;
if (demod->firmware) { if (!demod->firmware) {
mc_data_init = demod->firmware->data;
mc_data = (void *)mc_data_init;
/* Check data */
mc_magic_word = u_code_read16(mc_data);
mc_data += sizeof(u16);
mc_nr_of_blks = u_code_read16(mc_data);
mc_data += sizeof(u16);
} else {
const struct firmware *fw = NULL; const struct firmware *fw = NULL;
unsigned size = 0;
rc = request_firmware(&fw, mc_file, demod->i2c->dev.parent); rc = request_firmware(&fw, mc_file, demod->i2c->dev.parent);
if (rc < 0) { if (rc < 0) {
@ -987,95 +1058,49 @@ ctrl_u_code(struct drx_demod_instance *demod,
return -ENOENT; return -ENOENT;
} }
demod->firmware = fw; demod->firmware = fw;
mc_data_init = demod->firmware->data;
size = demod->firmware->size;
pr_info("Firmware %s, size %u\n", mc_file, size); if (demod->firmware->size < 2 * sizeof(u16)) {
rc = -EINVAL;
mc_data = (void *)mc_data_init; pr_err("Firmware is too short!\n");
/* Check data */
if (mc_data - mc_data_init + 2 * sizeof(u16) > size)
goto eof;
mc_magic_word = u_code_read16(mc_data);
mc_data += sizeof(u16);
mc_nr_of_blks = u_code_read16(mc_data);
mc_data += sizeof(u16);
if ((mc_magic_word != DRX_UCODE_MAGIC_WORD) || (mc_nr_of_blks == 0)) {
rc = -EINVAL; /* wrong endianess or wrong data ? */
pr_err("Firmware magic word doesn't match\n");
goto release; goto release;
} }
/* pr_info("Firmware %s, size %zu\n",
* Scan microcode blocks first for version info mc_file, demod->firmware->size);
* and firmware check }
*/
/* Clear version block */ mc_data_init = demod->firmware->data;
DRX_ATTR_MCRECORD(demod).aux_type = 0; size = demod->firmware->size;
DRX_ATTR_MCRECORD(demod).mc_dev_type = 0;
DRX_ATTR_MCRECORD(demod).mc_version = 0;
DRX_ATTR_MCRECORD(demod).mc_base_version = 0;
for (i = 0; i < mc_nr_of_blks; i++) {
struct drxu_code_block_hdr block_hdr;
if (mc_data - mc_data_init + mc_data = (void *)mc_data_init;
3 * sizeof(u16) + sizeof(u32) > size) /* Check data */
goto eof; mc_magic_word = u_code_read16(mc_data);
/* Process block header */ mc_data += sizeof(u16);
block_hdr.addr = u_code_read32(mc_data); mc_nr_of_blks = u_code_read16(mc_data);
mc_data += sizeof(u32); mc_data += sizeof(u16);
block_hdr.size = u_code_read16(mc_data);
mc_data += sizeof(u16);
block_hdr.flags = u_code_read16(mc_data);
mc_data += sizeof(u16);
block_hdr.CRC = u_code_read16(mc_data);
mc_data += sizeof(u16);
if (block_hdr.flags & 0x8) { if ((mc_magic_word != DRX_UCODE_MAGIC_WORD) || (mc_nr_of_blks == 0)) {
u8 *auxblk = ((void *)mc_data_init) + block_hdr.addr; rc = -EINVAL;
u16 auxtype; pr_err("Firmware magic word doesn't match\n");
goto release;
if (mc_data - mc_data_init + sizeof(u16) +
2 * sizeof(u32) > size)
goto eof;
/* Aux block. Check type */
auxtype = u_code_read16(auxblk);
if (DRX_ISMCVERTYPE(auxtype)) {
DRX_ATTR_MCRECORD(demod).aux_type = u_code_read16(auxblk);
auxblk += sizeof(u16);
DRX_ATTR_MCRECORD(demod).mc_dev_type = u_code_read32(auxblk);
auxblk += sizeof(u32);
DRX_ATTR_MCRECORD(demod).mc_version = u_code_read32(auxblk);
auxblk += sizeof(u32);
DRX_ATTR_MCRECORD(demod).mc_base_version = u_code_read32(auxblk);
}
}
if (mc_data - mc_data_init +
block_hdr.size * sizeof(u16) > size)
goto eof;
/* Next block */
mc_data += block_hdr.size * sizeof(u16);
}
/* Restore data pointer */
mc_data = ((void *)mc_data_init) + 2 * sizeof(u16);
} }
if (action == UCODE_UPLOAD) { if (action == UCODE_UPLOAD) {
rc = check_firmware(demod, (u8 *)mc_data_init, size);
if (rc)
goto release;
/* After scanning, validate the microcode. /* After scanning, validate the microcode.
It is also valid if no validation control exists. It is also valid if no validation control exists.
*/ */
rc = drx_ctrl(demod, DRX_CTRL_VALIDATE_UCODE, NULL); rc = drx_ctrl(demod, DRX_CTRL_VALIDATE_UCODE, NULL);
if (rc != 0 && rc != -ENOTSUPP) { if (rc != 0 && rc != -ENOTSUPP) {
pr_err("Validate ucode not supported\n"); pr_err("Validate ucode not supported\n");
goto release; return rc;
} }
pr_info("Uploading firmware %s\n", mc_file); pr_info("Uploading firmware %s\n", mc_file);
} else if (action == UCODE_VERIFY) {
pr_info("Verifying if firmware upload was ok.\n");
} }
/* Process microcode blocks */ /* Process microcode blocks */
@ -1093,6 +1118,10 @@ ctrl_u_code(struct drx_demod_instance *demod,
block_hdr.CRC = u_code_read16(mc_data); block_hdr.CRC = u_code_read16(mc_data);
mc_data += sizeof(u16); mc_data += sizeof(u16);
pr_debug("%u: addr %u, size %u, flags 0x%04x, CRC 0x%04x\n",
(unsigned)(mc_data - mc_data_init), block_hdr.addr,
block_hdr.size, block_hdr.flags, block_hdr.CRC);
/* Check block header on: /* Check block header on:
- data larger than 64Kb - data larger than 64Kb
- if CRC enabled check CRC - if CRC enabled check CRC
@ -1114,17 +1143,18 @@ ctrl_u_code(struct drx_demod_instance *demod,
/* Perform the desired action */ /* Perform the desired action */
switch (action) { switch (action) {
case UCODE_UPLOAD: case UCODE_UPLOAD: /* Upload microcode */
/* Upload microcode */
if (demod->my_access_funct->write_block_func(dev_addr, if (demod->my_access_funct->write_block_func(dev_addr,
block_hdr.addr, block_hdr.addr,
mc_block_nr_bytes, mc_block_nr_bytes,
mc_data, 0x0000)) { mc_data, 0x0000)) {
pr_err("error writing firmware\n"); rc = -EIO;
pr_err("error writing firmware at pos %u\n",
(unsigned)(mc_data - mc_data_init));
goto release; goto release;
} }
break; break;
case UCODE_VERIFY: { case UCODE_VERIFY: { /* Verify uploaded microcode */
int result = 0; int result = 0;
u8 mc_data_buffer[DRX_UCODE_MAX_BUF_SIZE]; u8 mc_data_buffer[DRX_UCODE_MAX_BUF_SIZE];
u32 bytes_to_comp = 0; u32 bytes_to_comp = 0;
@ -1144,8 +1174,9 @@ ctrl_u_code(struct drx_demod_instance *demod,
(u16)bytes_to_comp, (u16)bytes_to_comp,
(u8 *)mc_data_buffer, (u8 *)mc_data_buffer,
0x0000)) { 0x0000)) {
pr_err("error reading firmware\n"); pr_err("error reading firmware at pos %u\n",
goto release; (unsigned)(mc_data - mc_data_init));
return -EIO;
} }
result =drxbsp_hst_memcmp(curr_ptr, result =drxbsp_hst_memcmp(curr_ptr,
@ -1153,7 +1184,8 @@ ctrl_u_code(struct drx_demod_instance *demod,
bytes_to_comp); bytes_to_comp);
if (result) { if (result) {
pr_err("error verifying firmware\n"); pr_err("error verifying firmware at pos %u\n",
(unsigned)(mc_data - mc_data_init));
return -EIO; return -EIO;
} }
@ -1172,10 +1204,7 @@ ctrl_u_code(struct drx_demod_instance *demod,
} }
return 0; return 0;
eof:
rc = -ENOENT;
pr_err("Firmware file %s is truncated at pos %lu\n",
mc_file, (unsigned long)(mc_data - mc_data_init));
release: release:
release_firmware(demod->firmware); release_firmware(demod->firmware);
demod->firmware = NULL; demod->firmware = NULL;