blob: 3b540f3cbef69999b83ec75bcc999b8e2544db7b [file] [log] [blame]
/* Copyright (c) 2014-2017, 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/err.h>
#include <linux/ipc_logging.h>
#include <linux/list.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/random.h>
#include <linux/uio.h>
#include <soc/qcom/glink.h>
#include <soc/qcom/tracer_pkt.h>
#include "glink_loopback_commands.h"
/* Number of internal IPC Logging log pages */
#define GLINK_LBSRV_NUM_LOG_PAGES 3
static void *glink_lbsrv_log_ctx;
#define GLINK_LBSRV_IPC_LOG_STR(x...) do { \
if (glink_lbsrv_log_ctx) \
ipc_log_string(glink_lbsrv_log_ctx, x); \
} while (0)
#define LBSRV_INFO(x...) GLINK_LBSRV_IPC_LOG_STR("<LBSRV> " x)
#define LBSRV_ERR(x...) do { \
pr_err("<LBSRV> " x); \
GLINK_LBSRV_IPC_LOG_STR("<LBSRV> " x); \
} while (0)
enum ch_type {
CTL,
DATA,
};
enum buf_type {
LINEAR,
VECTOR,
};
struct tx_config_info {
uint32_t random_delay;
uint32_t delay_ms;
uint32_t echo_count;
uint32_t transform_type;
};
struct rx_done_config_info {
uint32_t random_delay;
uint32_t delay_ms;
};
struct rmt_rx_intent_req_work_info {
size_t req_intent_size;
struct delayed_work work;
struct ch_info *work_ch_info;
};
struct queue_rx_intent_work_info {
uint32_t req_id;
bool deferred;
struct ch_info *req_ch_info;
uint32_t num_intents;
uint32_t intent_size;
uint32_t random_delay;
uint32_t delay_ms;
struct delayed_work work;
struct ch_info *work_ch_info;
};
struct lbsrv_vec {
uint32_t num_bufs;
struct kvec vec[0];
};
struct tx_work_info {
struct tx_config_info tx_config;
struct delayed_work work;
struct ch_info *tx_ch_info;
void *data;
bool tracer_pkt;
uint32_t buf_type;
size_t size;
void * (*vbuf_provider)(void *iovec, size_t offset, size_t *size);
void * (*pbuf_provider)(void *iovec, size_t offset, size_t *size);
};
struct rx_done_work_info {
struct delayed_work work;
struct ch_info *rx_done_ch_info;
void *ptr;
};
struct rx_work_info {
struct ch_info *rx_ch_info;
void *pkt_priv;
void *ptr;
bool tracer_pkt;
uint32_t buf_type;
size_t size;
void * (*vbuf_provider)(void *iovec, size_t offset, size_t *size);
void * (*pbuf_provider)(void *iovec, size_t offset, size_t *size);
struct delayed_work work;
};
struct ch_info {
struct list_head list;
struct mutex ch_info_lock;
char name[MAX_NAME_LEN];
char edge[GLINK_NAME_SIZE];
char transport[GLINK_NAME_SIZE];
void *handle;
bool fully_opened;
uint32_t type;
struct delayed_work open_work;
struct delayed_work close_work;
struct tx_config_info tx_config;
struct rx_done_config_info rx_done_config;
struct queue_rx_intent_work_info *queue_rx_intent_work_info;
};
struct ctl_ch_info {
char name[MAX_NAME_LEN];
char edge[GLINK_NAME_SIZE];
char transport[GLINK_NAME_SIZE];
};
static struct ctl_ch_info ctl_ch_tbl[] = {
{"LOCAL_LOOPBACK_SRV", "local", "lloop"},
{"LOOPBACK_CTL_APSS", "mpss", "smem"},
{"LOOPBACK_CTL_APSS", "lpass", "smem"},
{"LOOPBACK_CTL_APSS", "dsps", "smem"},
{"LOOPBACK_CTL_APSS", "spss", "mailbox"},
{"LOOPBACK_CTL_APSS", "wdsp", "spi"},
};
static DEFINE_MUTEX(ctl_ch_list_lock);
static LIST_HEAD(ctl_ch_list);
static DEFINE_MUTEX(data_ch_list_lock);
static LIST_HEAD(data_ch_list);
struct workqueue_struct *glink_lbsrv_wq;
/**
* link_state_work_info - Information about work handling link state updates
* edge: Remote subsystem name in the link.
* transport: Name of the transport/link.
* link_state: State of the transport/link.
* work: Reference to the work item.
*/
struct link_state_work_info {
char edge[GLINK_NAME_SIZE];
char transport[GLINK_NAME_SIZE];
enum glink_link_state link_state;
struct delayed_work work;
};
static void glink_lbsrv_link_state_cb(struct glink_link_state_cb_info *cb_info,
void *priv);
static struct glink_link_info glink_lbsrv_link_info = {
NULL, NULL, glink_lbsrv_link_state_cb};
static void *glink_lbsrv_link_state_notif_handle;
static void glink_lbsrv_open_worker(struct work_struct *work);
static void glink_lbsrv_close_worker(struct work_struct *work);
static void glink_lbsrv_rmt_rx_intent_req_worker(struct work_struct *work);
static void glink_lbsrv_queue_rx_intent_worker(struct work_struct *work);
static void glink_lbsrv_rx_worker(struct work_struct *work);
static void glink_lbsrv_rx_done_worker(struct work_struct *work);
static void glink_lbsrv_tx_worker(struct work_struct *work);
int glink_lbsrv_send_response(void *handle, uint32_t req_id, uint32_t req_type,
uint32_t response)
{
struct resp *resp_pkt = kzalloc(sizeof(struct resp), GFP_KERNEL);
if (!resp_pkt) {
LBSRV_ERR("%s: Error allocating response packet\n", __func__);
return -ENOMEM;
}
resp_pkt->req_id = req_id;
resp_pkt->req_type = req_type;
resp_pkt->response = response;
return glink_tx(handle, (void *)LINEAR, (void *)resp_pkt,
sizeof(struct resp), 0);
}
static uint32_t calc_delay_ms(uint32_t random_delay, uint32_t delay_ms)
{
uint32_t tmp_delay_ms;
if (random_delay && delay_ms)
tmp_delay_ms = prandom_u32() % delay_ms;
else if (random_delay)
tmp_delay_ms = prandom_u32();
else
tmp_delay_ms = delay_ms;
return tmp_delay_ms;
}
static int create_ch_info(char *name, char *edge, char *transport,
uint32_t type, struct ch_info **ret_ch_info)
{
struct ch_info *tmp_ch_info;
tmp_ch_info = kzalloc(sizeof(struct ch_info), GFP_KERNEL);
if (!tmp_ch_info) {
LBSRV_ERR("%s: Error allocation ch_info\n", __func__);
return -ENOMEM;
}
INIT_LIST_HEAD(&tmp_ch_info->list);
mutex_init(&tmp_ch_info->ch_info_lock);
strlcpy(tmp_ch_info->name, name, MAX_NAME_LEN);
strlcpy(tmp_ch_info->edge, edge, GLINK_NAME_SIZE);
strlcpy(tmp_ch_info->transport, transport, GLINK_NAME_SIZE);
tmp_ch_info->type = type;
INIT_DELAYED_WORK(&tmp_ch_info->open_work,
glink_lbsrv_open_worker);
INIT_DELAYED_WORK(&tmp_ch_info->close_work,
glink_lbsrv_close_worker);
tmp_ch_info->tx_config.echo_count = 1;
if (type == CTL) {
mutex_lock(&ctl_ch_list_lock);
list_add_tail(&tmp_ch_info->list, &ctl_ch_list);
mutex_unlock(&ctl_ch_list_lock);
} else if (type == DATA) {
mutex_lock(&data_ch_list_lock);
list_add_tail(&tmp_ch_info->list, &data_ch_list);
mutex_unlock(&data_ch_list_lock);
} else {
LBSRV_ERR("%s:%s:%s %s: Invalid ch type %d\n", transport,
edge, name, __func__, type);
kfree(tmp_ch_info);
return -EINVAL;
}
*ret_ch_info = tmp_ch_info;
return 0;
}
struct ch_info *lookup_ch_list(char *name, char *edge, char *transport,
uint32_t type)
{
struct list_head *ch_list;
struct mutex *lock;
struct ch_info *tmp_ch_info;
if (type == DATA) {
ch_list = &data_ch_list;
lock = &data_ch_list_lock;
} else if (type == CTL) {
ch_list = &ctl_ch_list;
lock = &ctl_ch_list_lock;
} else {
LBSRV_ERR("%s:%s:%s %s: Invalid ch type %d\n", transport,
edge, name, __func__, type);
return NULL;
}
mutex_lock(lock);
list_for_each_entry(tmp_ch_info, ch_list, list) {
if (!strcmp(name, tmp_ch_info->name) &&
!strcmp(edge, tmp_ch_info->edge) &&
!strcmp(transport, tmp_ch_info->transport)) {
mutex_unlock(lock);
return tmp_ch_info;
}
}
mutex_unlock(lock);
return NULL;
}
int glink_lbsrv_handle_open_req(struct ch_info *rx_ch_info,
struct open_req req)
{
struct ch_info *tmp_ch_info;
int ret;
char name[MAX_NAME_LEN];
char *temp;
strlcpy(name, req.ch_name, MAX_NAME_LEN);
if (!strcmp(rx_ch_info->transport, "lloop")) {
temp = strnstr(name, "_CLNT", MAX_NAME_LEN);
if (temp)
*temp = '\0';
strlcat(name, "_SRV", MAX_NAME_LEN);
}
LBSRV_INFO("%s:%s:%s %s: delay_ms[%d]\n",
rx_ch_info->transport, rx_ch_info->edge,
name, __func__, req.delay_ms);
tmp_ch_info = lookup_ch_list(name, rx_ch_info->edge,
rx_ch_info->transport, DATA);
if (tmp_ch_info)
goto queue_open_work;
ret = create_ch_info(name, rx_ch_info->edge, rx_ch_info->transport,
DATA, &tmp_ch_info);
if (ret)
return ret;
queue_open_work:
queue_delayed_work(glink_lbsrv_wq, &tmp_ch_info->open_work,
msecs_to_jiffies(req.delay_ms));
return 0;
}
int glink_lbsrv_handle_close_req(struct ch_info *rx_ch_info,
struct close_req req)
{
struct ch_info *tmp_ch_info;
char name[MAX_NAME_LEN];
char *temp;
strlcpy(name, req.ch_name, MAX_NAME_LEN);
if (!strcmp(rx_ch_info->transport, "lloop")) {
temp = strnstr(name, "_CLNT", MAX_NAME_LEN);
if (temp)
*temp = '\0';
strlcat(name, "_SRV", MAX_NAME_LEN);
}
LBSRV_INFO("%s:%s:%s %s: delay_ms[%d]\n",
rx_ch_info->transport, rx_ch_info->edge,
name, __func__, req.delay_ms);
tmp_ch_info = lookup_ch_list(name, rx_ch_info->edge,
rx_ch_info->transport, DATA);
if (tmp_ch_info)
queue_delayed_work(glink_lbsrv_wq, &tmp_ch_info->close_work,
msecs_to_jiffies(req.delay_ms));
return 0;
}
int glink_lbsrv_handle_queue_rx_intent_config_req(struct ch_info *rx_ch_info,
struct queue_rx_intent_config_req req, uint32_t req_id)
{
struct ch_info *tmp_ch_info;
struct queue_rx_intent_work_info *tmp_work_info;
char name[MAX_NAME_LEN];
char *temp;
uint32_t delay_ms;
strlcpy(name, req.ch_name, MAX_NAME_LEN);
if (!strcmp(rx_ch_info->transport, "lloop")) {
temp = strnstr(name, "_CLNT", MAX_NAME_LEN);
if (temp)
*temp = '\0';
strlcat(name, "_SRV", MAX_NAME_LEN);
}
LBSRV_INFO("%s:%s:%s %s: num_intents[%d] size[%d]\n",
rx_ch_info->transport, rx_ch_info->edge, name, __func__,
req.num_intents, req.intent_size);
tmp_ch_info = lookup_ch_list(name, rx_ch_info->edge,
rx_ch_info->transport, DATA);
if (!tmp_ch_info) {
LBSRV_ERR("%s:%s:%s %s: Channel info not found\n",
rx_ch_info->transport, rx_ch_info->edge,
name, __func__);
return -EINVAL;
}
tmp_work_info = kzalloc(sizeof(struct queue_rx_intent_work_info),
GFP_KERNEL);
if (!tmp_work_info) {
LBSRV_ERR("%s: Error allocating work_info\n", __func__);
return -ENOMEM;
}
tmp_work_info->req_id = req_id;
tmp_work_info->req_ch_info = rx_ch_info;
tmp_work_info->num_intents = req.num_intents;
tmp_work_info->intent_size = req.intent_size;
tmp_work_info->random_delay = req.random_delay;
tmp_work_info->delay_ms = req.delay_ms;
INIT_DELAYED_WORK(&tmp_work_info->work,
glink_lbsrv_queue_rx_intent_worker);
tmp_work_info->work_ch_info = tmp_ch_info;
mutex_lock(&tmp_ch_info->ch_info_lock);
if (tmp_ch_info->fully_opened) {
mutex_unlock(&tmp_ch_info->ch_info_lock);
delay_ms = calc_delay_ms(tmp_work_info->random_delay,
tmp_work_info->delay_ms);
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work,
msecs_to_jiffies(delay_ms));
if (tmp_work_info->random_delay || tmp_work_info->delay_ms)
glink_lbsrv_send_response(rx_ch_info->handle, req_id,
QUEUE_RX_INTENT_CONFIG, 0);
} else {
tmp_work_info->deferred = true;
tmp_ch_info->queue_rx_intent_work_info = tmp_work_info;
mutex_unlock(&tmp_ch_info->ch_info_lock);
glink_lbsrv_send_response(rx_ch_info->handle, req_id,
QUEUE_RX_INTENT_CONFIG, 0);
}
return 0;
}
int glink_lbsrv_handle_tx_config_req(struct ch_info *rx_ch_info,
struct tx_config_req req)
{
struct ch_info *tmp_ch_info;
char name[MAX_NAME_LEN];
char *temp;
strlcpy(name, req.ch_name, MAX_NAME_LEN);
if (!strcmp(rx_ch_info->transport, "lloop")) {
temp = strnstr(name, "_CLNT", MAX_NAME_LEN);
if (temp)
*temp = '\0';
strlcat(name, "_SRV", MAX_NAME_LEN);
}
LBSRV_INFO("%s:%s:%s %s: echo_count[%d] transform[%d]\n",
rx_ch_info->transport, rx_ch_info->edge, name, __func__,
req.echo_count, req.transform_type);
tmp_ch_info = lookup_ch_list(name, rx_ch_info->edge,
rx_ch_info->transport, DATA);
if (!tmp_ch_info) {
LBSRV_ERR("%s:%s:%s %s: Channel info not found\n",
rx_ch_info->transport, rx_ch_info->edge,
name, __func__);
return -EINVAL;
}
mutex_lock(&tmp_ch_info->ch_info_lock);
tmp_ch_info->tx_config.random_delay = req.random_delay;
tmp_ch_info->tx_config.delay_ms = req.delay_ms;
tmp_ch_info->tx_config.echo_count = req.echo_count;
tmp_ch_info->tx_config.transform_type = req.transform_type;
mutex_unlock(&tmp_ch_info->ch_info_lock);
return 0;
}
int glink_lbsrv_handle_rx_done_config_req(struct ch_info *rx_ch_info,
struct rx_done_config_req req)
{
struct ch_info *tmp_ch_info;
char name[MAX_NAME_LEN];
char *temp;
strlcpy(name, req.ch_name, MAX_NAME_LEN);
if (!strcmp(rx_ch_info->transport, "lloop")) {
temp = strnstr(name, "_CLNT", MAX_NAME_LEN);
if (temp)
*temp = '\0';
strlcat(name, "_SRV", MAX_NAME_LEN);
}
LBSRV_INFO("%s:%s:%s %s: delay_ms[%d] random_delay[%d]\n",
rx_ch_info->transport, rx_ch_info->edge, name,
__func__, req.delay_ms, req.random_delay);
tmp_ch_info = lookup_ch_list(name, rx_ch_info->edge,
rx_ch_info->transport, DATA);
if (!tmp_ch_info) {
LBSRV_ERR("%s:%s:%s %s: Channel info not found\n",
rx_ch_info->transport, rx_ch_info->edge,
name, __func__);
return -EINVAL;
}
mutex_lock(&tmp_ch_info->ch_info_lock);
tmp_ch_info->rx_done_config.random_delay = req.random_delay;
tmp_ch_info->rx_done_config.delay_ms = req.delay_ms;
mutex_unlock(&tmp_ch_info->ch_info_lock);
return 0;
}
/**
* glink_lbsrv_handle_req() - Handle the request commands received by clients
*
* rx_ch_info: Channel info on which the request is received
* pkt: Request structure received from client
*
* This function handles the all supported request types received from client
* and send the response back to client
*/
void glink_lbsrv_handle_req(struct ch_info *rx_ch_info, struct req pkt)
{
int ret;
LBSRV_INFO("%s:%s:%s %s: Request packet type[%d]:id[%d]\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__, pkt.hdr.req_type,
pkt.hdr.req_id);
switch (pkt.hdr.req_type) {
case OPEN:
ret = glink_lbsrv_handle_open_req(rx_ch_info,
pkt.payload.open);
break;
case CLOSE:
ret = glink_lbsrv_handle_close_req(rx_ch_info,
pkt.payload.close);
break;
case QUEUE_RX_INTENT_CONFIG:
ret = glink_lbsrv_handle_queue_rx_intent_config_req(
rx_ch_info, pkt.payload.q_rx_int_conf, pkt.hdr.req_id);
break;
case TX_CONFIG:
ret = glink_lbsrv_handle_tx_config_req(rx_ch_info,
pkt.payload.tx_conf);
break;
case RX_DONE_CONFIG:
ret = glink_lbsrv_handle_rx_done_config_req(rx_ch_info,
pkt.payload.rx_done_conf);
break;
default:
LBSRV_ERR("%s:%s:%s %s: Invalid Request type [%d]\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__, pkt.hdr.req_type);
ret = -1;
break;
}
if (pkt.hdr.req_type != QUEUE_RX_INTENT_CONFIG)
glink_lbsrv_send_response(rx_ch_info->handle, pkt.hdr.req_id,
pkt.hdr.req_type, ret);
}
static void *glink_lbsrv_vbuf_provider(void *iovec, size_t offset,
size_t *buf_size)
{
struct lbsrv_vec *tmp_vec_info = (struct lbsrv_vec *)iovec;
uint32_t i;
size_t temp_size = 0;
for (i = 0; i < tmp_vec_info->num_bufs; i++) {
temp_size += tmp_vec_info->vec[i].iov_len;
if (offset >= temp_size)
continue;
*buf_size = temp_size - offset;
return (void *)tmp_vec_info->vec[i].iov_base +
tmp_vec_info->vec[i].iov_len - *buf_size;
}
*buf_size = 0;
return NULL;
}
static void glink_lbsrv_free_data(void *data, uint32_t buf_type)
{
struct lbsrv_vec *tmp_vec_info;
uint32_t i;
if (buf_type == LINEAR) {
kfree(data);
} else {
tmp_vec_info = (struct lbsrv_vec *)data;
for (i = 0; i < tmp_vec_info->num_bufs; i++) {
kfree(tmp_vec_info->vec[i].iov_base);
tmp_vec_info->vec[i].iov_base = NULL;
}
kfree(tmp_vec_info);
}
}
static void *copy_linear_data(struct rx_work_info *tmp_rx_work_info)
{
char *data;
struct ch_info *rx_ch_info = tmp_rx_work_info->rx_ch_info;
data = kmalloc(tmp_rx_work_info->size, GFP_KERNEL);
if (data)
memcpy(data, tmp_rx_work_info->ptr, tmp_rx_work_info->size);
else
LBSRV_ERR("%s:%s:%s %s: Error allocating the data\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
return data;
}
static void *copy_vector_data(struct rx_work_info *tmp_rx_work_info)
{
uint32_t num_bufs = 0;
struct ch_info *rx_ch_info = tmp_rx_work_info->rx_ch_info;
struct lbsrv_vec *tmp_vec_info;
void *buf, *pbuf, *dest_buf;
size_t offset = 0;
size_t buf_size;
uint32_t i;
do {
if (tmp_rx_work_info->vbuf_provider)
buf = tmp_rx_work_info->vbuf_provider(
tmp_rx_work_info->ptr, offset, &buf_size);
else
buf = tmp_rx_work_info->pbuf_provider(
tmp_rx_work_info->ptr, offset, &buf_size);
if (!buf)
break;
offset += buf_size;
num_bufs++;
} while (buf);
tmp_vec_info = kzalloc(sizeof(*tmp_vec_info) +
num_bufs * sizeof(struct kvec), GFP_KERNEL);
if (!tmp_vec_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating vector info\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
return NULL;
}
tmp_vec_info->num_bufs = num_bufs;
offset = 0;
for (i = 0; i < num_bufs; i++) {
if (tmp_rx_work_info->vbuf_provider) {
buf = tmp_rx_work_info->vbuf_provider(
tmp_rx_work_info->ptr, offset, &buf_size);
} else {
pbuf = tmp_rx_work_info->pbuf_provider(
tmp_rx_work_info->ptr, offset, &buf_size);
buf = phys_to_virt((unsigned long)pbuf);
}
dest_buf = kmalloc(buf_size, GFP_KERNEL);
if (!dest_buf) {
LBSRV_ERR("%s:%s:%s %s: Error allocating data\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
goto out_copy_vector_data;
}
memcpy(dest_buf, buf, buf_size);
tmp_vec_info->vec[i].iov_base = dest_buf;
tmp_vec_info->vec[i].iov_len = buf_size;
offset += buf_size;
}
return tmp_vec_info;
out_copy_vector_data:
glink_lbsrv_free_data((void *)tmp_vec_info, VECTOR);
return NULL;
}
static void *glink_lbsrv_copy_data(struct rx_work_info *tmp_rx_work_info)
{
if (tmp_rx_work_info->buf_type == LINEAR)
return copy_linear_data(tmp_rx_work_info);
else
return copy_vector_data(tmp_rx_work_info);
}
static int glink_lbsrv_handle_data(struct rx_work_info *tmp_rx_work_info)
{
void *data;
int ret;
struct ch_info *rx_ch_info = tmp_rx_work_info->rx_ch_info;
struct tx_work_info *tmp_tx_work_info;
struct rx_done_work_info *tmp_rx_done_work_info;
uint32_t delay_ms;
data = glink_lbsrv_copy_data(tmp_rx_work_info);
if (!data) {
ret = -ENOMEM;
goto out_handle_data;
}
tmp_rx_done_work_info = kmalloc(sizeof(struct rx_done_work_info),
GFP_KERNEL);
if (!tmp_rx_done_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating rx_done_work_info\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
glink_lbsrv_free_data(data, tmp_rx_work_info->buf_type);
ret = -ENOMEM;
goto out_handle_data;
}
INIT_DELAYED_WORK(&tmp_rx_done_work_info->work,
glink_lbsrv_rx_done_worker);
tmp_rx_done_work_info->rx_done_ch_info = rx_ch_info;
tmp_rx_done_work_info->ptr = tmp_rx_work_info->ptr;
delay_ms = calc_delay_ms(rx_ch_info->rx_done_config.random_delay,
rx_ch_info->rx_done_config.delay_ms);
queue_delayed_work(glink_lbsrv_wq, &tmp_rx_done_work_info->work,
msecs_to_jiffies(delay_ms));
tmp_tx_work_info = kmalloc(sizeof(struct tx_work_info), GFP_KERNEL);
if (!tmp_tx_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating tx_work_info\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
glink_lbsrv_free_data(data, tmp_rx_work_info->buf_type);
return -ENOMEM;
}
mutex_lock(&rx_ch_info->ch_info_lock);
tmp_tx_work_info->tx_config.random_delay =
rx_ch_info->tx_config.random_delay;
tmp_tx_work_info->tx_config.delay_ms = rx_ch_info->tx_config.delay_ms;
tmp_tx_work_info->tx_config.echo_count =
rx_ch_info->tx_config.echo_count;
tmp_tx_work_info->tx_config.transform_type =
rx_ch_info->tx_config.transform_type;
mutex_unlock(&rx_ch_info->ch_info_lock);
INIT_DELAYED_WORK(&tmp_tx_work_info->work, glink_lbsrv_tx_worker);
tmp_tx_work_info->tx_ch_info = rx_ch_info;
tmp_tx_work_info->data = data;
tmp_tx_work_info->tracer_pkt = tmp_rx_work_info->tracer_pkt;
tmp_tx_work_info->buf_type = tmp_rx_work_info->buf_type;
tmp_tx_work_info->size = tmp_rx_work_info->size;
if (tmp_tx_work_info->buf_type == VECTOR)
tmp_tx_work_info->vbuf_provider = glink_lbsrv_vbuf_provider;
else
tmp_tx_work_info->vbuf_provider = NULL;
tmp_tx_work_info->pbuf_provider = NULL;
delay_ms = calc_delay_ms(tmp_tx_work_info->tx_config.random_delay,
tmp_tx_work_info->tx_config.delay_ms);
queue_delayed_work(glink_lbsrv_wq, &tmp_tx_work_info->work,
msecs_to_jiffies(delay_ms));
return 0;
out_handle_data:
glink_rx_done(rx_ch_info->handle, tmp_rx_work_info->ptr, false);
return ret;
}
void glink_lpbsrv_notify_rx(void *handle, const void *priv,
const void *pkt_priv, const void *ptr, size_t size)
{
struct rx_work_info *tmp_work_info;
struct ch_info *rx_ch_info = (struct ch_info *)priv;
LBSRV_INFO(
"%s:%s:%s %s: end (Success) RX priv[%p] data[%p] size[%zu]\n",
rx_ch_info->transport, rx_ch_info->edge, rx_ch_info->name,
__func__, pkt_priv, (char *)ptr, size);
tmp_work_info = kzalloc(sizeof(struct rx_work_info), GFP_ATOMIC);
if (!tmp_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating rx_work\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
return;
}
tmp_work_info->rx_ch_info = rx_ch_info;
tmp_work_info->pkt_priv = (void *)pkt_priv;
tmp_work_info->ptr = (void *)ptr;
tmp_work_info->buf_type = LINEAR;
tmp_work_info->size = size;
INIT_DELAYED_WORK(&tmp_work_info->work, glink_lbsrv_rx_worker);
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work, 0);
}
void glink_lpbsrv_notify_rxv(void *handle, const void *priv,
const void *pkt_priv, void *ptr, size_t size,
void * (*vbuf_provider)(void *iovec, size_t offset, size_t *size),
void * (*pbuf_provider)(void *iovec, size_t offset, size_t *size))
{
struct rx_work_info *tmp_work_info;
struct ch_info *rx_ch_info = (struct ch_info *)priv;
LBSRV_INFO("%s:%s:%s %s: priv[%p] data[%p] size[%zu]\n",
rx_ch_info->transport, rx_ch_info->edge, rx_ch_info->name,
__func__, pkt_priv, (char *)ptr, size);
tmp_work_info = kzalloc(sizeof(struct rx_work_info), GFP_ATOMIC);
if (!tmp_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating rx_work\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
return;
}
tmp_work_info->rx_ch_info = rx_ch_info;
tmp_work_info->pkt_priv = (void *)pkt_priv;
tmp_work_info->ptr = (void *)ptr;
tmp_work_info->buf_type = VECTOR;
tmp_work_info->size = size;
tmp_work_info->vbuf_provider = vbuf_provider;
tmp_work_info->pbuf_provider = pbuf_provider;
INIT_DELAYED_WORK(&tmp_work_info->work, glink_lbsrv_rx_worker);
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work, 0);
}
void glink_lpbsrv_notify_rx_tp(void *handle, const void *priv,
const void *pkt_priv, const void *ptr, size_t size)
{
struct rx_work_info *tmp_work_info;
struct ch_info *rx_ch_info = (struct ch_info *)priv;
LBSRV_INFO(
"%s:%s:%s %s: end (Success) RX priv[%p] data[%p] size[%zu]\n",
rx_ch_info->transport, rx_ch_info->edge, rx_ch_info->name,
__func__, pkt_priv, (char *)ptr, size);
tracer_pkt_log_event((void *)ptr, LOOPBACK_SRV_RX);
tmp_work_info = kmalloc(sizeof(struct rx_work_info), GFP_ATOMIC);
if (!tmp_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating rx_work\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__);
return;
}
tmp_work_info->rx_ch_info = rx_ch_info;
tmp_work_info->pkt_priv = (void *)pkt_priv;
tmp_work_info->ptr = (void *)ptr;
tmp_work_info->tracer_pkt = true;
tmp_work_info->buf_type = LINEAR;
tmp_work_info->size = size;
INIT_DELAYED_WORK(&tmp_work_info->work, glink_lbsrv_rx_worker);
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work, 0);
}
void glink_lpbsrv_notify_tx_done(void *handle, const void *priv,
const void *pkt_priv, const void *ptr)
{
struct ch_info *tx_done_ch_info = (struct ch_info *)priv;
LBSRV_INFO("%s:%s:%s %s: end (Success) TX_DONE ptr[%p]\n",
tx_done_ch_info->transport, tx_done_ch_info->edge,
tx_done_ch_info->name, __func__, ptr);
if (pkt_priv != (const void *)0xFFFFFFFF)
glink_lbsrv_free_data((void *)ptr,
(uint32_t)(uintptr_t)pkt_priv);
}
void glink_lpbsrv_notify_state(void *handle, const void *priv,
unsigned int event)
{
int ret;
uint32_t delay_ms;
struct ch_info *tmp_ch_info = (struct ch_info *)priv;
struct queue_rx_intent_work_info *tmp_work_info = NULL;
LBSRV_INFO("%s:%s:%s %s: event[%d]\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__, event);
if (tmp_ch_info->type == CTL) {
if (event == GLINK_CONNECTED) {
ret = glink_queue_rx_intent(handle,
priv, sizeof(struct req));
LBSRV_INFO(
"%s:%s:%s %s: QUEUE RX INTENT size[%zu] ret[%d]\n",
tmp_ch_info->transport,
tmp_ch_info->edge,
tmp_ch_info->name,
__func__, sizeof(struct req), ret);
} else if (event == GLINK_LOCAL_DISCONNECTED) {
queue_delayed_work(glink_lbsrv_wq,
&tmp_ch_info->open_work,
msecs_to_jiffies(0));
} else if (event == GLINK_REMOTE_DISCONNECTED)
if (!IS_ERR_OR_NULL(tmp_ch_info->handle))
queue_delayed_work(glink_lbsrv_wq,
&tmp_ch_info->close_work, 0);
} else if (tmp_ch_info->type == DATA) {
if (event == GLINK_CONNECTED) {
mutex_lock(&tmp_ch_info->ch_info_lock);
tmp_ch_info->fully_opened = true;
tmp_work_info = tmp_ch_info->queue_rx_intent_work_info;
tmp_ch_info->queue_rx_intent_work_info = NULL;
mutex_unlock(&tmp_ch_info->ch_info_lock);
if (tmp_work_info) {
delay_ms = calc_delay_ms(
tmp_work_info->random_delay,
tmp_work_info->delay_ms);
queue_delayed_work(glink_lbsrv_wq,
&tmp_work_info->work,
msecs_to_jiffies(delay_ms));
}
} else if (event == GLINK_LOCAL_DISCONNECTED ||
event == GLINK_REMOTE_DISCONNECTED) {
mutex_lock(&tmp_ch_info->ch_info_lock);
tmp_ch_info->fully_opened = false;
/*
* If the state has changed to LOCAL_DISCONNECTED,
* the channel has been fully closed and can now be
* re-opened. If the handle value is -EBUSY, an earlier
* open request failed because the channel was in the
* process of closing. Requeue the work from the open
* request.
*/
if (event == GLINK_LOCAL_DISCONNECTED &&
tmp_ch_info->handle == ERR_PTR(-EBUSY)) {
queue_delayed_work(glink_lbsrv_wq,
&tmp_ch_info->open_work,
msecs_to_jiffies(0));
}
if (event == GLINK_REMOTE_DISCONNECTED)
if (!IS_ERR_OR_NULL(tmp_ch_info->handle))
queue_delayed_work(
glink_lbsrv_wq,
&tmp_ch_info->close_work, 0);
mutex_unlock(&tmp_ch_info->ch_info_lock);
}
}
}
bool glink_lpbsrv_rmt_rx_intent_req_cb(void *handle, const void *priv,
size_t sz)
{
struct rmt_rx_intent_req_work_info *tmp_work_info;
struct ch_info *tmp_ch_info = (struct ch_info *)priv;
LBSRV_INFO("%s:%s:%s %s: QUEUE RX INTENT to receive size[%zu]\n",
tmp_ch_info->transport, tmp_ch_info->edge, tmp_ch_info->name,
__func__, sz);
tmp_work_info = kmalloc(sizeof(struct rmt_rx_intent_req_work_info),
GFP_ATOMIC);
if (!tmp_work_info) {
LBSRV_ERR("%s:%s:%s %s: Error allocating rx_work\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__);
return false;
}
tmp_work_info->req_intent_size = sz;
tmp_work_info->work_ch_info = tmp_ch_info;
INIT_DELAYED_WORK(&tmp_work_info->work,
glink_lbsrv_rmt_rx_intent_req_worker);
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work, 0);
return true;
}
void glink_lpbsrv_notify_rx_sigs(void *handle, const void *priv,
uint32_t old_sigs, uint32_t new_sigs)
{
LBSRV_INFO(" %s old_sigs[0x%x] New_sigs[0x%x]\n",
__func__, old_sigs, new_sigs);
glink_sigs_set(handle, new_sigs);
}
static void glink_lbsrv_rx_worker(struct work_struct *work)
{
struct delayed_work *rx_work = to_delayed_work(work);
struct rx_work_info *tmp_rx_work_info =
container_of(rx_work, struct rx_work_info, work);
struct ch_info *rx_ch_info = tmp_rx_work_info->rx_ch_info;
struct req request_pkt;
int ret;
if (rx_ch_info->type == CTL) {
request_pkt = *((struct req *)tmp_rx_work_info->ptr);
glink_rx_done(rx_ch_info->handle, tmp_rx_work_info->ptr, false);
ret = glink_queue_rx_intent(rx_ch_info->handle, rx_ch_info,
sizeof(struct req));
LBSRV_INFO("%s:%s:%s %s: QUEUE RX INTENT size[%zu] ret[%d]\n",
rx_ch_info->transport, rx_ch_info->edge,
rx_ch_info->name, __func__,
sizeof(struct req), ret);
glink_lbsrv_handle_req(rx_ch_info, request_pkt);
} else {
ret = glink_lbsrv_handle_data(tmp_rx_work_info);
}
kfree(tmp_rx_work_info);
}
static void glink_lbsrv_open_worker(struct work_struct *work)
{
struct delayed_work *open_work = to_delayed_work(work);
struct ch_info *tmp_ch_info =
container_of(open_work, struct ch_info, open_work);
struct glink_open_config open_cfg;
LBSRV_INFO("%s: glink_loopback_server_init\n", __func__);
mutex_lock(&tmp_ch_info->ch_info_lock);
if (!IS_ERR_OR_NULL(tmp_ch_info->handle)) {
mutex_unlock(&tmp_ch_info->ch_info_lock);
return;
}
memset(&open_cfg, 0, sizeof(struct glink_open_config));
open_cfg.transport = tmp_ch_info->transport;
open_cfg.edge = tmp_ch_info->edge;
open_cfg.name = tmp_ch_info->name;
open_cfg.notify_rx = glink_lpbsrv_notify_rx;
if (tmp_ch_info->type == DATA)
open_cfg.notify_rxv = glink_lpbsrv_notify_rxv;
open_cfg.notify_tx_done = glink_lpbsrv_notify_tx_done;
open_cfg.notify_state = glink_lpbsrv_notify_state;
open_cfg.notify_rx_intent_req = glink_lpbsrv_rmt_rx_intent_req_cb;
open_cfg.notify_rx_sigs = glink_lpbsrv_notify_rx_sigs;
open_cfg.notify_rx_abort = NULL;
open_cfg.notify_tx_abort = NULL;
open_cfg.notify_rx_tracer_pkt = glink_lpbsrv_notify_rx_tp;
open_cfg.priv = tmp_ch_info;
tmp_ch_info->handle = glink_open(&open_cfg);
if (IS_ERR_OR_NULL(tmp_ch_info->handle)) {
LBSRV_ERR("%s:%s:%s %s: unable to open channel\n",
open_cfg.transport, open_cfg.edge, open_cfg.name,
__func__);
mutex_unlock(&tmp_ch_info->ch_info_lock);
return;
}
mutex_unlock(&tmp_ch_info->ch_info_lock);
LBSRV_INFO("%s:%s:%s %s: Open complete\n", open_cfg.transport,
open_cfg.edge, open_cfg.name, __func__);
}
static void glink_lbsrv_close_worker(struct work_struct *work)
{
struct delayed_work *close_work = to_delayed_work(work);
struct ch_info *tmp_ch_info =
container_of(close_work, struct ch_info, close_work);
mutex_lock(&tmp_ch_info->ch_info_lock);
if (!IS_ERR_OR_NULL(tmp_ch_info->handle)) {
glink_close(tmp_ch_info->handle);
tmp_ch_info->handle = NULL;
}
mutex_unlock(&tmp_ch_info->ch_info_lock);
LBSRV_INFO("%s:%s:%s %s: Close complete\n", tmp_ch_info->transport,
tmp_ch_info->edge, tmp_ch_info->name, __func__);
}
static void glink_lbsrv_rmt_rx_intent_req_worker(struct work_struct *work)
{
struct delayed_work *rmt_rx_intent_req_work = to_delayed_work(work);
struct rmt_rx_intent_req_work_info *tmp_work_info =
container_of(rmt_rx_intent_req_work,
struct rmt_rx_intent_req_work_info, work);
struct ch_info *tmp_ch_info = tmp_work_info->work_ch_info;
int ret;
mutex_lock(&tmp_ch_info->ch_info_lock);
if (IS_ERR_OR_NULL(tmp_ch_info->handle)) {
mutex_unlock(&tmp_ch_info->ch_info_lock);
LBSRV_ERR("%s:%s:%s %s: Invalid CH handle\n",
tmp_ch_info->transport,
tmp_ch_info->edge,
tmp_ch_info->name, __func__);
kfree(tmp_work_info);
return;
}
ret = glink_queue_rx_intent(tmp_ch_info->handle,
(void *)tmp_ch_info, tmp_work_info->req_intent_size);
mutex_unlock(&tmp_ch_info->ch_info_lock);
LBSRV_INFO("%s:%s:%s %s: QUEUE RX INTENT size[%zu] ret[%d]\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__, tmp_work_info->req_intent_size,
ret);
if (ret < 0) {
LBSRV_ERR("%s:%s:%s %s: Err %d q'ing intent size %zu\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__, ret,
tmp_work_info->req_intent_size);
}
kfree(tmp_work_info);
}
static void glink_lbsrv_queue_rx_intent_worker(struct work_struct *work)
{
struct delayed_work *queue_rx_intent_work = to_delayed_work(work);
struct queue_rx_intent_work_info *tmp_work_info =
container_of(queue_rx_intent_work,
struct queue_rx_intent_work_info, work);
struct ch_info *tmp_ch_info = tmp_work_info->work_ch_info;
int ret;
uint32_t delay_ms;
while (1) {
mutex_lock(&tmp_ch_info->ch_info_lock);
if (IS_ERR_OR_NULL(tmp_ch_info->handle)) {
mutex_unlock(&tmp_ch_info->ch_info_lock);
return;
}
ret = glink_queue_rx_intent(tmp_ch_info->handle,
(void *)tmp_ch_info, tmp_work_info->intent_size);
mutex_unlock(&tmp_ch_info->ch_info_lock);
if (ret < 0) {
LBSRV_ERR("%s:%s:%s %s: Err %d q'ing intent size %d\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__, ret,
tmp_work_info->intent_size);
kfree(tmp_work_info);
return;
}
LBSRV_INFO("%s:%s:%s %s: Queued rx intent of size %d\n",
tmp_ch_info->transport, tmp_ch_info->edge,
tmp_ch_info->name, __func__,
tmp_work_info->intent_size);
tmp_work_info->num_intents--;
if (!tmp_work_info->num_intents)
break;
delay_ms = calc_delay_ms(tmp_work_info->random_delay,
tmp_work_info->delay_ms);
if (delay_ms) {
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work,
msecs_to_jiffies(delay_ms));
return;
}
}
LBSRV_INFO("%s:%s:%s %s: Queued all intents. size:%d\n",
tmp_ch_info->transport, tmp_ch_info->edge, tmp_ch_info->name,
__func__, tmp_work_info->intent_size);
if (!tmp_work_info->deferred && !tmp_work_info->random_delay &&
!tmp_work_info->delay_ms)
glink_lbsrv_send_response(tmp_work_info->req_ch_info->handle,
tmp_work_info->req_id, QUEUE_RX_INTENT_CONFIG,
0);
kfree(tmp_work_info);
}
static void glink_lbsrv_rx_done_worker(struct work_struct *work)
{
struct delayed_work *rx_done_work = to_delayed_work(work);
struct rx_done_work_info *tmp_work_info =
container_of(rx_done_work, struct rx_done_work_info, work);
struct ch_info *tmp_ch_info = tmp_work_info->rx_done_ch_info;
mutex_lock(&tmp_ch_info->ch_info_lock);
if (!IS_ERR_OR_NULL(tmp_ch_info->handle))
glink_rx_done(tmp_ch_info->handle, tmp_work_info->ptr, false);
mutex_unlock(&tmp_ch_info->ch_info_lock);
kfree(tmp_work_info);
}
static void glink_lbsrv_tx_worker(struct work_struct *work)
{
struct delayed_work *tx_work = to_delayed_work(work);
struct tx_work_info *tmp_work_info =
container_of(tx_work, struct tx_work_info, work);
struct ch_info *tmp_ch_info = tmp_work_info->tx_ch_info;
int ret;
uint32_t delay_ms;
uint32_t flags;
LBSRV_INFO("%s:%s:%s %s: start TX data[%p] size[%zu]\n",
tmp_ch_info->transport, tmp_ch_info->edge, tmp_ch_info->name,
__func__, tmp_work_info->data, tmp_work_info->size);
while (1) {
mutex_lock(&tmp_ch_info->ch_info_lock);
if (IS_ERR_OR_NULL(tmp_ch_info->handle)) {
mutex_unlock(&tmp_ch_info->ch_info_lock);
return;
}
flags = 0;
if (tmp_work_info->tracer_pkt) {
flags |= GLINK_TX_TRACER_PKT;
tracer_pkt_log_event(tmp_work_info->data,
LOOPBACK_SRV_TX);
}
if (tmp_work_info->buf_type == LINEAR)
ret = glink_tx(tmp_ch_info->handle,
(tmp_work_info->tx_config.echo_count > 1 ?
(void *)0xFFFFFFFF :
(void *)(uintptr_t)
tmp_work_info->buf_type),
(void *)tmp_work_info->data,
tmp_work_info->size, flags);
else
ret = glink_txv(tmp_ch_info->handle,
(tmp_work_info->tx_config.echo_count > 1 ?
(void *)0xFFFFFFFF :
(void *)(uintptr_t)
tmp_work_info->buf_type),
(void *)tmp_work_info->data,
tmp_work_info->size,
tmp_work_info->vbuf_provider,
tmp_work_info->pbuf_provider,
flags);
mutex_unlock(&tmp_ch_info->ch_info_lock);
if (ret < 0 && ret != -EAGAIN) {
LBSRV_ERR("%s:%s:%s %s: TX Error %d\n",
tmp_ch_info->transport,
tmp_ch_info->edge,
tmp_ch_info->name, __func__, ret);
glink_lbsrv_free_data(tmp_work_info->data,
tmp_work_info->buf_type);
kfree(tmp_work_info);
return;
}
if (ret != -EAGAIN)
tmp_work_info->tx_config.echo_count--;
if (!tmp_work_info->tx_config.echo_count)
break;
delay_ms = calc_delay_ms(tmp_work_info->tx_config.random_delay,
tmp_work_info->tx_config.delay_ms);
if (delay_ms) {
queue_delayed_work(glink_lbsrv_wq, &tmp_work_info->work,
msecs_to_jiffies(delay_ms));
return;
}
}
kfree(tmp_work_info);
}
/**
* glink_lbsrv_link_state_worker() - Function to handle link state updates
* work: Pointer to the work item in the link_state_work_info.
*
* This worker function is scheduled when there is a link state update. Since
* the loopback server registers for all transports, it receives all link state
* updates about all transports that get registered in the system.
*/
static void glink_lbsrv_link_state_worker(struct work_struct *work)
{
struct delayed_work *ls_work = to_delayed_work(work);
struct link_state_work_info *ls_info =
container_of(ls_work, struct link_state_work_info, work);
struct ch_info *tmp_ch_info;
if (ls_info->link_state == GLINK_LINK_STATE_UP) {
LBSRV_INFO("%s: LINK_STATE_UP %s:%s\n",
__func__, ls_info->edge, ls_info->transport);
mutex_lock(&ctl_ch_list_lock);
list_for_each_entry(tmp_ch_info, &ctl_ch_list, list) {
if (strcmp(tmp_ch_info->edge, ls_info->edge) ||
strcmp(tmp_ch_info->transport, ls_info->transport))
continue;
queue_delayed_work(glink_lbsrv_wq,
&tmp_ch_info->open_work, 0);
}
mutex_unlock(&ctl_ch_list_lock);
} else if (ls_info->link_state == GLINK_LINK_STATE_DOWN) {
LBSRV_INFO("%s: LINK_STATE_DOWN %s:%s\n",
__func__, ls_info->edge, ls_info->transport);
}
kfree(ls_info);
}
/**
* glink_lbsrv_link_state_cb() - Callback to receive link state updates
* cb_info: Information containing link & its state.
* priv: Private data passed during the link state registration.
*
* This function is called by the GLINK core to notify the loopback server
* regarding the link state updates. This function is registered with the
* GLINK core by the loopback server during glink_register_link_state_cb().
*/
static void glink_lbsrv_link_state_cb(struct glink_link_state_cb_info *cb_info,
void *priv)
{
struct link_state_work_info *ls_info;
if (!cb_info)
return;
LBSRV_INFO("%s: %s:%s\n", __func__, cb_info->edge, cb_info->transport);
ls_info = kmalloc(sizeof(*ls_info), GFP_KERNEL);
if (!ls_info) {
LBSRV_ERR("%s: Error allocating link state info\n", __func__);
return;
}
strlcpy(ls_info->edge, cb_info->edge, GLINK_NAME_SIZE);
strlcpy(ls_info->transport, cb_info->transport, GLINK_NAME_SIZE);
ls_info->link_state = cb_info->link_state;
INIT_DELAYED_WORK(&ls_info->work, glink_lbsrv_link_state_worker);
queue_delayed_work(glink_lbsrv_wq, &ls_info->work, 0);
}
static int glink_loopback_server_init(void)
{
int i;
int ret;
struct ch_info *tmp_ch_info;
glink_lbsrv_log_ctx = ipc_log_context_create(GLINK_LBSRV_NUM_LOG_PAGES,
"glink_lbsrv", 0);
if (!glink_lbsrv_log_ctx)
pr_err("%s: unable to create log context\n", __func__);
glink_lbsrv_wq = create_singlethread_workqueue("glink_lbsrv");
if (!glink_lbsrv_wq) {
LBSRV_ERR("%s: Error creating glink_lbsrv_wq\n", __func__);
return -EFAULT;
}
for (i = 0; i < ARRAY_SIZE(ctl_ch_tbl); i++) {
ret = create_ch_info(ctl_ch_tbl[i].name, ctl_ch_tbl[i].edge,
ctl_ch_tbl[i].transport, CTL,
&tmp_ch_info);
if (ret < 0) {
LBSRV_ERR("%s: Error creating ctl ch index %d\n",
__func__, i);
continue;
}
}
glink_lbsrv_link_state_notif_handle = glink_register_link_state_cb(
&glink_lbsrv_link_info, NULL);
return 0;
}
module_init(glink_loopback_server_init);
MODULE_DESCRIPTION("MSM Generic Link (G-Link) Loopback Server");
MODULE_LICENSE("GPL v2");