caif: Handle dev_queue_xmit errors.
Do proper handling of dev_queue_xmit errors in order to avoid double free of skb and leaks in error conditions. In cfctrl pending requests are removed when CAIF Link layer goes down. Signed-off-by: Sjur Brændeland <sjur.brandeland@stericsson.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
bee925db9a
commit
c85c2951d4
7 changed files with 119 additions and 47 deletions
|
@ -124,6 +124,7 @@ int cfctrl_linkdown_req(struct cflayer *cfctrl, u8 linkid,
|
||||||
|
|
||||||
struct cflayer *cfctrl_create(void);
|
struct cflayer *cfctrl_create(void);
|
||||||
struct cfctrl_rsp *cfctrl_get_respfuncs(struct cflayer *layer);
|
struct cfctrl_rsp *cfctrl_get_respfuncs(struct cflayer *layer);
|
||||||
void cfctrl_cancel_req(struct cflayer *layr, struct cflayer *adap_layer);
|
int cfctrl_cancel_req(struct cflayer *layr, struct cflayer *adap_layer);
|
||||||
|
void cfctrl_remove(struct cflayer *layr);
|
||||||
|
|
||||||
#endif /* CFCTRL_H_ */
|
#endif /* CFCTRL_H_ */
|
||||||
|
|
|
@ -118,6 +118,7 @@ static struct caif_device_entry *caif_get(struct net_device *dev)
|
||||||
|
|
||||||
static int transmit(struct cflayer *layer, struct cfpkt *pkt)
|
static int transmit(struct cflayer *layer, struct cfpkt *pkt)
|
||||||
{
|
{
|
||||||
|
int err;
|
||||||
struct caif_device_entry *caifd =
|
struct caif_device_entry *caifd =
|
||||||
container_of(layer, struct caif_device_entry, layer);
|
container_of(layer, struct caif_device_entry, layer);
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
|
@ -125,9 +126,11 @@ static int transmit(struct cflayer *layer, struct cfpkt *pkt)
|
||||||
skb = cfpkt_tonative(pkt);
|
skb = cfpkt_tonative(pkt);
|
||||||
skb->dev = caifd->netdev;
|
skb->dev = caifd->netdev;
|
||||||
|
|
||||||
dev_queue_xmit(skb);
|
err = dev_queue_xmit(skb);
|
||||||
|
if (err > 0)
|
||||||
|
err = -EIO;
|
||||||
|
|
||||||
return 0;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -604,7 +604,9 @@ static int caif_seqpkt_sendmsg(struct kiocb *kiocb, struct socket *sock,
|
||||||
goto err;
|
goto err;
|
||||||
ret = transmit_skb(skb, cf_sk, noblock, timeo);
|
ret = transmit_skb(skb, cf_sk, noblock, timeo);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto err;
|
/* skb is already freed */
|
||||||
|
return ret;
|
||||||
|
|
||||||
return len;
|
return len;
|
||||||
err:
|
err:
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
|
@ -933,9 +935,9 @@ static int caif_release(struct socket *sock)
|
||||||
* caif_queue_rcv_skb checks SOCK_DEAD holding the queue lock,
|
* caif_queue_rcv_skb checks SOCK_DEAD holding the queue lock,
|
||||||
* this ensures no packets when sock is dead.
|
* this ensures no packets when sock is dead.
|
||||||
*/
|
*/
|
||||||
spin_lock(&sk->sk_receive_queue.lock);
|
spin_lock_bh(&sk->sk_receive_queue.lock);
|
||||||
sock_set_flag(sk, SOCK_DEAD);
|
sock_set_flag(sk, SOCK_DEAD);
|
||||||
spin_unlock(&sk->sk_receive_queue.lock);
|
spin_unlock_bh(&sk->sk_receive_queue.lock);
|
||||||
sock->sk = NULL;
|
sock->sk = NULL;
|
||||||
|
|
||||||
dbfs_atomic_inc(&cnt.num_disconnect);
|
dbfs_atomic_inc(&cnt.num_disconnect);
|
||||||
|
|
|
@ -126,7 +126,7 @@ void cfcnfg_remove(struct cfcnfg *cfg)
|
||||||
synchronize_rcu();
|
synchronize_rcu();
|
||||||
|
|
||||||
kfree(cfg->mux);
|
kfree(cfg->mux);
|
||||||
kfree(cfg->ctrl);
|
cfctrl_remove(cfg->ctrl);
|
||||||
kfree(cfg);
|
kfree(cfg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
#define UTILITY_NAME_LENGTH 16
|
#define UTILITY_NAME_LENGTH 16
|
||||||
#define CFPKT_CTRL_PKT_LEN 20
|
#define CFPKT_CTRL_PKT_LEN 20
|
||||||
|
|
||||||
|
|
||||||
#ifdef CAIF_NO_LOOP
|
#ifdef CAIF_NO_LOOP
|
||||||
static int handle_loop(struct cfctrl *ctrl,
|
static int handle_loop(struct cfctrl *ctrl,
|
||||||
int cmd, struct cfpkt *pkt){
|
int cmd, struct cfpkt *pkt){
|
||||||
|
@ -51,13 +50,29 @@ struct cflayer *cfctrl_create(void)
|
||||||
this->serv.layer.receive = cfctrl_recv;
|
this->serv.layer.receive = cfctrl_recv;
|
||||||
sprintf(this->serv.layer.name, "ctrl");
|
sprintf(this->serv.layer.name, "ctrl");
|
||||||
this->serv.layer.ctrlcmd = cfctrl_ctrlcmd;
|
this->serv.layer.ctrlcmd = cfctrl_ctrlcmd;
|
||||||
|
#ifndef CAIF_NO_LOOP
|
||||||
spin_lock_init(&this->loop_linkid_lock);
|
spin_lock_init(&this->loop_linkid_lock);
|
||||||
|
this->loop_linkid = 1;
|
||||||
|
#endif
|
||||||
spin_lock_init(&this->info_list_lock);
|
spin_lock_init(&this->info_list_lock);
|
||||||
INIT_LIST_HEAD(&this->list);
|
INIT_LIST_HEAD(&this->list);
|
||||||
this->loop_linkid = 1;
|
|
||||||
return &this->serv.layer;
|
return &this->serv.layer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cfctrl_remove(struct cflayer *layer)
|
||||||
|
{
|
||||||
|
struct cfctrl_request_info *p, *tmp;
|
||||||
|
struct cfctrl *ctrl = container_obj(layer);
|
||||||
|
|
||||||
|
spin_lock_bh(&ctrl->info_list_lock);
|
||||||
|
list_for_each_entry_safe(p, tmp, &ctrl->list, list) {
|
||||||
|
list_del(&p->list);
|
||||||
|
kfree(p);
|
||||||
|
}
|
||||||
|
spin_unlock_bh(&ctrl->info_list_lock);
|
||||||
|
kfree(layer);
|
||||||
|
}
|
||||||
|
|
||||||
static bool param_eq(const struct cfctrl_link_param *p1,
|
static bool param_eq(const struct cfctrl_link_param *p1,
|
||||||
const struct cfctrl_link_param *p2)
|
const struct cfctrl_link_param *p2)
|
||||||
{
|
{
|
||||||
|
@ -116,11 +131,11 @@ static bool cfctrl_req_eq(const struct cfctrl_request_info *r1,
|
||||||
static void cfctrl_insert_req(struct cfctrl *ctrl,
|
static void cfctrl_insert_req(struct cfctrl *ctrl,
|
||||||
struct cfctrl_request_info *req)
|
struct cfctrl_request_info *req)
|
||||||
{
|
{
|
||||||
spin_lock(&ctrl->info_list_lock);
|
spin_lock_bh(&ctrl->info_list_lock);
|
||||||
atomic_inc(&ctrl->req_seq_no);
|
atomic_inc(&ctrl->req_seq_no);
|
||||||
req->sequence_no = atomic_read(&ctrl->req_seq_no);
|
req->sequence_no = atomic_read(&ctrl->req_seq_no);
|
||||||
list_add_tail(&req->list, &ctrl->list);
|
list_add_tail(&req->list, &ctrl->list);
|
||||||
spin_unlock(&ctrl->info_list_lock);
|
spin_unlock_bh(&ctrl->info_list_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Compare and remove request */
|
/* Compare and remove request */
|
||||||
|
@ -129,7 +144,6 @@ static struct cfctrl_request_info *cfctrl_remove_req(struct cfctrl *ctrl,
|
||||||
{
|
{
|
||||||
struct cfctrl_request_info *p, *tmp, *first;
|
struct cfctrl_request_info *p, *tmp, *first;
|
||||||
|
|
||||||
spin_lock(&ctrl->info_list_lock);
|
|
||||||
first = list_first_entry(&ctrl->list, struct cfctrl_request_info, list);
|
first = list_first_entry(&ctrl->list, struct cfctrl_request_info, list);
|
||||||
|
|
||||||
list_for_each_entry_safe(p, tmp, &ctrl->list, list) {
|
list_for_each_entry_safe(p, tmp, &ctrl->list, list) {
|
||||||
|
@ -145,7 +159,6 @@ static struct cfctrl_request_info *cfctrl_remove_req(struct cfctrl *ctrl,
|
||||||
}
|
}
|
||||||
p = NULL;
|
p = NULL;
|
||||||
out:
|
out:
|
||||||
spin_unlock(&ctrl->info_list_lock);
|
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -179,10 +192,6 @@ void cfctrl_enum_req(struct cflayer *layer, u8 physlinkid)
|
||||||
cfpkt_addbdy(pkt, physlinkid);
|
cfpkt_addbdy(pkt, physlinkid);
|
||||||
ret =
|
ret =
|
||||||
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
||||||
if (ret < 0) {
|
|
||||||
pr_err("Could not transmit enum message\n");
|
|
||||||
cfpkt_destroy(pkt);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int cfctrl_linkup_request(struct cflayer *layer,
|
int cfctrl_linkup_request(struct cflayer *layer,
|
||||||
|
@ -196,14 +205,23 @@ int cfctrl_linkup_request(struct cflayer *layer,
|
||||||
struct cfctrl_request_info *req;
|
struct cfctrl_request_info *req;
|
||||||
int ret;
|
int ret;
|
||||||
char utility_name[16];
|
char utility_name[16];
|
||||||
struct cfpkt *pkt = cfpkt_create(CFPKT_CTRL_PKT_LEN);
|
struct cfpkt *pkt;
|
||||||
|
|
||||||
|
if (cfctrl_cancel_req(layer, user_layer) > 0) {
|
||||||
|
/* Slight Paranoia, check if already connecting */
|
||||||
|
pr_err("Duplicate connect request for same client\n");
|
||||||
|
WARN_ON(1);
|
||||||
|
return -EALREADY;
|
||||||
|
}
|
||||||
|
|
||||||
|
pkt = cfpkt_create(CFPKT_CTRL_PKT_LEN);
|
||||||
if (!pkt) {
|
if (!pkt) {
|
||||||
pr_warn("Out of memory\n");
|
pr_warn("Out of memory\n");
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
cfpkt_addbdy(pkt, CFCTRL_CMD_LINK_SETUP);
|
cfpkt_addbdy(pkt, CFCTRL_CMD_LINK_SETUP);
|
||||||
cfpkt_addbdy(pkt, (param->chtype << 4) + param->linktype);
|
cfpkt_addbdy(pkt, (param->chtype << 4) | param->linktype);
|
||||||
cfpkt_addbdy(pkt, (param->priority << 3) + param->phyid);
|
cfpkt_addbdy(pkt, (param->priority << 3) | param->phyid);
|
||||||
cfpkt_addbdy(pkt, param->endpoint & 0x03);
|
cfpkt_addbdy(pkt, param->endpoint & 0x03);
|
||||||
|
|
||||||
switch (param->linktype) {
|
switch (param->linktype) {
|
||||||
|
@ -266,9 +284,13 @@ int cfctrl_linkup_request(struct cflayer *layer,
|
||||||
ret =
|
ret =
|
||||||
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
pr_err("Could not transmit linksetup request\n");
|
int count;
|
||||||
cfpkt_destroy(pkt);
|
|
||||||
return -ENODEV;
|
count = cfctrl_cancel_req(&cfctrl->serv.layer,
|
||||||
|
user_layer);
|
||||||
|
if (count != 1)
|
||||||
|
pr_err("Could not remove request (%d)", count);
|
||||||
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -288,28 +310,29 @@ int cfctrl_linkdown_req(struct cflayer *layer, u8 channelid,
|
||||||
init_info(cfpkt_info(pkt), cfctrl);
|
init_info(cfpkt_info(pkt), cfctrl);
|
||||||
ret =
|
ret =
|
||||||
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
cfctrl->serv.layer.dn->transmit(cfctrl->serv.layer.dn, pkt);
|
||||||
if (ret < 0) {
|
#ifndef CAIF_NO_LOOP
|
||||||
pr_err("Could not transmit link-down request\n");
|
cfctrl->loop_linkused[channelid] = 0;
|
||||||
cfpkt_destroy(pkt);
|
#endif
|
||||||
}
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cfctrl_cancel_req(struct cflayer *layr, struct cflayer *adap_layer)
|
int cfctrl_cancel_req(struct cflayer *layr, struct cflayer *adap_layer)
|
||||||
{
|
{
|
||||||
struct cfctrl_request_info *p, *tmp;
|
struct cfctrl_request_info *p, *tmp;
|
||||||
struct cfctrl *ctrl = container_obj(layr);
|
struct cfctrl *ctrl = container_obj(layr);
|
||||||
spin_lock(&ctrl->info_list_lock);
|
int found = 0;
|
||||||
|
spin_lock_bh(&ctrl->info_list_lock);
|
||||||
|
|
||||||
list_for_each_entry_safe(p, tmp, &ctrl->list, list) {
|
list_for_each_entry_safe(p, tmp, &ctrl->list, list) {
|
||||||
if (p->client_layer == adap_layer) {
|
if (p->client_layer == adap_layer) {
|
||||||
pr_debug("cancel req :%d\n", p->sequence_no);
|
|
||||||
list_del(&p->list);
|
list_del(&p->list);
|
||||||
kfree(p);
|
kfree(p);
|
||||||
|
found++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_unlock(&ctrl->info_list_lock);
|
spin_unlock_bh(&ctrl->info_list_lock);
|
||||||
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int cfctrl_recv(struct cflayer *layer, struct cfpkt *pkt)
|
static int cfctrl_recv(struct cflayer *layer, struct cfpkt *pkt)
|
||||||
|
@ -461,6 +484,7 @@ static int cfctrl_recv(struct cflayer *layer, struct cfpkt *pkt)
|
||||||
|
|
||||||
rsp.cmd = cmd;
|
rsp.cmd = cmd;
|
||||||
rsp.param = linkparam;
|
rsp.param = linkparam;
|
||||||
|
spin_lock_bh(&cfctrl->info_list_lock);
|
||||||
req = cfctrl_remove_req(cfctrl, &rsp);
|
req = cfctrl_remove_req(cfctrl, &rsp);
|
||||||
|
|
||||||
if (CFCTRL_ERR_BIT == (CFCTRL_ERR_BIT & cmdrsp) ||
|
if (CFCTRL_ERR_BIT == (CFCTRL_ERR_BIT & cmdrsp) ||
|
||||||
|
@ -480,6 +504,8 @@ static int cfctrl_recv(struct cflayer *layer, struct cfpkt *pkt)
|
||||||
|
|
||||||
if (req != NULL)
|
if (req != NULL)
|
||||||
kfree(req);
|
kfree(req);
|
||||||
|
|
||||||
|
spin_unlock_bh(&cfctrl->info_list_lock);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case CFCTRL_CMD_LINK_DESTROY:
|
case CFCTRL_CMD_LINK_DESTROY:
|
||||||
|
@ -523,12 +549,29 @@ static void cfctrl_ctrlcmd(struct cflayer *layr, enum caif_ctrlcmd ctrl,
|
||||||
switch (ctrl) {
|
switch (ctrl) {
|
||||||
case _CAIF_CTRLCMD_PHYIF_FLOW_OFF_IND:
|
case _CAIF_CTRLCMD_PHYIF_FLOW_OFF_IND:
|
||||||
case CAIF_CTRLCMD_FLOW_OFF_IND:
|
case CAIF_CTRLCMD_FLOW_OFF_IND:
|
||||||
spin_lock(&this->info_list_lock);
|
spin_lock_bh(&this->info_list_lock);
|
||||||
if (!list_empty(&this->list)) {
|
if (!list_empty(&this->list)) {
|
||||||
pr_debug("Received flow off in control layer\n");
|
pr_debug("Received flow off in control layer\n");
|
||||||
}
|
}
|
||||||
spin_unlock(&this->info_list_lock);
|
spin_unlock_bh(&this->info_list_lock);
|
||||||
break;
|
break;
|
||||||
|
case _CAIF_CTRLCMD_PHYIF_DOWN_IND: {
|
||||||
|
struct cfctrl_request_info *p, *tmp;
|
||||||
|
|
||||||
|
/* Find all connect request and report failure */
|
||||||
|
spin_lock_bh(&this->info_list_lock);
|
||||||
|
list_for_each_entry_safe(p, tmp, &this->list, list) {
|
||||||
|
if (p->param.phyid == phyid) {
|
||||||
|
list_del(&p->list);
|
||||||
|
p->client_layer->ctrlcmd(p->client_layer,
|
||||||
|
CAIF_CTRLCMD_INIT_FAIL_RSP,
|
||||||
|
phyid);
|
||||||
|
kfree(p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
spin_unlock_bh(&this->info_list_lock);
|
||||||
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -538,27 +581,33 @@ static void cfctrl_ctrlcmd(struct cflayer *layr, enum caif_ctrlcmd ctrl,
|
||||||
static int handle_loop(struct cfctrl *ctrl, int cmd, struct cfpkt *pkt)
|
static int handle_loop(struct cfctrl *ctrl, int cmd, struct cfpkt *pkt)
|
||||||
{
|
{
|
||||||
static int last_linkid;
|
static int last_linkid;
|
||||||
|
static int dec;
|
||||||
u8 linkid, linktype, tmp;
|
u8 linkid, linktype, tmp;
|
||||||
switch (cmd) {
|
switch (cmd) {
|
||||||
case CFCTRL_CMD_LINK_SETUP:
|
case CFCTRL_CMD_LINK_SETUP:
|
||||||
spin_lock(&ctrl->loop_linkid_lock);
|
spin_lock_bh(&ctrl->loop_linkid_lock);
|
||||||
for (linkid = last_linkid + 1; linkid < 255; linkid++)
|
if (!dec) {
|
||||||
if (!ctrl->loop_linkused[linkid])
|
for (linkid = last_linkid + 1; linkid < 255; linkid++)
|
||||||
goto found;
|
if (!ctrl->loop_linkused[linkid])
|
||||||
|
goto found;
|
||||||
|
}
|
||||||
|
dec = 1;
|
||||||
for (linkid = last_linkid - 1; linkid > 0; linkid--)
|
for (linkid = last_linkid - 1; linkid > 0; linkid--)
|
||||||
if (!ctrl->loop_linkused[linkid])
|
if (!ctrl->loop_linkused[linkid])
|
||||||
goto found;
|
goto found;
|
||||||
spin_unlock(&ctrl->loop_linkid_lock);
|
spin_unlock_bh(&ctrl->loop_linkid_lock);
|
||||||
pr_err("Out of link-ids\n");
|
|
||||||
return -EINVAL;
|
|
||||||
found:
|
found:
|
||||||
|
if (linkid < 10)
|
||||||
|
dec = 0;
|
||||||
|
|
||||||
if (!ctrl->loop_linkused[linkid])
|
if (!ctrl->loop_linkused[linkid])
|
||||||
ctrl->loop_linkused[linkid] = 1;
|
ctrl->loop_linkused[linkid] = 1;
|
||||||
|
|
||||||
last_linkid = linkid;
|
last_linkid = linkid;
|
||||||
|
|
||||||
cfpkt_add_trail(pkt, &linkid, 1);
|
cfpkt_add_trail(pkt, &linkid, 1);
|
||||||
spin_unlock(&ctrl->loop_linkid_lock);
|
spin_unlock_bh(&ctrl->loop_linkid_lock);
|
||||||
cfpkt_peek_head(pkt, &linktype, 1);
|
cfpkt_peek_head(pkt, &linktype, 1);
|
||||||
if (linktype == CFCTRL_SRV_UTIL) {
|
if (linktype == CFCTRL_SRV_UTIL) {
|
||||||
tmp = 0x01;
|
tmp = 0x01;
|
||||||
|
@ -568,10 +617,10 @@ static int handle_loop(struct cfctrl *ctrl, int cmd, struct cfpkt *pkt)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CFCTRL_CMD_LINK_DESTROY:
|
case CFCTRL_CMD_LINK_DESTROY:
|
||||||
spin_lock(&ctrl->loop_linkid_lock);
|
spin_lock_bh(&ctrl->loop_linkid_lock);
|
||||||
cfpkt_peek_head(pkt, &linkid, 1);
|
cfpkt_peek_head(pkt, &linkid, 1);
|
||||||
ctrl->loop_linkused[linkid] = 0;
|
ctrl->loop_linkused[linkid] = 0;
|
||||||
spin_unlock(&ctrl->loop_linkid_lock);
|
spin_unlock_bh(&ctrl->loop_linkid_lock);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -33,7 +33,6 @@ static void cffrml_ctrlcmd(struct cflayer *layr, enum caif_ctrlcmd ctrl,
|
||||||
static u32 cffrml_rcv_error;
|
static u32 cffrml_rcv_error;
|
||||||
static u32 cffrml_rcv_checsum_error;
|
static u32 cffrml_rcv_checsum_error;
|
||||||
struct cflayer *cffrml_create(u16 phyid, bool use_fcs)
|
struct cflayer *cffrml_create(u16 phyid, bool use_fcs)
|
||||||
|
|
||||||
{
|
{
|
||||||
struct cffrml *this = kmalloc(sizeof(struct cffrml), GFP_ATOMIC);
|
struct cffrml *this = kmalloc(sizeof(struct cffrml), GFP_ATOMIC);
|
||||||
if (!this) {
|
if (!this) {
|
||||||
|
@ -128,6 +127,13 @@ static int cffrml_receive(struct cflayer *layr, struct cfpkt *pkt)
|
||||||
cfpkt_destroy(pkt);
|
cfpkt_destroy(pkt);
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (layr->up == NULL) {
|
||||||
|
pr_err("Layr up is missing!\n");
|
||||||
|
cfpkt_destroy(pkt);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
return layr->up->receive(layr->up, pkt);
|
return layr->up->receive(layr->up, pkt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -150,15 +156,22 @@ static int cffrml_transmit(struct cflayer *layr, struct cfpkt *pkt)
|
||||||
cfpkt_info(pkt)->hdr_len += 2;
|
cfpkt_info(pkt)->hdr_len += 2;
|
||||||
if (cfpkt_erroneous(pkt)) {
|
if (cfpkt_erroneous(pkt)) {
|
||||||
pr_err("Packet is erroneous!\n");
|
pr_err("Packet is erroneous!\n");
|
||||||
|
cfpkt_destroy(pkt);
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (layr->dn == NULL) {
|
||||||
|
cfpkt_destroy(pkt);
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
}
|
||||||
return layr->dn->transmit(layr->dn, pkt);
|
return layr->dn->transmit(layr->dn, pkt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cffrml_ctrlcmd(struct cflayer *layr, enum caif_ctrlcmd ctrl,
|
static void cffrml_ctrlcmd(struct cflayer *layr, enum caif_ctrlcmd ctrl,
|
||||||
int phyid)
|
int phyid)
|
||||||
{
|
{
|
||||||
if (layr->up->ctrlcmd)
|
if (layr->up && layr->up->ctrlcmd)
|
||||||
layr->up->ctrlcmd(layr->up, ctrl, layr->id);
|
layr->up->ctrlcmd(layr->up, ctrl, layr->id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -82,13 +82,14 @@ static int cfvei_transmit(struct cflayer *layr, struct cfpkt *pkt)
|
||||||
int ret;
|
int ret;
|
||||||
struct cfsrvl *service = container_obj(layr);
|
struct cfsrvl *service = container_obj(layr);
|
||||||
if (!cfsrvl_ready(service, &ret))
|
if (!cfsrvl_ready(service, &ret))
|
||||||
return ret;
|
goto err;
|
||||||
caif_assert(layr->dn != NULL);
|
caif_assert(layr->dn != NULL);
|
||||||
caif_assert(layr->dn->transmit != NULL);
|
caif_assert(layr->dn->transmit != NULL);
|
||||||
|
|
||||||
if (cfpkt_add_head(pkt, &tmp, 1) < 0) {
|
if (cfpkt_add_head(pkt, &tmp, 1) < 0) {
|
||||||
pr_err("Packet is erroneous!\n");
|
pr_err("Packet is erroneous!\n");
|
||||||
return -EPROTO;
|
ret = -EPROTO;
|
||||||
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Add info-> for MUX-layer to route the packet out. */
|
/* Add info-> for MUX-layer to route the packet out. */
|
||||||
|
@ -97,4 +98,7 @@ static int cfvei_transmit(struct cflayer *layr, struct cfpkt *pkt)
|
||||||
info->hdr_len = 1;
|
info->hdr_len = 1;
|
||||||
info->dev_info = &service->dev_info;
|
info->dev_info = &service->dev_info;
|
||||||
return layr->dn->transmit(layr->dn, pkt);
|
return layr->dn->transmit(layr->dn, pkt);
|
||||||
|
err:
|
||||||
|
cfpkt_destroy(pkt);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue