cregit-Linux how code gets into the kernel

Release 4.16 drivers/bluetooth/btqcomsmd.c

/*
 * Copyright (c) 2016, Linaro Ltd.
 * Copyright (c) 2015, Sony Mobile Communications Inc.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 and
 * only version 2 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.
 */

#include <linux/module.h>
#include <linux/slab.h>
#include <linux/rpmsg.h>
#include <linux/of.h>

#include <linux/soc/qcom/wcnss_ctrl.h>
#include <linux/platform_device.h>

#include <net/bluetooth/bluetooth.h>
#include <net/bluetooth/hci_core.h>

#include "btqca.h"


struct btqcomsmd {
	
struct hci_dev *hdev;

	
bdaddr_t bdaddr;
	
struct rpmsg_endpoint *acl_channel;
	
struct rpmsg_endpoint *cmd_channel;
};


static int btqcomsmd_recv(struct hci_dev *hdev, unsigned int type, const void *data, size_t count) { struct sk_buff *skb; /* Use GFP_ATOMIC as we're in IRQ context */ skb = bt_skb_alloc(count, GFP_ATOMIC); if (!skb) { hdev->stat.err_rx++; return -ENOMEM; } hci_skb_pkt_type(skb) = type; skb_put_data(skb, data, count); return hci_recv_frame(hdev, skb); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson7998.75%150.00%
Johannes Berg11.25%150.00%
Total80100.00%2100.00%


static int btqcomsmd_acl_callback(struct rpmsg_device *rpdev, void *data, int count, void *priv, u32 addr) { struct btqcomsmd *btq = priv; btq->hdev->stat.byte_rx += count; return btqcomsmd_recv(btq->hdev, HCI_ACLDATA_PKT, data, count); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson56100.00%2100.00%
Total56100.00%2100.00%


static int btqcomsmd_cmd_callback(struct rpmsg_device *rpdev, void *data, int count, void *priv, u32 addr) { struct btqcomsmd *btq = priv; return btqcomsmd_recv(btq->hdev, HCI_EVENT_PKT, data, count); }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson46100.00%2100.00%
Total46100.00%2100.00%


static int btqcomsmd_send(struct hci_dev *hdev, struct sk_buff *skb) { struct btqcomsmd *btq = hci_get_drvdata(hdev); int ret; switch (hci_skb_pkt_type(skb)) { case HCI_ACLDATA_PKT: ret = rpmsg_send(btq->acl_channel, skb->data, skb->len); hdev->stat.acl_tx++; hdev->stat.byte_tx += skb->len; break; case HCI_COMMAND_PKT: ret = rpmsg_send(btq->cmd_channel, skb->data, skb->len); hdev->stat.cmd_tx++; break; default: ret = -EILSEQ; break; } if (!ret) kfree_skb(skb); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson11995.97%266.67%
Loic Poulain54.03%133.33%
Total124100.00%3100.00%


static int btqcomsmd_open(struct hci_dev *hdev) { return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson14100.00%1100.00%
Total14100.00%1100.00%


static int btqcomsmd_close(struct hci_dev *hdev) { return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson14100.00%1100.00%
Total14100.00%1100.00%


static int btqcomsmd_setup(struct hci_dev *hdev) { struct btqcomsmd *btq = hci_get_drvdata(hdev); struct sk_buff *skb; int err; skb = __hci_cmd_sync(hdev, HCI_OP_RESET, 0, NULL, HCI_INIT_TIMEOUT); if (IS_ERR(skb)) return PTR_ERR(skb); kfree_skb(skb); /* Devices do not have persistent storage for BD address. If no * BD address has been retrieved during probe, mark the device * as having an invalid BD address. */ if (!bacmp(&btq->bdaddr, BDADDR_ANY)) { set_bit(HCI_QUIRK_INVALID_BDADDR, &hdev->quirks); return 0; } /* When setting a configured BD address fails, mark the device * as having an invalid BD address. */ err = qca_set_bdaddr_rome(hdev, &btq->bdaddr); if (err) { set_bit(HCI_QUIRK_INVALID_BDADDR, &hdev->quirks); return 0; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Loic Poulain126100.00%1100.00%
Total126100.00%1100.00%


static int btqcomsmd_probe(struct platform_device *pdev) { struct btqcomsmd *btq; struct hci_dev *hdev; void *wcnss; int ret; btq = devm_kzalloc(&pdev->dev, sizeof(*btq), GFP_KERNEL); if (!btq) return -ENOMEM; wcnss = dev_get_drvdata(pdev->dev.parent); btq->acl_channel = qcom_wcnss_open_channel(wcnss, "APPS_RIVA_BT_ACL", btqcomsmd_acl_callback, btq); if (IS_ERR(btq->acl_channel)) return PTR_ERR(btq->acl_channel); btq->cmd_channel = qcom_wcnss_open_channel(wcnss, "APPS_RIVA_BT_CMD", btqcomsmd_cmd_callback, btq); if (IS_ERR(btq->cmd_channel)) return PTR_ERR(btq->cmd_channel); /* The local-bd-address property is usually injected by the * bootloader which has access to the allocated BD address. */ if (!of_property_read_u8_array(pdev->dev.of_node, "local-bd-address", (u8 *)&btq->bdaddr, sizeof(bdaddr_t))) { dev_info(&pdev->dev, "BD address %pMR retrieved from device-tree", &btq->bdaddr); } hdev = hci_alloc_dev(); if (!hdev) return -ENOMEM; hci_set_drvdata(hdev, btq); btq->hdev = hdev; SET_HCIDEV_DEV(hdev, &pdev->dev); hdev->bus = HCI_SMD; hdev->open = btqcomsmd_open; hdev->close = btqcomsmd_close; hdev->send = btqcomsmd_send; hdev->setup = btqcomsmd_setup; hdev->set_bdaddr = qca_set_bdaddr_rome; ret = hci_register_dev(hdev); if (ret < 0) { hci_free_dev(hdev); return ret; } platform_set_drvdata(pdev, btq); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson23081.56%250.00%
Loic Poulain5218.44%250.00%
Total282100.00%4100.00%


static int btqcomsmd_remove(struct platform_device *pdev) { struct btqcomsmd *btq = platform_get_drvdata(pdev); hci_unregister_dev(btq->hdev); hci_free_dev(btq->hdev); rpmsg_destroy_ept(btq->cmd_channel); rpmsg_destroy_ept(btq->acl_channel); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson52100.00%2100.00%
Total52100.00%2100.00%

static const struct of_device_id btqcomsmd_of_match[] = { { .compatible = "qcom,wcnss-bt", }, { }, }; MODULE_DEVICE_TABLE(of, btqcomsmd_of_match); static struct platform_driver btqcomsmd_driver = { .probe = btqcomsmd_probe, .remove = btqcomsmd_remove, .driver = { .name = "btqcomsmd", .of_match_table = btqcomsmd_of_match, }, }; module_platform_driver(btqcomsmd_driver); MODULE_AUTHOR("Bjorn Andersson <bjorn.andersson@sonymobile.com>"); MODULE_DESCRIPTION("Qualcomm SMD HCI driver"); MODULE_LICENSE("GPL v2");

Overall Contributors

PersonTokensPropCommitsCommitProp
Björn Andersson72578.63%228.57%
Loic Poulain18920.50%342.86%
Javier Martinez Canillas70.76%114.29%
Johannes Berg10.11%114.29%
Total922100.00%7100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.