Commit 5b208656 authored by Sjur Braendeland's avatar Sjur Braendeland Committed by David S. Miller

caif: Add reference counting to service layer

Changes:
o Added functions cfsrvl_get and cfsrvl_put.
o Added support release_client to use by socket and net device.
o Increase reference counting for in-flight packets from cfmuxl
Signed-off-by: default avatarSjur Braendeland <sjur.brandeland@stericsson.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent e539d83c
...@@ -69,6 +69,17 @@ int caif_connect_client(struct caif_connect_request *config, ...@@ -69,6 +69,17 @@ int caif_connect_client(struct caif_connect_request *config,
*/ */
int caif_disconnect_client(struct cflayer *client_layer); int caif_disconnect_client(struct cflayer *client_layer);
/**
* caif_release_client - Release adaptation layer reference to client.
*
* @client_layer: Client layer.
*
* Releases a client/adaptation layer use of the caif stack.
* This function must be used after caif_disconnect_client to
* decrease the reference count of the service layer.
*/
void caif_release_client(struct cflayer *client_layer);
/** /**
* connect_req_to_link_param - Translate configuration parameters * connect_req_to_link_param - Translate configuration parameters
* from socket format to internal format. * from socket format to internal format.
......
...@@ -96,6 +96,13 @@ int cfcnfg_del_phy_layer(struct cfcnfg *cnfg, struct cflayer *phy_layer); ...@@ -96,6 +96,13 @@ int cfcnfg_del_phy_layer(struct cfcnfg *cnfg, struct cflayer *phy_layer);
int cfcnfg_disconn_adapt_layer(struct cfcnfg *cnfg, int cfcnfg_disconn_adapt_layer(struct cfcnfg *cnfg,
struct cflayer *adap_layer); struct cflayer *adap_layer);
/**
* cfcnfg_release_adap_layer - Used by client to release the adaptation layer.
*
* @adap_layer: Adaptation layer.
*/
void cfcnfg_release_adap_layer(struct cflayer *adap_layer);
/** /**
* cfcnfg_add_adaptation_layer - Add an adaptation layer to the CAIF stack. * cfcnfg_add_adaptation_layer - Add an adaptation layer to the CAIF stack.
* *
......
...@@ -9,14 +9,18 @@ ...@@ -9,14 +9,18 @@
#include <linux/list.h> #include <linux/list.h>
#include <linux/stddef.h> #include <linux/stddef.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/kref.h>
struct cfsrvl { struct cfsrvl {
struct cflayer layer; struct cflayer layer;
bool open; bool open;
bool phy_flow_on; bool phy_flow_on;
bool modem_flow_on; bool modem_flow_on;
struct dev_info dev_info; struct dev_info dev_info;
struct kref ref;
}; };
void cfsrvl_release(struct kref *kref);
struct cflayer *cfvei_create(u8 linkid, struct dev_info *dev_info); struct cflayer *cfvei_create(u8 linkid, struct dev_info *dev_info);
struct cflayer *cfdgml_create(u8 linkid, struct dev_info *dev_info); struct cflayer *cfdgml_create(u8 linkid, struct dev_info *dev_info);
struct cflayer *cfutill_create(u8 linkid, struct dev_info *dev_info); struct cflayer *cfutill_create(u8 linkid, struct dev_info *dev_info);
...@@ -31,4 +35,22 @@ void cfsrvl_init(struct cfsrvl *service, ...@@ -31,4 +35,22 @@ void cfsrvl_init(struct cfsrvl *service,
bool cfsrvl_ready(struct cfsrvl *service, int *err); bool cfsrvl_ready(struct cfsrvl *service, int *err);
u8 cfsrvl_getphyid(struct cflayer *layer); u8 cfsrvl_getphyid(struct cflayer *layer);
static inline void cfsrvl_get(struct cflayer *layr)
{
struct cfsrvl *s;
if (layr == NULL)
return;
s = container_of(layr, struct cfsrvl, layer);
kref_get(&s->ref);
}
static inline void cfsrvl_put(struct cflayer *layr)
{
struct cfsrvl *s;
if (layr == NULL)
return;
s = container_of(layr, struct cfsrvl, layer);
kref_put(&s->ref, cfsrvl_release);
}
#endif /* CFSRVL_H_ */ #endif /* CFSRVL_H_ */
...@@ -346,6 +346,12 @@ int caif_disconnect_client(struct cflayer *adap_layer) ...@@ -346,6 +346,12 @@ int caif_disconnect_client(struct cflayer *adap_layer)
} }
EXPORT_SYMBOL(caif_disconnect_client); EXPORT_SYMBOL(caif_disconnect_client);
void caif_release_client(struct cflayer *adap_layer)
{
cfcnfg_release_adap_layer(adap_layer);
}
EXPORT_SYMBOL(caif_release_client);
/* Per-namespace Caif devices handling */ /* Per-namespace Caif devices handling */
static int caif_init_net(struct net *net) static int caif_init_net(struct net *net)
{ {
......
...@@ -247,6 +247,13 @@ int cfcnfg_disconn_adapt_layer(struct cfcnfg *cnfg, struct cflayer *adap_layer) ...@@ -247,6 +247,13 @@ int cfcnfg_disconn_adapt_layer(struct cfcnfg *cnfg, struct cflayer *adap_layer)
} }
EXPORT_SYMBOL(cfcnfg_disconn_adapt_layer); EXPORT_SYMBOL(cfcnfg_disconn_adapt_layer);
void cfcnfg_release_adap_layer(struct cflayer *adap_layer)
{
if (adap_layer->dn)
cfsrvl_put(adap_layer->dn);
}
EXPORT_SYMBOL(cfcnfg_release_adap_layer);
static void cfcnfg_linkdestroy_rsp(struct cflayer *layer, u8 channel_id, static void cfcnfg_linkdestroy_rsp(struct cflayer *layer, u8 channel_id,
struct cflayer *client_layer) struct cflayer *client_layer)
{ {
......
...@@ -62,6 +62,7 @@ int cfmuxl_set_uplayer(struct cflayer *layr, struct cflayer *up, u8 linkid) ...@@ -62,6 +62,7 @@ int cfmuxl_set_uplayer(struct cflayer *layr, struct cflayer *up, u8 linkid)
{ {
struct cfmuxl *muxl = container_obj(layr); struct cfmuxl *muxl = container_obj(layr);
spin_lock(&muxl->receive_lock); spin_lock(&muxl->receive_lock);
cfsrvl_get(up);
list_add(&up->node, &muxl->srvl_list); list_add(&up->node, &muxl->srvl_list);
spin_unlock(&muxl->receive_lock); spin_unlock(&muxl->receive_lock);
return 0; return 0;
...@@ -172,8 +173,11 @@ struct cflayer *cfmuxl_remove_uplayer(struct cflayer *layr, u8 id) ...@@ -172,8 +173,11 @@ struct cflayer *cfmuxl_remove_uplayer(struct cflayer *layr, u8 id)
struct cfmuxl *muxl = container_obj(layr); struct cfmuxl *muxl = container_obj(layr);
spin_lock(&muxl->receive_lock); spin_lock(&muxl->receive_lock);
up = get_up(muxl, id); up = get_up(muxl, id);
if (up == NULL)
return NULL;
memset(muxl->up_cache, 0, sizeof(muxl->up_cache)); memset(muxl->up_cache, 0, sizeof(muxl->up_cache));
list_del(&up->node); list_del(&up->node);
cfsrvl_put(up);
spin_unlock(&muxl->receive_lock); spin_unlock(&muxl->receive_lock);
return up; return up;
} }
...@@ -203,8 +207,9 @@ static int cfmuxl_receive(struct cflayer *layr, struct cfpkt *pkt) ...@@ -203,8 +207,9 @@ static int cfmuxl_receive(struct cflayer *layr, struct cfpkt *pkt)
*/ */
return /* CFGLU_EPROT; */ 0; return /* CFGLU_EPROT; */ 0;
} }
cfsrvl_get(up);
ret = up->receive(up, pkt); ret = up->receive(up, pkt);
cfsrvl_put(up);
return ret; return ret;
} }
......
...@@ -158,6 +158,13 @@ void cfsrvl_init(struct cfsrvl *service, ...@@ -158,6 +158,13 @@ void cfsrvl_init(struct cfsrvl *service,
service->layer.ctrlcmd = cfservl_ctrlcmd; service->layer.ctrlcmd = cfservl_ctrlcmd;
service->layer.modemcmd = cfservl_modemcmd; service->layer.modemcmd = cfservl_modemcmd;
service->dev_info = *dev_info; service->dev_info = *dev_info;
kref_init(&service->ref);
}
void cfsrvl_release(struct kref *kref)
{
struct cfsrvl *service = container_of(kref, struct cfsrvl, ref);
kfree(service);
} }
bool cfsrvl_ready(struct cfsrvl *service, int *err) bool cfsrvl_ready(struct cfsrvl *service, int *err)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment