blob: bd00d03f29f73ea88b24430ef5b7a5f7468674ab [file] [log] [blame]
/*
* Copyright (C) 2019, Samsung Electronics Co. Ltd. All Rights Reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/firmware.h>
#include <linux/io.h>
#include <linux/slab.h>
#include <linux/mm.h>
#include "ssp.h"
#include "ssp_firmware.h"
#include "factory/ssp_sensor.h"
enum fw_type
{
FW_TYPE_NONE,
FW_TYPE_BIN,
FW_TYPE_PUSH,
FW_TYPE_SPU,
};
#define SUPPORT_SPU_FW
#define UPDATE_BIN_FILE "shub.bin"
#ifdef SUPPORT_SPU_FW
#define SPU_FW_FILE "/spu/sensorhub/shub_spu.bin"
#define FW_VER_LEN 8
extern long spu_firmware_signature_verify(const char* fw_name, const u8* fw_data, const long fw_size);
static int request_spu_firmware(struct ssp_data *data, u8 **fw_buf)
{
int ret = 0;
size_t file_size = 0, remaining;
int offset = 0;
unsigned int read_size = PAGE_SIZE*10;
long fw_size = 0;
struct file *filp = NULL;
u8 *file_buf;
mm_segment_t old_fs = get_fs();
set_fs(KERNEL_DS);
ssp_infof("");
filp = filp_open(SPU_FW_FILE, O_RDONLY, 0);
if(IS_ERR(filp)){
ssp_infof("filp_open failed %d", PTR_ERR(filp));
set_fs(old_fs);
return 0;
}
file_size = filp->f_path.dentry->d_inode->i_size;
if(file_size <= 0)
{
filp_close(filp, NULL);
set_fs(old_fs);
return 0;
}
file_buf = kvzalloc(file_size, GFP_KERNEL);
if(file_buf == NULL)
{
ssp_errf("file buf kvzalloc error");
return 0;
}
remaining = file_size;
while (remaining > 0) {
int ret = 0;
if (read_size > remaining) {
read_size = remaining;
}
ret = (unsigned int)vfs_read(filp, (char __user *)(file_buf+offset),
read_size, &filp->f_pos);
if(ret != read_size)
{
ssp_errf("file read fail %d", ret);
break;
}
offset += read_size;
remaining -= read_size;
filp->f_pos = offset;
}
filp_close(filp, NULL);
set_fs(old_fs);
if (ret < 0) {
ssp_errf("file read fail %d", ret);
kfree(file_buf);
return 0;
}
//check signing
fw_size = spu_firmware_signature_verify("SENSORHUB", file_buf, file_size);
if(fw_size < 0) {
ssp_errf("signature verification failed %d", fw_size);
fw_size = 0;
} else {
u32 fw_version = 0;
char str_ver[9] = "";
ssp_infof("signature verification success %d", fw_size);
if(fw_size < FW_VER_LEN)
{
ssp_errf("fw size is wrong %d", fw_size);
kfree(file_buf);
return 0;
}
memcpy(str_ver, file_buf + fw_size - FW_VER_LEN, 8);
ret = kstrtou32(str_ver, 10, &fw_version);
ssp_infof("urgent fw_version %d kernel ver %u", fw_version, get_module_rev(data));
if(fw_version > get_module_rev(data))
{
fw_size -= FW_VER_LEN;
ssp_infof("use spu fw size %d", fw_size);
*fw_buf = kvzalloc(fw_size, GFP_KERNEL);
if(*fw_buf == NULL)
{
ssp_errf("fw buf kvzalloc error");
kfree(file_buf);
return 0;
}
memcpy(*fw_buf, file_buf, fw_size);
}
else
fw_size = 0;
}
kfree(file_buf);
return (int)fw_size;
}
static void release_spu_firmware(struct ssp_data *data, u8 *fw_buf)
{
kfree(fw_buf);
}
#endif
int download_sensorhub_firmware(struct ssp_data *data, struct device* dev, void * addr)
{
int ret = 0;
int fw_size;
char* fw_buf = NULL;
const struct firmware *entry = NULL;
if(addr == NULL)
return -EINVAL;
#ifdef CONFIG_SSP_ENG_DEBUG
ssp_infof("check push bin file");
ret = request_firmware(&entry, UPDATE_BIN_FILE, dev);
if(!ret)
{
data->fw_type = FW_TYPE_PUSH;
fw_size = (int)entry->size;
fw_buf = (char*)entry->data;
} else
#endif
{
#ifdef SUPPORT_SPU_FW
fw_size = request_spu_firmware(data, (u8**)&fw_buf);
if(fw_size > 0)
{
ssp_infof("dowload spu firmware");
data->fw_type = FW_TYPE_SPU;
}
else
#endif
{
ssp_infof("download %s", data->fw_name);
ret = request_firmware(&entry, data->fw_name, dev);
if(ret)
{
ssp_errf("request_firmware failed %d", ret);
data->fw_type = FW_TYPE_NONE;
release_firmware(entry);
return -EINVAL;
}
data->fw_type = FW_TYPE_BIN;
fw_size = (int)entry->size;
fw_buf = (char*)entry->data;
}
}
ssp_infof("fw type %d bin(size:%d) on %lx",data->fw_type, (int)fw_size, (unsigned long)addr);
memcpy(addr, fw_buf, fw_size);
if(entry)
release_firmware(entry);
#ifdef SUPPORT_SPU_FW
if(data->fw_type == FW_TYPE_SPU)
release_spu_firmware(data, fw_buf);
#endif
return 0;
}