cregit-Linux how code gets into the kernel

Release 4.11 drivers/nfc/s3fwrn5/firmware.c

/*
 * NCI based driver for Samsung S3FWRN5 NFC chip
 *
 * Copyright (C) 2015 Samsung Electrnoics
 * Robert Baldyga <r.baldyga@samsung.com>
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2 or later, as published by the Free Software Foundation.
 *
 * 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.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, see <http://www.gnu.org/licenses/>.
 */

#include <linux/completion.h>
#include <linux/firmware.h>
#include <crypto/hash.h>
#include <crypto/sha.h>

#include "s3fwrn5.h"
#include "firmware.h"


struct s3fwrn5_fw_version {
	
__u8 major;
	
__u8 build1;
	
__u8 build2;
	
__u8 target;
};


static int s3fwrn5_fw_send_msg(struct s3fwrn5_fw_info *fw_info, struct sk_buff *msg, struct sk_buff **rsp) { struct s3fwrn5_info *info = container_of(fw_info, struct s3fwrn5_info, fw_info); long ret; reinit_completion(&fw_info->completion); ret = s3fwrn5_write(info, msg); if (ret < 0) return ret; ret = wait_for_completion_interruptible_timeout( &fw_info->completion, msecs_to_jiffies(1000)); if (ret < 0) return ret; else if (ret == 0) return -ENXIO; if (!fw_info->rsp) return -EINVAL; *rsp = fw_info->rsp; fw_info->rsp = NULL; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga128100.00%1100.00%
Total128100.00%1100.00%


static int s3fwrn5_fw_prep_msg(struct s3fwrn5_fw_info *fw_info, struct sk_buff **msg, u8 type, u8 code, const void *data, u16 len) { struct s3fwrn5_fw_header hdr; struct sk_buff *skb; hdr.type = type | fw_info->parity; fw_info->parity ^= 0x80; hdr.code = code; hdr.len = len; skb = alloc_skb(S3FWRN5_FW_HDR_SIZE + len, GFP_KERNEL); if (!skb) return -ENOMEM; memcpy(skb_put(skb, S3FWRN5_FW_HDR_SIZE), &hdr, S3FWRN5_FW_HDR_SIZE); if (len) memcpy(skb_put(skb, len), data, len); *msg = skb; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga129100.00%1100.00%
Total129100.00%1100.00%


static int s3fwrn5_fw_get_bootinfo(struct s3fwrn5_fw_info *fw_info, struct s3fwrn5_fw_cmd_get_bootinfo_rsp *bootinfo) { struct sk_buff *msg, *rsp = NULL; struct s3fwrn5_fw_header *hdr; int ret; /* Send GET_BOOTINFO command */ ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_CMD, S3FWRN5_FW_CMD_GET_BOOTINFO, NULL, 0); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) { ret = -EINVAL; goto out; } memcpy(bootinfo, rsp->data + S3FWRN5_FW_HDR_SIZE, 10); out: kfree_skb(rsp); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga140100.00%1100.00%
Total140100.00%1100.00%


static int s3fwrn5_fw_enter_update_mode(struct s3fwrn5_fw_info *fw_info, const void *hash_data, u16 hash_size, const void *sig_data, u16 sig_size) { struct s3fwrn5_fw_cmd_enter_updatemode args; struct sk_buff *msg, *rsp = NULL; struct s3fwrn5_fw_header *hdr; int ret; /* Send ENTER_UPDATE_MODE command */ args.hashcode_size = hash_size; args.signature_size = sig_size; ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_CMD, S3FWRN5_FW_CMD_ENTER_UPDATE_MODE, &args, sizeof(args)); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) { ret = -EPROTO; goto out; } kfree_skb(rsp); /* Send hashcode data */ ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_DATA, 0, hash_data, hash_size); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) { ret = -EPROTO; goto out; } kfree_skb(rsp); /* Send signature data */ ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_DATA, 0, sig_data, sig_size); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) ret = -EPROTO; out: kfree_skb(rsp); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga329100.00%1100.00%
Total329100.00%1100.00%


static int s3fwrn5_fw_update_sector(struct s3fwrn5_fw_info *fw_info, u32 base_addr, const void *data) { struct s3fwrn5_fw_cmd_update_sector args; struct sk_buff *msg, *rsp = NULL; struct s3fwrn5_fw_header *hdr; int ret, i; /* Send UPDATE_SECTOR command */ args.base_address = base_addr; ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_CMD, S3FWRN5_FW_CMD_UPDATE_SECTOR, &args, sizeof(args)); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) { ret = -EPROTO; goto err; } kfree_skb(rsp); /* Send data split into 256-byte packets */ for (i = 0; i < 16; ++i) { ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_DATA, 0, data+256*i, 256); if (ret < 0) break; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) break; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) { ret = -EPROTO; goto err; } kfree_skb(rsp); } return ret; err: kfree_skb(rsp); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga257100.00%1100.00%
Total257100.00%1100.00%


static int s3fwrn5_fw_complete_update_mode(struct s3fwrn5_fw_info *fw_info) { struct sk_buff *msg, *rsp = NULL; struct s3fwrn5_fw_header *hdr; int ret; /* Send COMPLETE_UPDATE_MODE command */ ret = s3fwrn5_fw_prep_msg(fw_info, &msg, S3FWRN5_FW_MSG_CMD, S3FWRN5_FW_CMD_COMPLETE_UPDATE_MODE, NULL, 0); if (ret < 0) return ret; ret = s3fwrn5_fw_send_msg(fw_info, msg, &rsp); kfree_skb(msg); if (ret < 0) return ret; hdr = (struct s3fwrn5_fw_header *) rsp->data; if (hdr->code != S3FWRN5_FW_RET_SUCCESS) ret = -EPROTO; kfree_skb(rsp); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga115100.00%1100.00%
Total115100.00%1100.00%

/* * Firmware header stucture: * * 0x00 - 0x0B : Date and time string (w/o NUL termination) * 0x10 - 0x13 : Firmware version * 0x14 - 0x17 : Signature address * 0x18 - 0x1B : Signature size * 0x1C - 0x1F : Firmware image address * 0x20 - 0x23 : Firmware sectors count * 0x24 - 0x27 : Custom signature address * 0x28 - 0x2B : Custom signature size */ #define S3FWRN5_FW_IMAGE_HEADER_SIZE 44
static int s3fwrn5_fw_request_firmware(struct s3fwrn5_fw_info *fw_info) { struct s3fwrn5_fw_image *fw = &fw_info->fw; u32 sig_off; u32 image_off; u32 custom_sig_off; int ret; ret = request_firmware(&fw->fw, fw_info->fw_name, &fw_info->ndev->nfc_dev->dev); if (ret < 0) return ret; if (fw->fw->size < S3FWRN5_FW_IMAGE_HEADER_SIZE) return -EINVAL; memcpy(fw->date, fw->fw->data + 0x00, 12); fw->date[12] = '\0'; memcpy(&fw->version, fw->fw->data + 0x10, 4); memcpy(&sig_off, fw->fw->data + 0x14, 4); fw->sig = fw->fw->data + sig_off; memcpy(&fw->sig_size, fw->fw->data + 0x18, 4); memcpy(&image_off, fw->fw->data + 0x1C, 4); fw->image = fw->fw->data + image_off; memcpy(&fw->image_sectors, fw->fw->data + 0x20, 4); memcpy(&custom_sig_off, fw->fw->data + 0x24, 4); fw->custom_sig = fw->fw->data + custom_sig_off; memcpy(&fw->custom_sig_size, fw->fw->data + 0x28, 4); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga264100.00%1100.00%
Total264100.00%1100.00%


static void s3fwrn5_fw_release_firmware(struct s3fwrn5_fw_info *fw_info) { release_firmware(fw_info->fw.fw); }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga20100.00%1100.00%
Total20100.00%1100.00%


static int s3fwrn5_fw_get_base_addr( struct s3fwrn5_fw_cmd_get_bootinfo_rsp *bootinfo, u32 *base_addr) { int i; struct { u8 version[4]; u32 base_addr; } match[] = { {{0x05, 0x00, 0x00, 0x00}, 0x00005000}, {{0x05, 0x00, 0x00, 0x01}, 0x00003000}, {{0x05, 0x00, 0x00, 0x02}, 0x00003000}, {{0x05, 0x00, 0x00, 0x03}, 0x00003000}, {{0x05, 0x00, 0x00, 0x05}, 0x00003000} }; for (i = 0; i < ARRAY_SIZE(match); ++i) if (bootinfo->hw_version[0] == match[i].version[0] && bootinfo->hw_version[1] == match[i].version[1] && bootinfo->hw_version[3] == match[i].version[3]) { *base_addr = match[i].base_addr; return 0; } return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga193100.00%1100.00%
Total193100.00%1100.00%


static inline bool s3fwrn5_fw_is_custom(struct s3fwrn5_fw_cmd_get_bootinfo_rsp *bootinfo) { return !!bootinfo->hw_version[2]; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga22100.00%1100.00%
Total22100.00%1100.00%


int s3fwrn5_fw_setup(struct s3fwrn5_fw_info *fw_info) { struct s3fwrn5_fw_cmd_get_bootinfo_rsp bootinfo; int ret; /* Get firmware data */ ret = s3fwrn5_fw_request_firmware(fw_info); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Failed to get fw file, ret=%02x\n", ret); return ret; } /* Get bootloader info */ ret = s3fwrn5_fw_get_bootinfo(fw_info, &bootinfo); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Failed to get bootinfo, ret=%02x\n", ret); goto err; } /* Match hardware version to obtain firmware base address */ ret = s3fwrn5_fw_get_base_addr(&bootinfo, &fw_info->base_addr); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Unknown hardware version\n"); goto err; } fw_info->sector_size = bootinfo.sector_size; fw_info->sig_size = s3fwrn5_fw_is_custom(&bootinfo) ? fw_info->fw.custom_sig_size : fw_info->fw.sig_size; fw_info->sig = s3fwrn5_fw_is_custom(&bootinfo) ? fw_info->fw.custom_sig : fw_info->fw.sig; return 0; err: s3fwrn5_fw_release_firmware(fw_info); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga194100.00%1100.00%
Total194100.00%1100.00%


bool s3fwrn5_fw_check_version(struct s3fwrn5_fw_info *fw_info, u32 version) { struct s3fwrn5_fw_version *new = (void *) &fw_info->fw.version; struct s3fwrn5_fw_version *old = (void *) &version; if (new->major > old->major) return true; if (new->build1 > old->build1) return true; if (new->build2 > old->build2) return true; return false; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga83100.00%1100.00%
Total83100.00%1100.00%


int s3fwrn5_fw_download(struct s3fwrn5_fw_info *fw_info) { struct s3fwrn5_fw_image *fw = &fw_info->fw; u8 hash_data[SHA1_DIGEST_SIZE]; struct crypto_shash *tfm; u32 image_size, off; int ret; image_size = fw_info->sector_size * fw->image_sectors; /* Compute SHA of firmware data */ tfm = crypto_alloc_shash("sha1", 0, 0); if (IS_ERR(tfm)) { ret = PTR_ERR(tfm); dev_err(&fw_info->ndev->nfc_dev->dev, "Cannot allocate shash (code=%d)\n", ret); goto out; } { SHASH_DESC_ON_STACK(desc, tfm); desc->tfm = tfm; desc->flags = CRYPTO_TFM_REQ_MAY_SLEEP; ret = crypto_shash_digest(desc, fw->image, image_size, hash_data); shash_desc_zero(desc); } crypto_free_shash(tfm); if (ret) { dev_err(&fw_info->ndev->nfc_dev->dev, "Cannot compute hash (code=%d)\n", ret); goto out; } /* Firmware update process */ dev_info(&fw_info->ndev->nfc_dev->dev, "Firmware update: %s\n", fw_info->fw_name); ret = s3fwrn5_fw_enter_update_mode(fw_info, hash_data, SHA1_DIGEST_SIZE, fw_info->sig, fw_info->sig_size); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Unable to enter update mode\n"); goto out; } for (off = 0; off < image_size; off += fw_info->sector_size) { ret = s3fwrn5_fw_update_sector(fw_info, fw_info->base_addr + off, fw->image + off); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Firmware update error (code=%d)\n", ret); goto out; } } ret = s3fwrn5_fw_complete_update_mode(fw_info); if (ret < 0) { dev_err(&fw_info->ndev->nfc_dev->dev, "Unable to complete update mode\n"); goto out; } dev_info(&fw_info->ndev->nfc_dev->dev, "Firmware update: success\n"); out: return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga25674.20%150.00%
Herbert Xu8925.80%150.00%
Total345100.00%2100.00%


void s3fwrn5_fw_init(struct s3fwrn5_fw_info *fw_info, const char *fw_name) { fw_info->parity = 0x00; fw_info->rsp = NULL; fw_info->fw.fw = NULL; strcpy(fw_info->fw_name, fw_name); init_completion(&fw_info->completion); }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga52100.00%1100.00%
Total52100.00%1100.00%


void s3fwrn5_fw_cleanup(struct s3fwrn5_fw_info *fw_info) { s3fwrn5_fw_release_firmware(fw_info); }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga15100.00%1100.00%
Total15100.00%1100.00%


int s3fwrn5_fw_recv_frame(struct nci_dev *ndev, struct sk_buff *skb) { struct s3fwrn5_info *info = nci_get_drvdata(ndev); struct s3fwrn5_fw_info *fw_info = &info->fw_info; BUG_ON(fw_info->rsp); fw_info->rsp = skb; complete(&fw_info->completion); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga59100.00%1100.00%
Total59100.00%1100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Robert Baldyga229696.23%150.00%
Herbert Xu903.77%150.00%
Total2386100.00%2100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.