blob: 76a589667104c5e78c0faa4ccb809220e0cfdbf3 [file] [log] [blame]
/*
* f_qdss.c -- QDSS function Driver
*
* Copyright (c) 2012-2018, 2020, The Linux Foundation. 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 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/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/device.h>
#include <linux/usb/usb_qdss.h>
#include <linux/usb/cdc.h>
#include "f_qdss.h"
static DEFINE_SPINLOCK(qdss_lock);
static LIST_HEAD(usb_qdss_ch_list);
static struct usb_interface_descriptor qdss_data_intf_desc = {
.bLength = sizeof(qdss_data_intf_desc),
.bDescriptorType = USB_DT_INTERFACE,
.bAlternateSetting = 0,
.bNumEndpoints = 1,
.bInterfaceClass = 0xff,
.bInterfaceSubClass = 0xff,
.bInterfaceProtocol = 0xff,
};
static struct usb_endpoint_descriptor qdss_hs_data_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(512),
};
static struct usb_endpoint_descriptor qdss_ss_data_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(1024),
};
static struct usb_ss_ep_comp_descriptor qdss_data_ep_comp_desc = {
.bLength = sizeof(qdss_data_ep_comp_desc),
.bDescriptorType = USB_DT_SS_ENDPOINT_COMP,
.bMaxBurst = 1,
.bmAttributes = 0,
.wBytesPerInterval = 0,
};
static struct usb_interface_descriptor qdss_ctrl_intf_desc = {
.bLength = sizeof(qdss_ctrl_intf_desc),
.bDescriptorType = USB_DT_INTERFACE,
.bAlternateSetting = 0,
.bNumEndpoints = 2,
.bInterfaceClass = 0xff,
.bInterfaceSubClass = 0xff,
.bInterfaceProtocol = 0xff,
};
static struct usb_endpoint_descriptor qdss_hs_ctrl_in_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(512),
};
static struct usb_endpoint_descriptor qdss_ss_ctrl_in_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(1024),
};
static struct usb_endpoint_descriptor qdss_hs_ctrl_out_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_OUT,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(512),
};
static struct usb_endpoint_descriptor qdss_ss_ctrl_out_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_OUT,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(0x400),
};
static struct usb_ss_ep_comp_descriptor qdss_ctrl_in_ep_comp_desc = {
.bLength = sizeof(qdss_ctrl_in_ep_comp_desc),
.bDescriptorType = USB_DT_SS_ENDPOINT_COMP,
.bMaxBurst = 0,
.bmAttributes = 0,
.wBytesPerInterval = 0,
};
static struct usb_ss_ep_comp_descriptor qdss_ctrl_out_ep_comp_desc = {
.bLength = sizeof(qdss_ctrl_out_ep_comp_desc),
.bDescriptorType = USB_DT_SS_ENDPOINT_COMP,
.bMaxBurst = 0,
.bmAttributes = 0,
.wBytesPerInterval = 0,
};
/* Full speed support */
static struct usb_endpoint_descriptor qdss_fs_data_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(64),
};
static struct usb_endpoint_descriptor qdss_fs_ctrl_in_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_IN,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(64),
};
static struct usb_endpoint_descriptor qdss_fs_ctrl_out_desc = {
.bLength = USB_DT_ENDPOINT_SIZE,
.bDescriptorType = USB_DT_ENDPOINT,
.bEndpointAddress = USB_DIR_OUT,
.bmAttributes = USB_ENDPOINT_XFER_BULK,
.wMaxPacketSize = cpu_to_le16(64),
};
static struct usb_descriptor_header *qdss_fs_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_fs_data_desc,
(struct usb_descriptor_header *) &qdss_ctrl_intf_desc,
(struct usb_descriptor_header *) &qdss_fs_ctrl_in_desc,
(struct usb_descriptor_header *) &qdss_fs_ctrl_out_desc,
NULL,
};
static struct usb_descriptor_header *qdss_hs_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_hs_data_desc,
(struct usb_descriptor_header *) &qdss_ctrl_intf_desc,
(struct usb_descriptor_header *) &qdss_hs_ctrl_in_desc,
(struct usb_descriptor_header *) &qdss_hs_ctrl_out_desc,
NULL,
};
static struct usb_descriptor_header *qdss_ss_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_ss_data_desc,
(struct usb_descriptor_header *) &qdss_data_ep_comp_desc,
(struct usb_descriptor_header *) &qdss_ctrl_intf_desc,
(struct usb_descriptor_header *) &qdss_ss_ctrl_in_desc,
(struct usb_descriptor_header *) &qdss_ctrl_in_ep_comp_desc,
(struct usb_descriptor_header *) &qdss_ss_ctrl_out_desc,
(struct usb_descriptor_header *) &qdss_ctrl_out_ep_comp_desc,
NULL,
};
static struct usb_descriptor_header *qdss_fs_data_only_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_fs_data_desc,
NULL,
};
static struct usb_descriptor_header *qdss_hs_data_only_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_hs_data_desc,
NULL,
};
static struct usb_descriptor_header *qdss_ss_data_only_desc[] = {
(struct usb_descriptor_header *) &qdss_data_intf_desc,
(struct usb_descriptor_header *) &qdss_ss_data_desc,
(struct usb_descriptor_header *) &qdss_data_ep_comp_desc,
NULL,
};
/* string descriptors: */
#define QDSS_DATA_IDX 0
#define QDSS_CTRL_IDX 1
static struct usb_string qdss_string_defs[] = {
[QDSS_DATA_IDX].s = "QDSS DATA",
[QDSS_CTRL_IDX].s = "QDSS CTRL",
{}, /* end of list */
};
static struct usb_gadget_strings qdss_string_table = {
.language = 0x0409,
.strings = qdss_string_defs,
};
static struct usb_gadget_strings *qdss_strings[] = {
&qdss_string_table,
NULL,
};
static inline struct f_qdss *func_to_qdss(struct usb_function *f)
{
return container_of(f, struct f_qdss, port.function);
}
static
struct usb_qdss_opts *to_fi_usb_qdss_opts(struct usb_function_instance *fi)
{
return container_of(fi, struct usb_qdss_opts, func_inst);
}
/*----------------------------------------------------------------------*/
static void qdss_write_complete(struct usb_ep *ep,
struct usb_request *req)
{
struct f_qdss *qdss = ep->driver_data;
struct qdss_request *d_req = req->context;
struct usb_ep *in;
struct list_head *list_pool;
enum qdss_state state;
unsigned long flags;
pr_debug("qdss_ctrl_write_complete\n");
if (qdss->debug_inface_enabled) {
in = qdss->port.ctrl_in;
list_pool = &qdss->ctrl_write_pool;
state = USB_QDSS_CTRL_WRITE_DONE;
} else {
in = qdss->port.data;
list_pool = &qdss->data_write_pool;
state = USB_QDSS_DATA_WRITE_DONE;
}
if (!req->status) {
/* send zlp */
if ((req->length >= ep->maxpacket) &&
((req->length % ep->maxpacket) == 0)) {
req->length = 0;
d_req->actual = req->actual;
d_req->status = req->status;
if (!usb_ep_queue(in, req, GFP_ATOMIC))
return;
}
}
spin_lock_irqsave(&qdss->lock, flags);
list_add_tail(&req->list, list_pool);
if (req->length != 0) {
d_req->actual = req->actual;
d_req->status = req->status;
}
spin_unlock_irqrestore(&qdss->lock, flags);
if (qdss->ch.notify)
qdss->ch.notify(qdss->ch.priv, state, d_req, NULL);
}
static void qdss_ctrl_read_complete(struct usb_ep *ep,
struct usb_request *req)
{
struct f_qdss *qdss = ep->driver_data;
struct qdss_request *d_req = req->context;
unsigned long flags;
pr_debug("qdss_ctrl_read_complete\n");
d_req->actual = req->actual;
d_req->status = req->status;
spin_lock_irqsave(&qdss->lock, flags);
list_add_tail(&req->list, &qdss->ctrl_read_pool);
spin_unlock_irqrestore(&qdss->lock, flags);
if (qdss->ch.notify)
qdss->ch.notify(qdss->ch.priv, USB_QDSS_CTRL_READ_DONE, d_req,
NULL);
}
void usb_qdss_free_req(struct usb_qdss_ch *ch)
{
struct f_qdss *qdss;
struct usb_request *req;
struct list_head *act, *tmp;
pr_debug("usb_qdss_free_req\n");
qdss = ch->priv_usb;
if (!qdss) {
pr_err("usb_qdss_free_req: qdss ctx is NULL\n");
return;
}
list_for_each_safe(act, tmp, &qdss->data_write_pool) {
req = list_entry(act, struct usb_request, list);
list_del(&req->list);
usb_ep_free_request(qdss->port.data, req);
}
list_for_each_safe(act, tmp, &qdss->ctrl_write_pool) {
req = list_entry(act, struct usb_request, list);
list_del(&req->list);
usb_ep_free_request(qdss->port.ctrl_in, req);
}
list_for_each_safe(act, tmp, &qdss->ctrl_read_pool) {
req = list_entry(act, struct usb_request, list);
list_del(&req->list);
usb_ep_free_request(qdss->port.ctrl_out, req);
}
}
EXPORT_SYMBOL(usb_qdss_free_req);
int usb_qdss_alloc_req(struct usb_qdss_ch *ch, int no_write_buf,
int no_read_buf)
{
struct f_qdss *qdss = ch->priv_usb;
struct usb_request *req;
struct usb_ep *in;
struct list_head *list_pool;
int i;
pr_debug("usb_qdss_alloc_req\n");
if (!qdss) {
pr_err("usb_qdss_alloc_req: channel %s closed\n", ch->name);
return -ENODEV;
}
if ((qdss->debug_inface_enabled &&
(no_write_buf <= 0 || no_read_buf <= 0)) ||
(!qdss->debug_inface_enabled &&
(no_write_buf <= 0 || no_read_buf))) {
pr_err("usb_qdss_alloc_req: missing params\n");
return -ENODEV;
}
if (qdss->debug_inface_enabled) {
in = qdss->port.ctrl_in;
list_pool = &qdss->ctrl_write_pool;
} else {
in = qdss->port.data;
list_pool = &qdss->data_write_pool;
}
for (i = 0; i < no_write_buf; i++) {
req = usb_ep_alloc_request(in, GFP_ATOMIC);
if (!req) {
pr_err("usb_qdss_alloc_req: ctrl_in allocation err\n");
goto fail;
}
req->complete = qdss_write_complete;
list_add_tail(&req->list, list_pool);
}
for (i = 0; i < no_read_buf; i++) {
req = usb_ep_alloc_request(qdss->port.ctrl_out, GFP_ATOMIC);
if (!req) {
pr_err("usb_qdss_alloc_req:ctrl_out allocation err\n");
goto fail;
}
req->complete = qdss_ctrl_read_complete;
list_add_tail(&req->list, &qdss->ctrl_read_pool);
}
return 0;
fail:
usb_qdss_free_req(ch);
return -ENOMEM;
}
EXPORT_SYMBOL(usb_qdss_alloc_req);
static void clear_eps(struct usb_function *f)
{
struct f_qdss *qdss = func_to_qdss(f);
pr_debug("clear_eps\n");
if (qdss->port.ctrl_in)
qdss->port.ctrl_in->driver_data = NULL;
if (qdss->port.ctrl_out)
qdss->port.ctrl_out->driver_data = NULL;
if (qdss->port.data)
qdss->port.data->driver_data = NULL;
}
static void clear_desc(struct usb_gadget *gadget, struct usb_function *f)
{
pr_debug("clear_desc\n");
if (gadget_is_superspeed(gadget) && f->ss_descriptors)
usb_free_descriptors(f->ss_descriptors);
if (gadget_is_dualspeed(gadget) && f->hs_descriptors)
usb_free_descriptors(f->hs_descriptors);
if (f->fs_descriptors)
usb_free_descriptors(f->fs_descriptors);
}
static int qdss_bind(struct usb_configuration *c, struct usb_function *f)
{
struct usb_gadget *gadget = c->cdev->gadget;
struct f_qdss *qdss = func_to_qdss(f);
struct usb_ep *ep;
int iface, id;
pr_debug("qdss_bind\n");
/* Allocate data I/F */
iface = usb_interface_id(c, f);
if (iface < 0) {
pr_err("interface allocation error\n");
return iface;
}
qdss_data_intf_desc.bInterfaceNumber = iface;
qdss->data_iface_id = iface;
id = usb_string_id(c->cdev);
if (id < 0)
return id;
qdss_string_defs[QDSS_DATA_IDX].id = id;
qdss_data_intf_desc.iInterface = id;
if (qdss->debug_inface_enabled) {
/* Allocate ctrl I/F */
iface = usb_interface_id(c, f);
if (iface < 0) {
pr_err("interface allocation error\n");
return iface;
}
qdss_ctrl_intf_desc.bInterfaceNumber = iface;
qdss->ctrl_iface_id = iface;
id = usb_string_id(c->cdev);
if (id < 0)
return id;
qdss_string_defs[QDSS_CTRL_IDX].id = id;
qdss_ctrl_intf_desc.iInterface = id;
}
/* for non-accelerated path keep tx fifo size 1k */
if (!strcmp(qdss->ch.name, USB_QDSS_CH_MDM))
qdss_data_ep_comp_desc.bMaxBurst = 0;
ep = usb_ep_autoconfig_ss(gadget, &qdss_ss_data_desc,
&qdss_data_ep_comp_desc);
if (!ep) {
pr_err("ep_autoconfig error\n");
goto fail;
}
qdss->port.data = ep;
ep->driver_data = qdss;
if (qdss->debug_inface_enabled) {
ep = usb_ep_autoconfig_ss(gadget, &qdss_ss_ctrl_in_desc,
&qdss_ctrl_in_ep_comp_desc);
if (!ep) {
pr_err("ep_autoconfig error\n");
goto fail;
}
qdss->port.ctrl_in = ep;
ep->driver_data = qdss;
ep = usb_ep_autoconfig_ss(gadget, &qdss_ss_ctrl_out_desc,
&qdss_ctrl_out_ep_comp_desc);
if (!ep) {
pr_err("ep_autoconfig error\n");
goto fail;
}
qdss->port.ctrl_out = ep;
ep->driver_data = qdss;
}
/*update fs descriptors*/
qdss_fs_data_desc.bEndpointAddress =
qdss_ss_data_desc.bEndpointAddress;
if (qdss->debug_inface_enabled) {
qdss_fs_ctrl_in_desc.bEndpointAddress =
qdss_ss_ctrl_in_desc.bEndpointAddress;
qdss_fs_ctrl_out_desc.bEndpointAddress =
qdss_ss_ctrl_out_desc.bEndpointAddress;
f->fs_descriptors = usb_copy_descriptors(qdss_fs_desc);
} else
f->fs_descriptors = usb_copy_descriptors(
qdss_fs_data_only_desc);
/*update descriptors*/
qdss_hs_data_desc.bEndpointAddress =
qdss_ss_data_desc.bEndpointAddress;
if (qdss->debug_inface_enabled) {
qdss_hs_ctrl_in_desc.bEndpointAddress =
qdss_ss_ctrl_in_desc.bEndpointAddress;
qdss_hs_ctrl_out_desc.bEndpointAddress =
qdss_ss_ctrl_out_desc.bEndpointAddress;
f->hs_descriptors = usb_copy_descriptors(qdss_hs_desc);
} else
f->hs_descriptors = usb_copy_descriptors(
qdss_hs_data_only_desc);
if (!f->hs_descriptors) {
pr_err("usb_copy_descriptors error\n");
goto fail;
}
/* update ss descriptors */
if (gadget_is_superspeed(gadget)) {
if (qdss->debug_inface_enabled)
f->ss_descriptors =
usb_copy_descriptors(qdss_ss_desc);
else
f->ss_descriptors =
usb_copy_descriptors(qdss_ss_data_only_desc);
if (!f->ss_descriptors) {
pr_err("usb_copy_descriptors error\n");
goto fail;
}
}
c->cdev->gadget->bam2bam_func_enabled = true;
return 0;
fail:
clear_eps(f);
clear_desc(gadget, f);
return -ENOTSUPP;
}
static void qdss_unbind(struct usb_configuration *c, struct usb_function *f)
{
struct f_qdss *qdss = func_to_qdss(f);
struct usb_gadget *gadget = c->cdev->gadget;
pr_debug("qdss_unbind\n");
flush_workqueue(qdss->wq);
c->cdev->gadget->bam2bam_func_enabled = false;
clear_eps(f);
clear_desc(gadget, f);
}
static void qdss_eps_disable(struct usb_function *f)
{
struct f_qdss *qdss = func_to_qdss(f);
pr_debug("qdss_eps_disable\n");
if (qdss->ctrl_in_enabled) {
usb_ep_disable(qdss->port.ctrl_in);
qdss->ctrl_in_enabled = 0;
}
if (qdss->ctrl_out_enabled) {
usb_ep_disable(qdss->port.ctrl_out);
qdss->ctrl_out_enabled = 0;
}
if (qdss->data_enabled) {
usb_ep_disable(qdss->port.data);
qdss->data_enabled = 0;
}
}
static void usb_qdss_disconnect_work(struct work_struct *work)
{
struct f_qdss *qdss;
int status;
unsigned long flags;
qdss = container_of(work, struct f_qdss, disconnect_w);
pr_debug("usb_qdss_disconnect_work\n");
/* Notify qdss to cancel all active transfers */
if (qdss->ch.notify)
qdss->ch.notify(qdss->ch.priv,
USB_QDSS_DISCONNECT,
NULL,
NULL);
/* Uninitialized init data i.e. ep specific operation */
if (qdss->ch.app_conn && !strcmp(qdss->ch.name, USB_QDSS_CH_MSM)) {
status = uninit_data(qdss->port.data);
if (status)
pr_err("%s: uninit_data error\n", __func__);
status = set_qdss_data_connection(qdss, 0);
if (status)
pr_err("qdss_disconnect error");
spin_lock_irqsave(&qdss->lock, flags);
if (qdss->endless_req) {
usb_ep_free_request(qdss->port.data,
qdss->endless_req);
qdss->endless_req = NULL;
}
spin_unlock_irqrestore(&qdss->lock, flags);
}
/*
* Decrement usage count which was incremented
* before calling connect work
*/
usb_gadget_autopm_put_async(qdss->gadget);
}
static void qdss_disable(struct usb_function *f)
{
struct f_qdss *qdss = func_to_qdss(f);
unsigned long flags;
pr_debug("qdss_disable\n");
spin_lock_irqsave(&qdss->lock, flags);
if (!qdss->usb_connected) {
spin_unlock_irqrestore(&qdss->lock, flags);
return;
}
qdss->usb_connected = 0;
spin_unlock_irqrestore(&qdss->lock, flags);
/*cancell all active xfers*/
qdss_eps_disable(f);
queue_work(qdss->wq, &qdss->disconnect_w);
}
static void usb_qdss_connect_work(struct work_struct *work)
{
struct f_qdss *qdss;
int status;
struct usb_request *req = NULL;
unsigned long flags;
qdss = container_of(work, struct f_qdss, connect_w);
/* If cable is already removed, discard connect_work */
if (qdss->usb_connected == 0) {
pr_debug("%s: discard connect_work\n", __func__);
cancel_work_sync(&qdss->disconnect_w);
return;
}
pr_debug("usb_qdss_connect_work\n");
if (!strcmp(qdss->ch.name, USB_QDSS_CH_MDM))
goto notify;
status = set_qdss_data_connection(qdss, 1);
if (status) {
pr_err("set_qdss_data_connection error(%d)", status);
return;
}
spin_lock_irqsave(&qdss->lock, flags);
req = qdss->endless_req;
spin_unlock_irqrestore(&qdss->lock, flags);
if (!req)
return;
status = usb_ep_queue(qdss->port.data, req, GFP_ATOMIC);
if (status) {
pr_err("%s: usb_ep_queue error (%d)\n", __func__, status);
return;
}
notify:
if (qdss->ch.notify)
qdss->ch.notify(qdss->ch.priv, USB_QDSS_CONNECT,
NULL, &qdss->ch);
}
static int qdss_set_alt(struct usb_function *f, unsigned int intf,
unsigned int alt)
{
struct f_qdss *qdss = func_to_qdss(f);
struct usb_gadget *gadget = f->config->cdev->gadget;
struct usb_qdss_ch *ch = &qdss->ch;
int ret = 0;
pr_debug("qdss_set_alt qdss pointer = %pK\n", qdss);
qdss->gadget = gadget;
if (alt != 0)
goto fail1;
if (gadget->speed != USB_SPEED_SUPER &&
gadget->speed != USB_SPEED_HIGH) {
pr_err("qdss_st_alt: qdss supportes HS or SS only\n");
ret = -EINVAL;
goto fail1;
}
if (intf == qdss->data_iface_id && !qdss->data_enabled) {
/* Increment usage count on connect */
usb_gadget_autopm_get_async(qdss->gadget);
if (config_ep_by_speed(gadget, f, qdss->port.data)) {
ret = -EINVAL;
goto fail;
}
ret = usb_ep_enable(qdss->port.data);
if (ret)
goto fail;
qdss->port.data->driver_data = qdss;
qdss->data_enabled = 1;
} else if ((intf == qdss->ctrl_iface_id) &&
(qdss->debug_inface_enabled)) {
if (config_ep_by_speed(gadget, f, qdss->port.ctrl_in)) {
ret = -EINVAL;
goto fail1;
}
ret = usb_ep_enable(qdss->port.ctrl_in);
if (ret)
goto fail1;
qdss->port.ctrl_in->driver_data = qdss;
qdss->ctrl_in_enabled = 1;
if (config_ep_by_speed(gadget, f, qdss->port.ctrl_out)) {
ret = -EINVAL;
goto fail1;
}
ret = usb_ep_enable(qdss->port.ctrl_out);
if (ret)
goto fail1;
qdss->port.ctrl_out->driver_data = qdss;
qdss->ctrl_out_enabled = 1;
}
if (qdss->debug_inface_enabled) {
if (qdss->ctrl_out_enabled && qdss->ctrl_in_enabled &&
qdss->data_enabled) {
qdss->usb_connected = 1;
pr_debug("qdss_set_alt usb_connected INTF enabled\n");
}
} else {
if (qdss->data_enabled) {
qdss->usb_connected = 1;
pr_debug("qdss_set_alt usb_connected INTF disabled\n");
}
}
if (qdss->usb_connected && ch->app_conn)
queue_work(qdss->wq, &qdss->connect_w);
return 0;
fail:
/* Decrement usage count in case of failure */
usb_gadget_autopm_put_async(qdss->gadget);
fail1:
pr_err("qdss_set_alt failed\n");
qdss_eps_disable(f);
return ret;
}
static struct f_qdss *alloc_usb_qdss(char *channel_name)
{
struct f_qdss *qdss;
int found = 0;
struct usb_qdss_ch *ch;
unsigned long flags;
spin_lock_irqsave(&qdss_lock, flags);
list_for_each_entry(ch, &usb_qdss_ch_list, list) {
if (!strcmp(channel_name, ch->name)) {
found = 1;
break;
}
}
if (found) {
spin_unlock_irqrestore(&qdss_lock, flags);
pr_err("%s: (%s) is already available.\n",
__func__, channel_name);
return ERR_PTR(-EEXIST);
}
spin_unlock_irqrestore(&qdss_lock, flags);
qdss = kzalloc(sizeof(struct f_qdss), GFP_KERNEL);
if (!qdss) {
pr_err("%s: Unable to allocate qdss device\n", __func__);
return ERR_PTR(-ENOMEM);
}
qdss->wq = create_singlethread_workqueue(channel_name);
if (!qdss->wq) {
kfree(qdss);
return ERR_PTR(-ENOMEM);
}
spin_lock_irqsave(&qdss_lock, flags);
ch = &qdss->ch;
ch->name = channel_name;
list_add_tail(&ch->list, &usb_qdss_ch_list);
spin_unlock_irqrestore(&qdss_lock, flags);
spin_lock_init(&qdss->lock);
INIT_LIST_HEAD(&qdss->ctrl_read_pool);
INIT_LIST_HEAD(&qdss->ctrl_write_pool);
INIT_LIST_HEAD(&qdss->data_write_pool);
INIT_WORK(&qdss->connect_w, usb_qdss_connect_work);
INIT_WORK(&qdss->disconnect_w, usb_qdss_disconnect_work);
return qdss;
}
int usb_qdss_ctrl_read(struct usb_qdss_ch *ch, struct qdss_request *d_req)
{
struct f_qdss *qdss = ch->priv_usb;
unsigned long flags;
struct usb_request *req = NULL;
pr_debug("usb_qdss_ctrl_read\n");
if (!qdss)
return -ENODEV;
spin_lock_irqsave(&qdss->lock, flags);
if (qdss->usb_connected == 0) {
spin_unlock_irqrestore(&qdss->lock, flags);
return -EIO;
}
if (list_empty(&qdss->ctrl_read_pool)) {
spin_unlock_irqrestore(&qdss->lock, flags);
pr_err("error: usb_qdss_ctrl_read list is empty\n");
return -EAGAIN;
}
req = list_first_entry(&qdss->ctrl_read_pool, struct usb_request, list);
list_del(&req->list);
spin_unlock_irqrestore(&qdss->lock, flags);
req->buf = d_req->buf;
req->length = d_req->length;
req->context = d_req;
if (usb_ep_queue(qdss->port.ctrl_out, req, GFP_ATOMIC)) {
/* If error add the link to linked list again*/
spin_lock_irqsave(&qdss->lock, flags);
list_add_tail(&req->list, &qdss->ctrl_read_pool);
spin_unlock_irqrestore(&qdss->lock, flags);
pr_err("qdss usb_ep_queue failed\n");
return -EIO;
}
return 0;
}
EXPORT_SYMBOL(usb_qdss_ctrl_read);
int usb_qdss_ctrl_write(struct usb_qdss_ch *ch, struct qdss_request *d_req)
{
struct f_qdss *qdss = ch->priv_usb;
unsigned long flags;
struct usb_request *req = NULL;
pr_debug("usb_qdss_ctrl_write\n");
if (!qdss)
return -ENODEV;
spin_lock_irqsave(&qdss->lock, flags);
if (qdss->usb_connected == 0) {
spin_unlock_irqrestore(&qdss->lock, flags);
return -EIO;
}
if (list_empty(&qdss->ctrl_write_pool)) {
pr_err("error: usb_qdss_ctrl_write list is empty\n");
spin_unlock_irqrestore(&qdss->lock, flags);
return -EAGAIN;
}
req = list_first_entry(&qdss->ctrl_write_pool, struct usb_request,
list);
list_del(&req->list);
spin_unlock_irqrestore(&qdss->lock, flags);
req->buf = d_req->buf;
req->length = d_req->length;
req->context = d_req;
if (usb_ep_queue(qdss->port.ctrl_in, req, GFP_ATOMIC)) {
spin_lock_irqsave(&qdss->lock, flags);
list_add_tail(&req->list, &qdss->ctrl_write_pool);
spin_unlock_irqrestore(&qdss->lock, flags);
pr_err("qdss usb_ep_queue failed\n");
return -EIO;
}
return 0;
}
EXPORT_SYMBOL(usb_qdss_ctrl_write);
int usb_qdss_write(struct usb_qdss_ch *ch, struct qdss_request *d_req)
{
struct f_qdss *qdss = ch->priv_usb;
unsigned long flags;
struct usb_request *req = NULL;
pr_debug("usb_qdss_ctrl_write\n");
if (!qdss)
return -ENODEV;
spin_lock_irqsave(&qdss->lock, flags);
if (qdss->usb_connected == 0) {
spin_unlock_irqrestore(&qdss->lock, flags);
return -EIO;
}
if (list_empty(&qdss->data_write_pool)) {
pr_err("error: usb_qdss_data_write list is empty\n");
spin_unlock_irqrestore(&qdss->lock, flags);
return -EAGAIN;
}
req = list_first_entry(&qdss->data_write_pool, struct usb_request,
list);
list_del(&req->list);
spin_unlock_irqrestore(&qdss->lock, flags);
req->buf = d_req->buf;
req->length = d_req->length;
req->context = d_req;
if (usb_ep_queue(qdss->port.data, req, GFP_ATOMIC)) {
spin_lock_irqsave(&qdss->lock, flags);
list_add_tail(&req->list, &qdss->data_write_pool);
spin_unlock_irqrestore(&qdss->lock, flags);
pr_err("qdss usb_ep_queue failed\n");
return -EIO;
}
return 0;
}
EXPORT_SYMBOL(usb_qdss_write);
struct usb_qdss_ch *usb_qdss_open(const char *name, void *priv,
void (*notify)(void *priv, unsigned int event,
struct qdss_request *d_req, struct usb_qdss_ch *))
{
struct usb_qdss_ch *ch;
struct f_qdss *qdss;
unsigned long flags;
int found = 0;
pr_debug("usb_qdss_open\n");
if (!notify) {
pr_err("usb_qdss_open: notification func is missing\n");
return NULL;
}
spin_lock_irqsave(&qdss_lock, flags);
/* Check if we already have a channel with this name */
list_for_each_entry(ch, &usb_qdss_ch_list, list) {
if (!strcmp(name, ch->name)) {
found = 1;
break;
}
}
if (!found) {
spin_unlock_irqrestore(&qdss_lock, flags);
pr_debug("usb_qdss_open failed as %s not found\n", name);
return NULL;
}
pr_debug("usb_qdss_open: qdss ctx found\n");
qdss = container_of(ch, struct f_qdss, ch);
ch->priv_usb = qdss;
ch->priv = priv;
ch->notify = notify;
ch->app_conn = 1;
spin_unlock_irqrestore(&qdss_lock, flags);
/* the case USB cabel was connected before qdss called qdss_open */
if (qdss->usb_connected == 1)
queue_work(qdss->wq, &qdss->connect_w);
return ch;
}
EXPORT_SYMBOL(usb_qdss_open);
void usb_qdss_close(struct usb_qdss_ch *ch)
{
struct f_qdss *qdss = ch->priv_usb;
struct usb_gadget *gadget;
unsigned long flags;
int status;
pr_debug("usb_qdss_close\n");
spin_lock_irqsave(&qdss_lock, flags);
ch->priv_usb = NULL;
if (!qdss || !qdss->usb_connected ||
!strcmp(qdss->ch.name, USB_QDSS_CH_MDM)) {
ch->app_conn = 0;
spin_unlock_irqrestore(&qdss_lock, flags);
return;
}
if (qdss->endless_req) {
usb_ep_dequeue(qdss->port.data, qdss->endless_req);
usb_ep_free_request(qdss->port.data, qdss->endless_req);
qdss->endless_req = NULL;
}
gadget = qdss->gadget;
ch->app_conn = 0;
spin_unlock_irqrestore(&qdss_lock, flags);
status = uninit_data(qdss->port.data);
if (status)
pr_err("%s: uninit_data error\n", __func__);
status = set_qdss_data_connection(qdss, 0);
if (status)
pr_err("%s:qdss_disconnect error\n", __func__);
}
EXPORT_SYMBOL(usb_qdss_close);
static void qdss_cleanup(void)
{
struct f_qdss *qdss;
struct list_head *act, *tmp;
struct usb_qdss_ch *_ch;
unsigned long flags;
pr_debug("qdss_cleanup\n");
list_for_each_safe(act, tmp, &usb_qdss_ch_list) {
_ch = list_entry(act, struct usb_qdss_ch, list);
qdss = container_of(_ch, struct f_qdss, ch);
spin_lock_irqsave(&qdss_lock, flags);
destroy_workqueue(qdss->wq);
if (!_ch->priv) {
list_del(&_ch->list);
kfree(qdss);
}
spin_unlock_irqrestore(&qdss_lock, flags);
}
}
static void qdss_free_func(struct usb_function *f)
{
/* Do nothing as usb_qdss_alloc() doesn't alloc anything. */
}
static inline struct usb_qdss_opts *to_f_qdss_opts(struct config_item *item)
{
return container_of(to_config_group(item), struct usb_qdss_opts,
func_inst.group);
}
static void qdss_attr_release(struct config_item *item)
{
struct usb_qdss_opts *opts = to_f_qdss_opts(item);
usb_put_function_instance(&opts->func_inst);
}
static struct configfs_item_operations qdss_item_ops = {
.release = qdss_attr_release,
};
static ssize_t qdss_enable_debug_inface_show(struct config_item *item,
char *page)
{
return snprintf(page, PAGE_SIZE, "%s\n",
(to_f_qdss_opts(item)->usb_qdss->debug_inface_enabled == 1) ?
"Enabled" : "Disabled");
}
static ssize_t qdss_enable_debug_inface_store(struct config_item *item,
const char *page, size_t len)
{
struct f_qdss *qdss = to_f_qdss_opts(item)->usb_qdss;
unsigned long flags;
u8 stats;
if (page == NULL) {
pr_err("Invalid buffer");
return len;
}
if (kstrtou8(page, 0, &stats) != 0 && (stats != 0 || stats != 1)) {
pr_err("(%u)Wrong value. enter 0 to disable or 1 to enable.\n",
stats);
return len;
}
spin_lock_irqsave(&qdss->lock, flags);
qdss->debug_inface_enabled = (stats == 1 ? "true" : "false");
spin_unlock_irqrestore(&qdss->lock, flags);
return len;
}
CONFIGFS_ATTR(qdss_, enable_debug_inface);
static struct configfs_attribute *qdss_attrs[] = {
&qdss_attr_enable_debug_inface,
NULL,
};
static struct config_item_type qdss_func_type = {
.ct_item_ops = &qdss_item_ops,
.ct_attrs = qdss_attrs,
.ct_owner = THIS_MODULE,
};
static void usb_qdss_free_inst(struct usb_function_instance *fi)
{
struct usb_qdss_opts *opts;
struct usb_qdss_ch *ch;
unsigned long flags;
opts = container_of(fi, struct usb_qdss_opts, func_inst);
spin_lock_irqsave(&qdss_lock, flags);
list_for_each_entry(ch, &usb_qdss_ch_list, list) {
if (!strcmp(opts->channel_name, ch->name)) {
list_del(&ch->list);
break;
}
}
spin_unlock_irqrestore(&qdss_lock, flags);
kfree(opts->channel_name);
destroy_workqueue(opts->usb_qdss->wq);
kfree(opts->usb_qdss);
kfree(opts);
}
static int usb_qdss_set_inst_name(struct usb_function_instance *f,
const char *name)
{
struct usb_qdss_opts *opts =
container_of(f, struct usb_qdss_opts, func_inst);
char *ptr;
size_t name_len;
struct f_qdss *usb_qdss;
/* get channel_name as expected input qdss.<channel_name> */
name_len = strlen(name) + 1;
if (name_len > 15)
return -ENAMETOOLONG;
/* get channel name */
ptr = kstrndup(name, name_len, GFP_KERNEL);
if (!ptr) {
pr_err("error:%ld\n", PTR_ERR(ptr));
return -ENOMEM;
}
opts->channel_name = ptr;
pr_debug("qdss: channel_name:%s\n", opts->channel_name);
usb_qdss = alloc_usb_qdss(opts->channel_name);
if (IS_ERR(usb_qdss)) {
pr_err("Failed to create usb_qdss port(%s)\n",
opts->channel_name);
return -ENOMEM;
}
opts->usb_qdss = usb_qdss;
return 0;
}
static struct usb_function_instance *qdss_alloc_inst(void)
{
struct usb_qdss_opts *opts;
opts = kzalloc(sizeof(*opts), GFP_KERNEL);
if (!opts)
return ERR_PTR(-ENOMEM);
opts->func_inst.free_func_inst = usb_qdss_free_inst;
opts->func_inst.set_inst_name = usb_qdss_set_inst_name;
config_group_init_type_name(&opts->func_inst.group, "",
&qdss_func_type);
return &opts->func_inst;
}
static struct usb_function *qdss_alloc(struct usb_function_instance *fi)
{
struct usb_qdss_opts *opts = to_fi_usb_qdss_opts(fi);
struct f_qdss *usb_qdss = opts->usb_qdss;
usb_qdss->port.function.name = "usb_qdss";
usb_qdss->port.function.fs_descriptors = qdss_fs_desc;
usb_qdss->port.function.hs_descriptors = qdss_hs_desc;
usb_qdss->port.function.strings = qdss_strings;
usb_qdss->port.function.bind = qdss_bind;
usb_qdss->port.function.unbind = qdss_unbind;
usb_qdss->port.function.set_alt = qdss_set_alt;
usb_qdss->port.function.disable = qdss_disable;
usb_qdss->port.function.setup = NULL;
usb_qdss->port.function.free_func = qdss_free_func;
return &usb_qdss->port.function;
}
DECLARE_USB_FUNCTION(qdss, qdss_alloc_inst, qdss_alloc);
static int __init usb_qdss_init(void)
{
int ret;
INIT_LIST_HEAD(&usb_qdss_ch_list);
ret = usb_function_register(&qdssusb_func);
if (ret) {
pr_err("%s: failed to register diag %d\n", __func__, ret);
return ret;
}
return ret;
}
static void __exit usb_qdss_exit(void)
{
usb_function_unregister(&qdssusb_func);
qdss_cleanup();
}
module_init(usb_qdss_init);
module_exit(usb_qdss_exit);
MODULE_DESCRIPTION("USB QDSS Function Driver");